/* GtkPrinter * Copyright (C) 2006 John (J5) Palmieri * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see . */ #include "config.h" #include #include #include #include "gtkintl.h" #include "gtkprivate.h" #include "gtkprinter.h" #include "gtkprinter-private.h" #include "gtkprintbackend.h" #include "gtkprintjob.h" /** * SECTION:gtkprinter * @Short_description: Represents a printer * @Title: GtkPrinter * * A #GtkPrinter object represents a printer. You only need to * deal directly with printers if you use the non-portable * #GtkPrintUnixDialog API. * * A #GtkPrinter allows to get status information about the printer, * such as its description, its location, the number of queued jobs, * etc. Most importantly, a #GtkPrinter object can be used to create * a #GtkPrintJob object, which lets you print to the printer. * * Printing support was added in GTK+ 2.10. */ static void gtk_printer_finalize (GObject *object); struct _GtkPrinterPrivate { gchar *name; gchar *location; gchar *description; gchar *icon_name; guint is_active : 1; guint is_paused : 1; guint is_accepting_jobs : 1; guint is_new : 1; guint is_virtual : 1; guint is_default : 1; guint has_details : 1; guint accepts_pdf : 1; guint accepts_ps : 1; gchar *state_message; gint job_count; GtkPrintBackend *backend; }; enum { DETAILS_ACQUIRED, LAST_SIGNAL }; enum { PROP_0, PROP_NAME, PROP_BACKEND, PROP_IS_VIRTUAL, PROP_STATE_MESSAGE, PROP_LOCATION, PROP_ICON_NAME, PROP_JOB_COUNT, PROP_ACCEPTS_PDF, PROP_ACCEPTS_PS, PROP_PAUSED, PROP_ACCEPTING_JOBS }; static guint signals[LAST_SIGNAL] = { 0 }; static void gtk_printer_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec); static void gtk_printer_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); G_DEFINE_TYPE_WITH_PRIVATE (GtkPrinter, gtk_printer, G_TYPE_OBJECT) static void gtk_printer_class_init (GtkPrinterClass *class) { GObjectClass *object_class; object_class = (GObjectClass *) class; object_class->finalize = gtk_printer_finalize; object_class->set_property = gtk_printer_set_property; object_class->get_property = gtk_printer_get_property; g_object_class_install_property (G_OBJECT_CLASS (class), PROP_NAME, g_param_spec_string ("name", P_("Name"), P_("Name of the printer"), "", GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); g_object_class_install_property (G_OBJECT_CLASS (class), PROP_BACKEND, g_param_spec_object ("backend", P_("Backend"), P_("Backend for the printer"), GTK_TYPE_PRINT_BACKEND, GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); g_object_class_install_property (G_OBJECT_CLASS (class), PROP_IS_VIRTUAL, g_param_spec_boolean ("is-virtual", P_("Is Virtual"), P_("FALSE if this represents a real hardware printer"), FALSE, GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); g_object_class_install_property (G_OBJECT_CLASS (class), PROP_ACCEPTS_PDF, g_param_spec_boolean ("accepts-pdf", P_("Accepts PDF"), P_("TRUE if this printer can accept PDF"), FALSE, GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); g_object_class_install_property (G_OBJECT_CLASS (class), PROP_ACCEPTS_PS, g_param_spec_boolean ("accepts-ps", P_("Accepts PostScript"), P_("TRUE if this printer can accept PostScript"), TRUE, GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); g_object_class_install_property (G_OBJECT_CLASS (class), PROP_STATE_MESSAGE, g_param_spec_string ("state-message", P_("State Message"), P_("String giving the current state of the printer"), "", GTK_PARAM_READABLE)); g_object_class_install_property (G_OBJECT_CLASS (class), PROP_LOCATION, g_param_spec_string ("location", P_("Location"), P_("The location of the printer"), "", GTK_PARAM_READABLE)); g_object_class_install_property (G_OBJECT_CLASS (class), PROP_ICON_NAME, g_param_spec_string ("icon-name", P_("Icon Name"), P_("The icon name to use for the printer"), "", GTK_PARAM_READABLE)); g_object_class_install_property (G_OBJECT_CLASS (class), PROP_JOB_COUNT, g_param_spec_int ("job-count", P_("Job Count"), P_("Number of jobs queued in the printer"), 0, G_MAXINT, 0, GTK_PARAM_READABLE)); /** * GtkPrinter:paused: * * This property is %TRUE if this printer is paused. * A paused printer still accepts jobs, but it does * not print them. * * Since: 2.14 */ g_object_class_install_property (G_OBJECT_CLASS (class), PROP_PAUSED, g_param_spec_boolean ("paused", P_("Paused Printer"), P_("TRUE if this printer is paused"), FALSE, GTK_PARAM_READABLE)); /** * GtkPrinter:accepting-jobs: * * This property is %TRUE if the printer is accepting jobs. * * Since: 2.14 */ g_object_class_install_property (G_OBJECT_CLASS (class), PROP_ACCEPTING_JOBS, g_param_spec_boolean ("accepting-jobs", P_("Accepting Jobs"), P_("TRUE if this printer is accepting new jobs"), TRUE, GTK_PARAM_READABLE)); /** * GtkPrinter::details-acquired: * @printer: the #GtkPrinter on which the signal is emitted * @success: %TRUE if the details were successfully acquired * * Gets emitted in response to a request for detailed information * about a printer from the print backend. The @success parameter * indicates if the information was actually obtained. * * Since: 2.10 */ signals[DETAILS_ACQUIRED] = g_signal_new (I_("details-acquired"), G_TYPE_FROM_CLASS (class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GtkPrinterClass, details_acquired), NULL, NULL, g_cclosure_marshal_VOID__BOOLEAN, G_TYPE_NONE, 1, G_TYPE_BOOLEAN); } static void gtk_printer_init (GtkPrinter *printer) { GtkPrinterPrivate *priv; priv = printer->priv = gtk_printer_get_instance_private (printer); priv->name = NULL; priv->location = NULL; priv->description = NULL; priv->icon_name = g_strdup ("printer"); priv->is_active = TRUE; priv->is_paused = FALSE; priv->is_accepting_jobs = TRUE; priv->is_new = TRUE; priv->has_details = FALSE; priv->accepts_pdf = FALSE; priv->accepts_ps = TRUE; priv->state_message = NULL; priv->job_count = 0; } static void gtk_printer_finalize (GObject *object) { GtkPrinter *printer = GTK_PRINTER (object); GtkPrinterPrivate *priv = printer->priv; g_free (priv->name); g_free (priv->location); g_free (priv->description); g_free (priv->state_message); g_free (priv->icon_name); if (priv->backend) g_object_unref (priv->backend); G_OBJECT_CLASS (gtk_printer_parent_class)->finalize (object); } static void gtk_printer_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { GtkPrinter *printer = GTK_PRINTER (object); GtkPrinterPrivate *priv = printer->priv; switch (prop_id) { case PROP_NAME: priv->name = g_value_dup_string (value); break; case PROP_BACKEND: priv->backend = GTK_PRINT_BACKEND (g_value_dup_object (value)); break; case PROP_IS_VIRTUAL: priv->is_virtual = g_value_get_boolean (value); break; case PROP_ACCEPTS_PDF: priv->accepts_pdf = g_value_get_boolean (value); break; case PROP_ACCEPTS_PS: priv->accepts_ps = g_value_get_boolean (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gtk_printer_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GtkPrinter *printer = GTK_PRINTER (object); GtkPrinterPrivate *priv = printer->priv; switch (prop_id) { case PROP_NAME: if (priv->name) g_value_set_string (value, priv->name); else g_value_set_static_string (value, ""); break; case PROP_BACKEND: g_value_set_object (value, priv->backend); break; case PROP_STATE_MESSAGE: if (priv->state_message) g_value_set_string (value, priv->state_message); else g_value_set_static_string (value, ""); break; case PROP_LOCATION: if (priv->location) g_value_set_string (value, priv->location); else g_value_set_static_string (value, ""); break; case PROP_ICON_NAME: if (priv->icon_name) g_value_set_string (value, priv->icon_name); else g_value_set_static_string (value, ""); break; case PROP_JOB_COUNT: g_value_set_int (value, priv->job_count); break; case PROP_IS_VIRTUAL: g_value_set_boolean (value, priv->is_virtual); break; case PROP_ACCEPTS_PDF: g_value_set_boolean (value, priv->accepts_pdf); break; case PROP_ACCEPTS_PS: g_value_set_boolean (value, priv->accepts_ps); break; case PROP_PAUSED: g_value_set_boolean (value, priv->is_paused); break; case PROP_ACCEPTING_JOBS: g_value_set_boolean (value, priv->is_accepting_jobs); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } /** * gtk_printer_new: * @name: the name of the printer * @backend: a #GtkPrintBackend * @virtual_: whether the printer is virtual * * Creates a new #GtkPrinter. * * Returns: a new #GtkPrinter * * Since: 2.10 **/ GtkPrinter * gtk_printer_new (const gchar *name, GtkPrintBackend *backend, gboolean virtual_) { GObject *result; result = g_object_new (GTK_TYPE_PRINTER, "name", name, "backend", backend, "is-virtual", virtual_, NULL); return (GtkPrinter *) result; } /** * gtk_printer_get_backend: * @printer: a #GtkPrinter * * Returns the backend of the printer. * * Returns: (transfer none): the backend of @printer * * Since: 2.10 */ GtkPrintBackend * gtk_printer_get_backend (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), NULL); return printer->priv->backend; } /** * gtk_printer_get_name: * @printer: a #GtkPrinter * * Returns the name of the printer. * * Returns: the name of @printer * * Since: 2.10 */ const gchar * gtk_printer_get_name (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), NULL); return printer->priv->name; } /** * gtk_printer_get_description: * @printer: a #GtkPrinter * * Gets the description of the printer. * * Returns: the description of @printer * * Since: 2.10 */ const gchar * gtk_printer_get_description (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), NULL); return printer->priv->description; } gboolean gtk_printer_set_description (GtkPrinter *printer, const gchar *description) { GtkPrinterPrivate *priv; g_return_val_if_fail (GTK_IS_PRINTER (printer), FALSE); priv = printer->priv; if (g_strcmp0 (priv->description, description) == 0) return FALSE; g_free (priv->description); priv->description = g_strdup (description); return TRUE; } /** * gtk_printer_get_state_message: * @printer: a #GtkPrinter * * Returns the state message describing the current state * of the printer. * * Returns: the state message of @printer * * Since: 2.10 */ const gchar * gtk_printer_get_state_message (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), NULL); return printer->priv->state_message; } gboolean gtk_printer_set_state_message (GtkPrinter *printer, const gchar *message) { GtkPrinterPrivate *priv; g_return_val_if_fail (GTK_IS_PRINTER (printer), FALSE); priv = printer->priv; if (g_strcmp0 (priv->state_message, message) == 0) return FALSE; g_free (priv->state_message); priv->state_message = g_strdup (message); g_object_notify (G_OBJECT (printer), "state-message"); return TRUE; } /** * gtk_printer_get_location: * @printer: a #GtkPrinter * * Returns a description of the location of the printer. * * Returns: the location of @printer * * Since: 2.10 */ const gchar * gtk_printer_get_location (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), NULL); return printer->priv->location; } gboolean gtk_printer_set_location (GtkPrinter *printer, const gchar *location) { GtkPrinterPrivate *priv; g_return_val_if_fail (GTK_IS_PRINTER (printer), FALSE); priv = printer->priv; if (g_strcmp0 (priv->location, location) == 0) return FALSE; g_free (priv->location); priv->location = g_strdup (location); g_object_notify (G_OBJECT (printer), "location"); return TRUE; } /** * gtk_printer_get_icon_name: * @printer: a #GtkPrinter * * Gets the name of the icon to use for the printer. * * Returns: the icon name for @printer * * Since: 2.10 */ const gchar * gtk_printer_get_icon_name (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), NULL); return printer->priv->icon_name; } void gtk_printer_set_icon_name (GtkPrinter *printer, const gchar *icon) { GtkPrinterPrivate *priv; g_return_if_fail (GTK_IS_PRINTER (printer)); priv = printer->priv; g_free (priv->icon_name); priv->icon_name = g_strdup (icon); g_object_notify (G_OBJECT (printer), "icon-name"); } /** * gtk_printer_get_job_count: * @printer: a #GtkPrinter * * Gets the number of jobs currently queued on the printer. * * Returns: the number of jobs on @printer * * Since: 2.10 */ gint gtk_printer_get_job_count (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), 0); return printer->priv->job_count; } gboolean gtk_printer_set_job_count (GtkPrinter *printer, gint count) { GtkPrinterPrivate *priv; g_return_val_if_fail (GTK_IS_PRINTER (printer), FALSE); priv = printer->priv; if (priv->job_count == count) return FALSE; priv->job_count = count; g_object_notify (G_OBJECT (printer), "job-count"); return TRUE; } /** * gtk_printer_has_details: * @printer: a #GtkPrinter * * Returns whether the printer details are available. * * Returns: %TRUE if @printer details are available * * Since: 2.12 */ gboolean gtk_printer_has_details (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), FALSE); return printer->priv->has_details; } void gtk_printer_set_has_details (GtkPrinter *printer, gboolean val) { printer->priv->has_details = val; } /** * gtk_printer_is_active: * @printer: a #GtkPrinter * * Returns whether the printer is currently active (i.e. * accepts new jobs). * * Returns: %TRUE if @printer is active * * Since: 2.10 */ gboolean gtk_printer_is_active (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), TRUE); return printer->priv->is_active; } void gtk_printer_set_is_active (GtkPrinter *printer, gboolean val) { g_return_if_fail (GTK_IS_PRINTER (printer)); printer->priv->is_active = val; } /** * gtk_printer_is_paused: * @printer: a #GtkPrinter * * Returns whether the printer is currently paused. * A paused printer still accepts jobs, but it is not * printing them. * * Returns: %TRUE if @printer is paused * * Since: 2.14 */ gboolean gtk_printer_is_paused (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), TRUE); return printer->priv->is_paused; } gboolean gtk_printer_set_is_paused (GtkPrinter *printer, gboolean val) { GtkPrinterPrivate *priv; g_return_val_if_fail (GTK_IS_PRINTER (printer), FALSE); priv = printer->priv; if (val == priv->is_paused) return FALSE; priv->is_paused = val; return TRUE; } /** * gtk_printer_is_accepting_jobs: * @printer: a #GtkPrinter * * Returns whether the printer is accepting jobs * * Returns: %TRUE if @printer is accepting jobs * * Since: 2.14 */ gboolean gtk_printer_is_accepting_jobs (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), TRUE); return printer->priv->is_accepting_jobs; } gboolean gtk_printer_set_is_accepting_jobs (GtkPrinter *printer, gboolean val) { GtkPrinterPrivate *priv; g_return_val_if_fail (GTK_IS_PRINTER (printer), FALSE); priv = printer->priv; if (val == priv->is_accepting_jobs) return FALSE; priv->is_accepting_jobs = val; return TRUE; } /** * gtk_printer_is_virtual: * @printer: a #GtkPrinter * * Returns whether the printer is virtual (i.e. does not * represent actual printer hardware, but something like * a CUPS class). * * Returns: %TRUE if @printer is virtual * * Since: 2.10 */ gboolean gtk_printer_is_virtual (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), TRUE); return printer->priv->is_virtual; } /** * gtk_printer_accepts_pdf: * @printer: a #GtkPrinter * * Returns whether the printer accepts input in * PDF format. * * Returns: %TRUE if @printer accepts PDF * * Since: 2.10 */ gboolean gtk_printer_accepts_pdf (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), TRUE); return printer->priv->accepts_pdf; } void gtk_printer_set_accepts_pdf (GtkPrinter *printer, gboolean val) { g_return_if_fail (GTK_IS_PRINTER (printer)); printer->priv->accepts_pdf = val; } /** * gtk_printer_accepts_ps: * @printer: a #GtkPrinter * * Returns whether the printer accepts input in * PostScript format. * * Returns: %TRUE if @printer accepts PostScript * * Since: 2.10 */ gboolean gtk_printer_accepts_ps (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), TRUE); return printer->priv->accepts_ps; } void gtk_printer_set_accepts_ps (GtkPrinter *printer, gboolean val) { g_return_if_fail (GTK_IS_PRINTER (printer)); printer->priv->accepts_ps = val; } gboolean gtk_printer_is_new (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), FALSE); return printer->priv->is_new; } void gtk_printer_set_is_new (GtkPrinter *printer, gboolean val) { g_return_if_fail (GTK_IS_PRINTER (printer)); printer->priv->is_new = val; } /** * gtk_printer_is_default: * @printer: a #GtkPrinter * * Returns whether the printer is the default printer. * * Returns: %TRUE if @printer is the default * * Since: 2.10 */ gboolean gtk_printer_is_default (GtkPrinter *printer) { g_return_val_if_fail (GTK_IS_PRINTER (printer), FALSE); return printer->priv->is_default; } void gtk_printer_set_is_default (GtkPrinter *printer, gboolean val) { g_return_if_fail (GTK_IS_PRINTER (printer)); printer->priv->is_default = val; } /** * gtk_printer_request_details: * @printer: a #GtkPrinter * * Requests the printer details. When the details are available, * the #GtkPrinter::details-acquired signal will be emitted on @printer. * * Since: 2.12 */ void gtk_printer_request_details (GtkPrinter *printer) { GtkPrintBackendClass *backend_class; g_return_if_fail (GTK_IS_PRINTER (printer)); backend_class = GTK_PRINT_BACKEND_GET_CLASS (printer->priv->backend); backend_class->printer_request_details (printer); } GtkPrinterOptionSet * _gtk_printer_get_options (GtkPrinter *printer, GtkPrintSettings *settings, GtkPageSetup *page_setup, GtkPrintCapabilities capabilities) { GtkPrintBackendClass *backend_class = GTK_PRINT_BACKEND_GET_CLASS (printer->priv->backend); return backend_class->printer_get_options (printer, settings, page_setup, capabilities); } gboolean _gtk_printer_mark_conflicts (GtkPrinter *printer, GtkPrinterOptionSet *options) { GtkPrintBackendClass *backend_class = GTK_PRINT_BACKEND_GET_CLASS (printer->priv->backend); return backend_class->printer_mark_conflicts (printer, options); } void _gtk_printer_get_settings_from_options (GtkPrinter *printer, GtkPrinterOptionSet *options, GtkPrintSettings *settings) { GtkPrintBackendClass *backend_class = GTK_PRINT_BACKEND_GET_CLASS (printer->priv->backend); backend_class->printer_get_settings_from_options (printer, options, settings); } void _gtk_printer_prepare_for_print (GtkPrinter *printer, GtkPrintJob *print_job, GtkPrintSettings *settings, GtkPageSetup *page_setup) { GtkPrintBackendClass *backend_class = GTK_PRINT_BACKEND_GET_CLASS (printer->priv->backend); backend_class->printer_prepare_for_print (printer, print_job, settings, page_setup); } cairo_surface_t * _gtk_printer_create_cairo_surface (GtkPrinter *printer, GtkPrintSettings *settings, gdouble width, gdouble height, GIOChannel *cache_io) { GtkPrintBackendClass *backend_class = GTK_PRINT_BACKEND_GET_CLASS (printer->priv->backend); return backend_class->printer_create_cairo_surface (printer, settings, width, height, cache_io); } /** * gtk_printer_list_papers: * @printer: a #GtkPrinter * * Lists all the paper sizes @printer supports. * This will return and empty list unless the printer’s details are * available, see gtk_printer_has_details() and gtk_printer_request_details(). * * Returns: (element-type GtkPageSetup) (transfer full): a newly allocated list of newly allocated #GtkPageSetup s. * * Since: 2.12 */ GList * gtk_printer_list_papers (GtkPrinter *printer) { GtkPrintBackendClass *backend_class; g_return_val_if_fail (GTK_IS_PRINTER (printer), NULL); backend_class = GTK_PRINT_BACKEND_GET_CLASS (printer->priv->backend); return backend_class->printer_list_papers (printer); } /** * gtk_printer_get_default_page_size: * @printer: a #GtkPrinter * * Returns default page size of @printer. * * Returns: a newly allocated #GtkPageSetup with default page size of the printer. * * Since: 2.14 */ GtkPageSetup * gtk_printer_get_default_page_size (GtkPrinter *printer) { GtkPrintBackendClass *backend_class; g_return_val_if_fail (GTK_IS_PRINTER (printer), NULL); backend_class = GTK_PRINT_BACKEND_GET_CLASS (printer->priv->backend); return backend_class->printer_get_default_page_size (printer); } /** * gtk_printer_get_hard_margins: * @printer: a #GtkPrinter * @top: (out): a location to store the top margin in * @bottom: (out): a location to store the bottom margin in * @left: (out): a location to store the left margin in * @right: (out): a location to store the right margin in * * Retrieve the hard margins of @printer, i.e. the margins that define * the area at the borders of the paper that the printer cannot print to. * * Note: This will not succeed unless the printer’s details are available, * see gtk_printer_has_details() and gtk_printer_request_details(). * * Returns: %TRUE iff the hard margins were retrieved * * Since: 2.20 */ gboolean gtk_printer_get_hard_margins (GtkPrinter *printer, gdouble *top, gdouble *bottom, gdouble *left, gdouble *right) { GtkPrintBackendClass *backend_class = GTK_PRINT_BACKEND_GET_CLASS (printer->priv->backend); return backend_class->printer_get_hard_margins (printer, top, bottom, left, right); } /** * gtk_printer_get_capabilities: * @printer: a #GtkPrinter * * Returns the printer’s capabilities. * * This is useful when you’re using #GtkPrintUnixDialog’s manual-capabilities * setting and need to know which settings the printer can handle and which * you must handle yourself. * * This will return 0 unless the printer’s details are available, see * gtk_printer_has_details() and gtk_printer_request_details(). * * Returns: the printer’s capabilities * * Since: 2.12 */ GtkPrintCapabilities gtk_printer_get_capabilities (GtkPrinter *printer) { GtkPrintBackendClass *backend_class; g_return_val_if_fail (GTK_IS_PRINTER (printer), 0); backend_class = GTK_PRINT_BACKEND_GET_CLASS (printer->priv->backend); return backend_class->printer_get_capabilities (printer); } /** * gtk_printer_compare: * @a: a #GtkPrinter * @b: another #GtkPrinter * * Compares two printers. * * Returns: 0 if the printer match, a negative value if @a < @b, * or a positive value if @a > @b * * Since: 2.10 */ gint gtk_printer_compare (GtkPrinter *a, GtkPrinter *b) { const char *name_a, *name_b; g_assert (GTK_IS_PRINTER (a) && GTK_IS_PRINTER (b)); name_a = gtk_printer_get_name (a); name_b = gtk_printer_get_name (b); if (name_a == NULL && name_b == NULL) return 0; else if (name_a == NULL) return G_MAXINT; else if (name_b == NULL) return G_MININT; else return g_ascii_strcasecmp (name_a, name_b); } typedef struct { GList *backends; GtkPrinterFunc func; gpointer data; GDestroyNotify destroy; GMainLoop *loop; } PrinterList; static void list_done_cb (GtkPrintBackend *backend, PrinterList *printer_list); static void stop_enumeration (PrinterList *printer_list) { GList *list, *next; GtkPrintBackend *backend; for (list = printer_list->backends; list; list = next) { next = list->next; backend = GTK_PRINT_BACKEND (list->data); list_done_cb (backend, printer_list); } } static void free_printer_list (PrinterList *printer_list) { if (printer_list->destroy) printer_list->destroy (printer_list->data); if (printer_list->loop) { g_main_loop_quit (printer_list->loop); g_main_loop_unref (printer_list->loop); } g_free (printer_list); } static gboolean list_added_cb (GtkPrintBackend *backend, GtkPrinter *printer, PrinterList *printer_list) { if (printer_list->func (printer, printer_list->data)) { stop_enumeration (printer_list); return TRUE; } return FALSE; } static void backend_status_changed (GObject *object, GParamSpec *pspec, gpointer data) { GtkPrintBackend *backend = GTK_PRINT_BACKEND (object); PrinterList *printer_list = data; GtkPrintBackendStatus status; g_object_get (backend, "status", &status, NULL); if (status == GTK_PRINT_BACKEND_STATUS_UNAVAILABLE) list_done_cb (backend, printer_list); } static gboolean list_printers_remove_backend (PrinterList *printer_list, GtkPrintBackend *backend) { printer_list->backends = g_list_remove (printer_list->backends, backend); gtk_print_backend_destroy (backend); g_object_unref (backend); if (printer_list->backends == NULL) { free_printer_list (printer_list); return TRUE; } return FALSE; } static void list_done_cb (GtkPrintBackend *backend, PrinterList *printer_list) { g_signal_handlers_disconnect_by_func (backend, list_added_cb, printer_list); g_signal_handlers_disconnect_by_func (backend, list_done_cb, printer_list); g_signal_handlers_disconnect_by_func (backend, backend_status_changed, printer_list); list_printers_remove_backend(printer_list, backend); } static gboolean list_printers_init (PrinterList *printer_list, GtkPrintBackend *backend) { GList *list, *node; GtkPrintBackendStatus status; list = gtk_print_backend_get_printer_list (backend); for (node = list; node != NULL; node = node->next) { if (list_added_cb (backend, node->data, printer_list)) { g_list_free (list); return TRUE; } } g_list_free (list); g_object_get (backend, "status", &status, NULL); if (status == GTK_PRINT_BACKEND_STATUS_UNAVAILABLE || gtk_print_backend_printer_list_is_done (backend)) { if (list_printers_remove_backend (printer_list, backend)) return TRUE; } else { g_signal_connect (backend, "printer-added", (GCallback) list_added_cb, printer_list); g_signal_connect (backend, "printer-list-done", (GCallback) list_done_cb, printer_list); g_signal_connect (backend, "notify::status", (GCallback) backend_status_changed, printer_list); } return FALSE; } /** * gtk_enumerate_printers: * @func: a function to call for each printer * @data: user data to pass to @func * @destroy: function to call if @data is no longer needed * @wait: if %TRUE, wait in a recursive mainloop until * all printers are enumerated; otherwise return early * * Calls a function for all #GtkPrinters. * If @func returns %TRUE, the enumeration is stopped. * * Since: 2.10 */ void gtk_enumerate_printers (GtkPrinterFunc func, gpointer data, GDestroyNotify destroy, gboolean wait) { PrinterList *printer_list; GList *node, *next; GtkPrintBackend *backend; printer_list = g_new0 (PrinterList, 1); printer_list->func = func; printer_list->data = data; printer_list->destroy = destroy; if (g_module_supported ()) printer_list->backends = gtk_print_backend_load_modules (); if (printer_list->backends == NULL) { free_printer_list (printer_list); return; } for (node = printer_list->backends; node != NULL; node = next) { next = node->next; backend = GTK_PRINT_BACKEND (node->data); if (list_printers_init (printer_list, backend)) return; } if (wait && printer_list->backends) { printer_list->loop = g_main_loop_new (NULL, FALSE); g_main_loop_run (printer_list->loop); } } GType gtk_print_capabilities_get_type (void) { static GType etype = 0; if (G_UNLIKELY (etype == 0)) { static const GFlagsValue values[] = { { GTK_PRINT_CAPABILITY_PAGE_SET, "GTK_PRINT_CAPABILITY_PAGE_SET", "page-set" }, { GTK_PRINT_CAPABILITY_COPIES, "GTK_PRINT_CAPABILITY_COPIES", "copies" }, { GTK_PRINT_CAPABILITY_COLLATE, "GTK_PRINT_CAPABILITY_COLLATE", "collate" }, { GTK_PRINT_CAPABILITY_REVERSE, "GTK_PRINT_CAPABILITY_REVERSE", "reverse" }, { GTK_PRINT_CAPABILITY_SCALE, "GTK_PRINT_CAPABILITY_SCALE", "scale" }, { GTK_PRINT_CAPABILITY_GENERATE_PDF, "GTK_PRINT_CAPABILITY_GENERATE_PDF", "generate-pdf" }, { GTK_PRINT_CAPABILITY_GENERATE_PS, "GTK_PRINT_CAPABILITY_GENERATE_PS", "generate-ps" }, { GTK_PRINT_CAPABILITY_PREVIEW, "GTK_PRINT_CAPABILITY_PREVIEW", "preview" }, { GTK_PRINT_CAPABILITY_NUMBER_UP, "GTK_PRINT_CAPABILITY_NUMBER_UP", "number-up"}, { GTK_PRINT_CAPABILITY_NUMBER_UP_LAYOUT, "GTK_PRINT_CAPABILITY_NUMBER_UP_LAYOUT", "number-up-layout" }, { 0, NULL, NULL } }; etype = g_flags_register_static (I_("GtkPrintCapabilities"), values); } return etype; }