/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
*
* Copyright 2012 Red Hat, Inc,
*
* 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, see .
*
* Author: Marek Kasik
*/
#include "config.h"
#include
#include
#include
#include
#include
#include
#include
#include
#include "list-box-helper.h"
#include "pp-jobs-dialog.h"
#include "pp-utils.h"
#include "pp-job.h"
#include "pp-cups.h"
#include "pp-printer.h"
#define EMPTY_TEXT "\xe2\x80\x94"
#define CLOCK_SCHEMA "org.gnome.desktop.interface"
#define CLOCK_FORMAT_KEY "clock-format"
struct _PpJobsDialog {
GtkDialog parent_instance;
GtkButton *authenticate_button;
GtkMenuButton *authenticate_jobs_button;
GtkLabel *authenticate_jobs_label;
GtkInfoBar *authentication_infobar;
GtkLabel *authentication_label;
GtkEntry *domain_entry;
GtkLabel *domain_label;
GtkButton *jobs_clear_all_button;
GtkListBox *jobs_listbox;
GtkScrolledWindow *list_jobs_page;
GtkBox *no_jobs_page;
GtkEntry *password_entry;
GtkLabel *password_label;
GtkStack *stack;
GListStore *store;
GtkEntry *username_entry;
GtkLabel *username_label;
gchar *printer_name;
gchar **actual_auth_info_required;
gboolean jobs_filled;
gboolean pop_up_authentication_popup;
GCancellable *get_jobs_cancellable;
};
G_DEFINE_TYPE (PpJobsDialog, pp_jobs_dialog, GTK_TYPE_DIALOG)
static gboolean
is_info_required (PpJobsDialog *self,
const gchar *info)
{
gint i;
if (self->actual_auth_info_required == NULL)
return FALSE;
for (i = 0; self->actual_auth_info_required[i] != NULL; i++)
if (g_strcmp0 (self->actual_auth_info_required[i], info) == 0)
return TRUE;
return FALSE;
}
static gboolean
is_domain_required (PpJobsDialog *self)
{
return is_info_required (self, "domain");
}
static gboolean
is_username_required (PpJobsDialog *self)
{
return is_info_required (self, "username");
}
static gboolean
is_password_required (PpJobsDialog *self)
{
return is_info_required (self, "password");
}
static gboolean
auth_popup_filled (PpJobsDialog *self)
{
gboolean domain_required;
gboolean username_required;
gboolean password_required;
guint16 domain_length;
guint16 username_length;
guint16 password_length;
domain_required = is_domain_required (self);
username_required = is_username_required (self);
password_required = is_password_required (self);
domain_length = gtk_entry_get_text_length (self->domain_entry);
username_length = gtk_entry_get_text_length (self->username_entry);
password_length = gtk_entry_get_text_length (self->password_entry);
return (!domain_required || domain_length > 0) &&
(!username_required || username_length > 0) &&
(!password_required || password_length > 0);
}
static void
auth_entries_changed (PpJobsDialog *self)
{
gtk_widget_set_sensitive (GTK_WIDGET (self->authenticate_button), auth_popup_filled (self));
}
static void
auth_entries_activated (PpJobsDialog *self)
{
if (auth_popup_filled (self))
gtk_button_clicked (self->authenticate_button);
}
static void
authenticate_popover_update (PpJobsDialog *self)
{
gboolean domain_required;
gboolean username_required;
gboolean password_required;
domain_required = is_domain_required (self);
username_required = is_username_required (self);
password_required = is_password_required (self);
gtk_widget_set_visible (GTK_WIDGET (self->domain_label), domain_required);
gtk_widget_set_visible (GTK_WIDGET (self->domain_entry), domain_required);
if (domain_required)
gtk_entry_set_text (self->domain_entry, "");
gtk_widget_set_visible (GTK_WIDGET (self->username_label), username_required);
gtk_widget_set_visible (GTK_WIDGET (self->username_entry), username_required);
if (username_required)
gtk_entry_set_text (self->username_entry, cupsUser ());
gtk_widget_set_visible (GTK_WIDGET (self->password_label), password_required);
gtk_widget_set_visible (GTK_WIDGET (self->password_entry), password_required);
if (password_required)
gtk_entry_set_text (self->password_entry, "");
gtk_widget_set_sensitive (GTK_WIDGET (self->authenticate_button), FALSE);
}
static void
job_stop_cb (GtkButton *button,
PpJob *job)
{
pp_job_cancel_purge_async (job, FALSE);
}
static void
job_pause_cb (GtkButton *button,
PpJob *job)
{
gint job_state;
g_object_get (job, "state", &job_state, NULL);
pp_job_set_hold_until_async (job, job_state == IPP_JOB_HELD ? "no-hold" : "indefinite");
gtk_button_set_image (button,
gtk_image_new_from_icon_name (job_state == IPP_JOB_HELD ?
"media-playback-pause-symbolic" : "media-playback-start-symbolic",
GTK_ICON_SIZE_SMALL_TOOLBAR));
}
static GtkWidget *
create_listbox_row (gpointer item,
gpointer user_data)
{
GtkWidget *widget;
GtkWidget *box;
PpJob *job = (PpJob *)item;
gchar **auth_info_required;
gchar *title;
gchar *state_string = NULL;
gint job_state;
g_object_get (job,
"title", &title,
"state", &job_state,
"auth-info-required", &auth_info_required,
NULL);
switch (job_state)
{
case IPP_JOB_PENDING:
/* Translators: Job's state (job is waiting to be printed) */
state_string = g_strdup (C_("print job", "Pending"));
break;
case IPP_JOB_HELD:
if (auth_info_required == NULL)
{
/* Translators: Job's state (job is held for printing) */
state_string = g_strdup (C_("print job", "Paused"));
}
else
{
/* Translators: Job's state (job needs authentication to proceed further) */
state_string = g_strdup_printf ("%s", C_("print job", "Authentication required"));
}
break;
case IPP_JOB_PROCESSING:
/* Translators: Job's state (job is currently printing) */
state_string = g_strdup (C_("print job", "Processing"));
break;
case IPP_JOB_STOPPED:
/* Translators: Job's state (job has been stopped) */
state_string = g_strdup (C_("print job", "Stopped"));
break;
case IPP_JOB_CANCELED:
/* Translators: Job's state (job has been canceled) */
state_string = g_strdup (C_("print job", "Canceled"));
break;
case IPP_JOB_ABORTED:
/* Translators: Job's state (job has aborted due to error) */
state_string = g_strdup (C_("print job", "Aborted"));
break;
case IPP_JOB_COMPLETED:
/* Translators: Job's state (job has completed successfully) */
state_string = g_strdup (C_("print job", "Completed"));
break;
}
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
g_object_set (box, "margin", 6, NULL);
gtk_container_set_border_width (GTK_CONTAINER (box), 2);
widget = gtk_label_new (title);
gtk_label_set_max_width_chars (GTK_LABEL (widget), 40);
gtk_label_set_ellipsize (GTK_LABEL (widget), PANGO_ELLIPSIZE_END);
gtk_widget_set_halign (widget, GTK_ALIGN_START);
gtk_box_pack_start (GTK_BOX (box), widget, TRUE, TRUE, 10);
widget = gtk_label_new (NULL);
gtk_label_set_markup (GTK_LABEL (widget), state_string);
gtk_widget_set_halign (widget, GTK_ALIGN_END);
gtk_widget_set_margin_end (widget, 64);
gtk_widget_set_margin_start (widget, 64);
gtk_box_pack_start (GTK_BOX (box), widget, FALSE, FALSE, 10);
widget = gtk_button_new_from_icon_name (job_state == IPP_JOB_HELD ? "media-playback-start-symbolic" : "media-playback-pause-symbolic",
GTK_ICON_SIZE_SMALL_TOOLBAR);
g_signal_connect (widget, "clicked", G_CALLBACK (job_pause_cb), item);
gtk_widget_set_sensitive (widget, auth_info_required == NULL);
gtk_box_pack_start (GTK_BOX (box), widget, FALSE, FALSE, 4);
widget = gtk_button_new_from_icon_name ("edit-delete-symbolic",
GTK_ICON_SIZE_SMALL_TOOLBAR);
g_signal_connect (widget, "clicked", G_CALLBACK (job_stop_cb), item);
gtk_box_pack_start (GTK_BOX (box), widget, FALSE, FALSE, 4);
gtk_widget_show_all (box);
return box;
}
static void
pop_up_authentication_popup (PpJobsDialog *self)
{
if (self->actual_auth_info_required != NULL)
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (self->authenticate_jobs_button), TRUE);
}
static void
update_jobs_list_cb (GObject *source_object,
GAsyncResult *result,
gpointer user_data)
{
PpJobsDialog *self = user_data;
PpPrinter *printer = PP_PRINTER (source_object);
g_autoptr(GError) error = NULL;
g_autoptr(GPtrArray) jobs;
PpJob *job;
gchar **auth_info_required = NULL;
gint num_of_auth_jobs = 0;
guint i;
g_list_store_remove_all (self->store);
jobs = pp_printer_get_jobs_finish (printer, result, &error);
if (error != NULL)
{
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
{
g_warning ("Could not get jobs: %s", error->message);
}
return;
}
if (jobs->len > 0)
{
gtk_widget_set_sensitive (GTK_WIDGET (self->jobs_clear_all_button), TRUE);
gtk_stack_set_visible_child (self->stack, GTK_WIDGET (self->list_jobs_page));
}
else
{
gtk_widget_set_sensitive (GTK_WIDGET (self->jobs_clear_all_button), FALSE);
gtk_stack_set_visible_child (self->stack, GTK_WIDGET (self->no_jobs_page));
}
for (i = 0; i < jobs->len; i++)
{
job = PP_JOB (g_ptr_array_index (jobs, i));
g_list_store_append (self->store, g_object_ref (job));
g_object_get (G_OBJECT (job),
"auth-info-required", &auth_info_required,
NULL);
if (auth_info_required != NULL)
{
num_of_auth_jobs++;
if (self->actual_auth_info_required == NULL)
self->actual_auth_info_required = auth_info_required;
else
g_strfreev (auth_info_required);
auth_info_required = NULL;
}
}
if (num_of_auth_jobs > 0)
{
g_autofree gchar *text = NULL;
/* Translators: This label shows how many jobs of this printer needs to be authenticated to be printed. */
text = g_strdup_printf (ngettext ("%u Job Requires Authentication", "%u Jobs Require Authentication", num_of_auth_jobs), num_of_auth_jobs);
gtk_label_set_text (self->authenticate_jobs_label, text);
gtk_widget_show (GTK_WIDGET (self->authentication_infobar));
}
else
{
gtk_widget_hide (GTK_WIDGET (self->authentication_infobar));
}
authenticate_popover_update (self);
g_clear_object (&self->get_jobs_cancellable);
if (!self->jobs_filled)
{
if (self->pop_up_authentication_popup)
{
pop_up_authentication_popup (self);
self->pop_up_authentication_popup = FALSE;
}
self->jobs_filled = TRUE;
}
}
static void
update_jobs_list (PpJobsDialog *self)
{
PpPrinter *printer;
if (self->printer_name != NULL)
{
g_cancellable_cancel (self->get_jobs_cancellable);
g_clear_object (&self->get_jobs_cancellable);
self->get_jobs_cancellable = g_cancellable_new ();
printer = pp_printer_new (self->printer_name);
pp_printer_get_jobs_async (printer,
TRUE,
CUPS_WHICHJOBS_ACTIVE,
self->get_jobs_cancellable,
update_jobs_list_cb,
self);
}
}
static void
on_clear_all_button_clicked (PpJobsDialog *self)
{
guint num_items;
guint i;
num_items = g_list_model_get_n_items (G_LIST_MODEL (self->store));
for (i = 0; i < num_items; i++)
{
PpJob *job = PP_JOB (g_list_model_get_item (G_LIST_MODEL (self->store), i));
pp_job_cancel_purge_async (job, FALSE);
}
}
static void
pp_job_authenticate_cb (GObject *source_object,
GAsyncResult *res,
gpointer user_data)
{
PpJobsDialog *self = user_data;
gboolean result;
g_autoptr(GError) error = NULL;
PpJob *job = PP_JOB (source_object);
result = pp_job_authenticate_finish (job, res, &error);
if (result)
{
pp_jobs_dialog_update (self);
}
else if (error != NULL)
{
if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
{
g_warning ("Could not authenticate job: %s", error->message);
}
}
}
static void
authenticate_button_clicked (PpJobsDialog *self)
{
PpJob *job;
gchar **auth_info_required = NULL;
gchar **auth_info;
guint num_items;
gint i;
auth_info = g_new0 (gchar *, g_strv_length (self->actual_auth_info_required) + 1);
for (i = 0; self->actual_auth_info_required[i] != NULL; i++)
{
if (g_strcmp0 (self->actual_auth_info_required[i], "domain") == 0)
auth_info[i] = g_strdup (gtk_entry_get_text (GTK_ENTRY (self->domain_entry)));
else if (g_strcmp0 (self->actual_auth_info_required[i], "username") == 0)
auth_info[i] = g_strdup (gtk_entry_get_text (GTK_ENTRY (self->username_entry)));
else if (g_strcmp0 (self->actual_auth_info_required[i], "password") == 0)
auth_info[i] = g_strdup (gtk_entry_get_text (GTK_ENTRY (self->password_entry)));
}
num_items = g_list_model_get_n_items (G_LIST_MODEL (self->store));
for (i = 0; i < num_items; i++)
{
job = PP_JOB (g_list_model_get_item (G_LIST_MODEL (self->store), i));
g_object_get (job, "auth-info-required", &auth_info_required, NULL);
if (auth_info_required != NULL)
{
pp_job_authenticate_async (job, auth_info, NULL, pp_job_authenticate_cb, self);
g_strfreev (auth_info_required);
auth_info_required = NULL;
}
}
g_strfreev (auth_info);
}
static gboolean
key_press_event_cb (GtkWidget *widget,
GdkEventKey *event,
gpointer user_data)
{
if (event->keyval == GDK_KEY_Escape)
gtk_dialog_response (GTK_DIALOG (widget), GTK_RESPONSE_CLOSE);
return FALSE;
}
PpJobsDialog *
pp_jobs_dialog_new (const gchar *printer_name)
{
PpJobsDialog *self;
g_autofree gchar *text = NULL;
g_autofree gchar *title = NULL;
self = g_object_new (PP_TYPE_JOBS_DIALOG,
"use-header-bar", 1,
NULL);
self->printer_name = g_strdup (printer_name);
self->actual_auth_info_required = NULL;
self->jobs_filled = FALSE;
self->pop_up_authentication_popup = FALSE;
/* connect signals */
g_signal_connect (self, "key-press-event", G_CALLBACK (key_press_event_cb), NULL);
/* Translators: This is the printer name for which we are showing the active jobs */
title = g_strdup_printf (C_("Printer jobs dialog title", "%s — Active Jobs"), printer_name);
gtk_window_set_title (GTK_WINDOW (self), title);
/* Translators: The printer needs authentication info to print. */
text = g_strdup_printf (_("Enter credentials to print from %s."), printer_name);
gtk_label_set_text (self->authentication_label, text);
gtk_list_box_set_header_func (self->jobs_listbox,
cc_list_box_update_header_func, NULL, NULL);
self->store = g_list_store_new (pp_job_get_type ());
gtk_list_box_bind_model (self->jobs_listbox, G_LIST_MODEL (self->store),
create_listbox_row, NULL, NULL);
update_jobs_list (self);
return self;
}
void
pp_jobs_dialog_update (PpJobsDialog *self)
{
update_jobs_list (self);
}
void
pp_jobs_dialog_authenticate_jobs (PpJobsDialog *self)
{
if (self->jobs_filled)
pop_up_authentication_popup (self);
else
self->pop_up_authentication_popup = TRUE;
}
static void
pp_jobs_dialog_init (PpJobsDialog *dialog)
{
gtk_widget_init_template (GTK_WIDGET (dialog));
}
static void
pp_jobs_dialog_dispose (GObject *object)
{
PpJobsDialog *self = PP_JOBS_DIALOG (object);
g_cancellable_cancel (self->get_jobs_cancellable);
g_clear_object (&self->get_jobs_cancellable);
g_clear_pointer (&self->actual_auth_info_required, g_strfreev);
g_clear_pointer (&self->printer_name, g_free);
G_OBJECT_CLASS (pp_jobs_dialog_parent_class)->dispose (object);
}
static void
pp_jobs_dialog_class_init (PpJobsDialogClass *klass)
{
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
GObjectClass *object_class = G_OBJECT_CLASS (klass);
gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/control-center/printers/pp-jobs-dialog.ui");
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, authenticate_button);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, authenticate_jobs_button);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, authenticate_jobs_label);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, authentication_infobar);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, authentication_label);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, domain_entry);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, domain_label);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, jobs_clear_all_button);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, jobs_listbox);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, list_jobs_page);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, no_jobs_page);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, password_entry);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, password_label);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, stack);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, username_entry);
gtk_widget_class_bind_template_child (widget_class, PpJobsDialog, username_label);
gtk_widget_class_bind_template_callback (widget_class, authenticate_button_clicked);
gtk_widget_class_bind_template_callback (widget_class, on_clear_all_button_clicked);
gtk_widget_class_bind_template_callback (widget_class, auth_entries_activated);
gtk_widget_class_bind_template_callback (widget_class, auth_entries_changed);
object_class->dispose = pp_jobs_dialog_dispose;
}