#!/usr/bin/python3
## system-config-printer
## Copyright (C) 2006, 2007, 2008, 2009, 2010, 2011, 2012, 2013, 2014 Red Hat, Inc.
## Authors:
## Tim Waugh <twaugh@redhat.com>
## Florian Festi <ffesti@redhat.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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
# config is generated from config.py.in by configure
import config
import os, tempfile
from gi.repository import Gtk
import cups
import locale
import gettext
gettext.install(domain=config.PACKAGE, localedir=config.localedir)
import cupshelpers, options
from gi.repository import GObject
from gi.repository import GLib
from gui import GtkGUI
import html # requires python3.2
from optionwidgets import OptionWidget
from debug import *
import authconn
from errordialogs import *
import gtkinklevel
import ppdcache
import statereason
import monitor
import newprinter
from newprinter import busy, ready
import ppdippstr
pkgdata = config.pkgdatadir
def CUPS_server_hostname ():
host = cups.getServer ()
if host[0] == '/':
return 'localhost'
return host
def on_delete_just_hide (widget, event):
widget.hide ()
return True # stop other handlers
class PrinterPropertiesDialog(GtkGUI):
__gsignals__ = {
'destroy': ( GObject.SignalFlags.RUN_LAST, None, ()),
'dialog-closed': ( GObject.SignalFlags.RUN_LAST, None, ()),
}
printer_states = { cups.IPP_PRINTER_IDLE:
_("Idle"),
cups.IPP_PRINTER_PROCESSING:
_("Processing"),
cups.IPP_PRINTER_BUSY:
_("Busy"),
cups.IPP_PRINTER_STOPPED:
_("Stopped") }
def __init__(self):
GObject.GObject.__init__ (self)
try:
self.language = locale.getlocale(locale.LC_MESSAGES)
self.encoding = locale.getlocale(locale.LC_CTYPE)
except:
nonfatalException()
os.environ['LC_ALL'] = 'C'
locale.setlocale (locale.LC_ALL, "")
self.language = locale.getlocale(locale.LC_MESSAGES)
self.encoding = locale.getlocale(locale.LC_CTYPE)
self.parent = None
self.printer = self.ppd = None
self.conflicts = set() # of options
self.changed = set() # of options
self.signal_ids = dict()
# WIDGETS
# =======
self.updating_widgets = False
self.getWidgets({"PrinterPropertiesDialog":
["PrinterPropertiesDialog",
"tvPrinterProperties",
"btnPrinterPropertiesCancel",
"btnPrinterPropertiesOK",
"btnPrinterPropertiesApply",
"btnPrinterPropertiesClose",
"ntbkPrinter",
"entPDescription",
"entPLocation",
"entPMakeModel",
"lblPMakeModel2",
"entPState",
"entPDevice",
"lblPDevice2",
"btnSelectDevice",
"btnChangePPD",
"chkPEnabled",
"chkPAccepting",
"chkPShared",
"lblNotPublished",
"btnPrintTestPage",
"btnSelfTest",
"btnCleanHeads",
"btnConflict",
"cmbPStartBanner",
"cmbPEndBanner",
"cmbPErrorPolicy",
"cmbPOperationPolicy",
"rbtnPAllow",
"rbtnPDeny",
"tvPUsers",
"entPUser",
"btnPAddUser",
"btnPDelUser",
"lblPInstallOptions",
"swPInstallOptions",
"vbPInstallOptions",
"swPOptions",
"lblPOptions",
"vbPOptions",
"vbClassMembers",
"lblClassMembers",
"tvClassMembers",
"tvClassNotMembers",
"btnClassAddMember",
"btnClassDelMember",
"btnRefreshMarkerLevels",
"tvPrinterStateReasons",
"ntbkPrinterStateReasons",
# Job options
"sbJOCopies", "btnJOResetCopies",
"cmbJOOrientationRequested", "btnJOResetOrientationRequested",
"cbJOFitplot", "btnJOResetFitplot",
"cmbJONumberUp", "btnJOResetNumberUp",
"cmbJONumberUpLayout", "btnJOResetNumberUpLayout",
"sbJOBrightness", "btnJOResetBrightness",
"cmbJOFinishings", "btnJOResetFinishings",
"sbJOJobPriority", "btnJOResetJobPriority",
"cmbJOMedia", "btnJOResetMedia",
"cmbJOSides", "btnJOResetSides",
"cmbJOHoldUntil", "btnJOResetHoldUntil",
"cmbJOOutputOrder", "btnJOResetOutputOrder",
"cmbJOPrintQuality", "btnJOResetPrintQuality",
"cmbJOPrinterResolution",
"btnJOResetPrinterResolution",
"cmbJOOutputBin", "btnJOResetOutputBin",
"cbJOMirror", "btnJOResetMirror",
"sbJOScaling", "btnJOResetScaling",
"sbJOSaturation", "btnJOResetSaturation",
"sbJOHue", "btnJOResetHue",
"sbJOGamma", "btnJOResetGamma",
"sbJOCpi", "btnJOResetCpi",
"sbJOLpi", "btnJOResetLpi",
"sbJOPageLeft", "btnJOResetPageLeft",
"sbJOPageRight", "btnJOResetPageRight",
"sbJOPageTop", "btnJOResetPageTop",
"sbJOPageBottom", "btnJOResetPageBottom",
"cbJOPrettyPrint", "btnJOResetPrettyPrint",
"cbJOWrap", "btnJOResetWrap",
"sbJOColumns", "btnJOResetColumns",
"tblJOOther",
"entNewJobOption", "btnNewJobOption",
# Marker levels
"vboxMarkerLevels",
"btnRefreshMarkerLevels"]},
domain=config.PACKAGE)
self.dialog = self.PrinterPropertiesDialog
# Don't let delete-event destroy the dialog.
self.dialog.connect ("delete-event", self.on_delete)
# Printer properties combo boxes
for combobox in [self.cmbPStartBanner,
self.cmbPEndBanner,
self.cmbPErrorPolicy,
self.cmbPOperationPolicy]:
cell = Gtk.CellRendererText ()
combobox.clear ()
combobox.pack_start (cell, True)
combobox.add_attribute (cell, 'text', 0)
btn = self.btnRefreshMarkerLevels
btn.connect ("clicked", self.on_btnRefreshMarkerLevels_clicked)
# Printer state reasons list
column = Gtk.TreeViewColumn (_("Message"))
icon = Gtk.CellRendererPixbuf ()
column.pack_start (icon, False)
text = Gtk.CellRendererText ()
column.pack_start (text, False)
column.set_cell_data_func (icon, self.set_printer_state_reason_icon, None)
column.set_cell_data_func (text, self.set_printer_state_reason_text, None)
column.set_resizable (True)
self.tvPrinterStateReasons.append_column (column)
selection = self.tvPrinterStateReasons.get_selection ()
selection.set_mode (Gtk.SelectionMode.NONE)
store = Gtk.ListStore (int, str)
self.tvPrinterStateReasons.set_model (store)
self.PrinterPropertiesDialog.connect ("delete-event",
on_delete_just_hide)
self.static_tabs = 3
# setup some lists
for name, treeview in (
(_("Members of this class"), self.tvClassMembers),
(_("Others"), self.tvClassNotMembers),
(_("Users"), self.tvPUsers),
):
model = Gtk.ListStore(str)
cell = Gtk.CellRendererText()
column = Gtk.TreeViewColumn(name, cell, text=0)
treeview.set_model(model)
treeview.append_column(column)
treeview.get_selection().set_mode(Gtk.SelectionMode.MULTIPLE)
# Printer Properties dialog
self.dialog.connect ('response', self.printer_properties_response)
# Printer Properties tree view
col = Gtk.TreeViewColumn ('', Gtk.CellRendererText (), markup=0)
self.tvPrinterProperties.append_column (col)
sel = self.tvPrinterProperties.get_selection ()
sel.connect ('changed', self.on_tvPrinterProperties_selection_changed)
sel.set_mode (Gtk.SelectionMode.SINGLE)
# Job Options widgets.
for (widget,
opts) in [(self.cmbJOOrientationRequested,
[[_("Portrait (no rotation)")],
[_("Landscape (90 degrees)")],
[_("Reverse landscape (270 degrees)")],
[_("Reverse portrait (180 degrees)")]]),
(self.cmbJONumberUp,
[["1"], ["2"], ["4"], ["6"], ["9"], ["16"]]),
(self.cmbJONumberUpLayout,
[[_("Left to right, top to bottom")],
[_("Left to right, bottom to top")],
[_("Right to left, top to bottom")],
[_("Right to left, bottom to top")],
[_("Top to bottom, left to right")],
[_("Top to bottom, right to left")],
[_("Bottom to top, left to right")],
[_("Bottom to top, right to left")]]),
(self.cmbJOFinishings,
# See section 4.2.6 of this document for explanation of finishing types:
# ftp://ftp.pwg.org/pub/pwg/candidates/cs-ippfinishings10-20010205-5100.1.pdf
[[_("None")],
[_("Staple")],
[_("Punch")],
[_("Cover")],
[_("Bind")],
[_("Saddle stitch")],
[_("Edge stitch")],
[_("Fold")],
[_("Trim")],
[_("Bale")],
[_("Booklet maker")],
[_("Job offset")],
[_("Staple (top left)")],
[_("Staple (bottom left)")],
[_("Staple (top right)")],
[_("Staple (bottom right)")],
[_("Edge stitch (left)")],
[_("Edge stitch (top)")],
[_("Edge stitch (right)")],
[_("Edge stitch (bottom)")],
[_("Staple dual (left)")],
[_("Staple dual (top)")],
[_("Staple dual (right)")],
[_("Staple dual (bottom)")],
[_("Bind (left)")],
[_("Bind (top)")],
[_("Bind (right)")],
[_("Bind (bottom)")]]),
(self.cmbJOMedia, []),
(self.cmbJOSides,
[[_("One-sided")],
[_("Two-sided (long edge)")],
[_("Two-sided (short edge)")]]),
(self.cmbJOHoldUntil, []),
(self.cmbJOOutputOrder,
[[_("Normal")],
[_("Reverse")]]),
(self.cmbJOPrintQuality,
[[_("Draft")],
[_("Normal")],
[_("High")]]),
(self.cmbJOOutputBin, []),
]:
model = Gtk.ListStore (str)
for row in opts:
model.append (row=row)
cell = Gtk.CellRendererText ()
widget.pack_start (cell, True)
widget.add_attribute (cell, 'text', 0)
widget.set_model (model)
opts = [ options.OptionAlwaysShown ("copies", int, 1,
self.sbJOCopies,
self.btnJOResetCopies),
options.OptionAlwaysShownSpecial \
("orientation-requested", int, 3,
self.cmbJOOrientationRequested,
self.btnJOResetOrientationRequested,
combobox_map = [3, 4, 5, 6],
special_choice=_("Automatic rotation")),
options.OptionAlwaysShown ("fitplot", bool, False,
self.cbJOFitplot,
self.btnJOResetFitplot),
options.OptionAlwaysShown ("number-up", int, 1,
self.cmbJONumberUp,
self.btnJOResetNumberUp,
combobox_map=[1, 2, 4, 6, 9, 16],
use_supported = True),
options.OptionAlwaysShown ("number-up-layout", str, "lrtb",
self.cmbJONumberUpLayout,
self.btnJOResetNumberUpLayout,
combobox_map = [ "lrtb",
"lrbt",
"rltb",
"rlbt",
"tblr",
"tbrl",
"btlr",
"btrl" ]),
options.OptionAlwaysShown ("brightness", int, 100,
self.sbJOBrightness,
self.btnJOResetBrightness),
options.OptionAlwaysShown ("finishings", int, 3,
self.cmbJOFinishings,
self.btnJOResetFinishings,
combobox_map = [ 3, 4, 5, 6,
7, 8, 9, 10,
11, 12, 13, 14,
20, 21, 22, 23,
24, 25, 26, 27,
28, 29, 30, 31,
50, 51, 52, 53 ],
use_supported = True),
options.OptionAlwaysShown ("job-priority", int, 50,
self.sbJOJobPriority,
self.btnJOResetJobPriority),
options.OptionAlwaysShown ("media", str,
"A4", # This is the default for
# when media-default is
# not supplied by the IPP
# server. Fortunately it
# is a mandatory attribute.
self.cmbJOMedia,
self.btnJOResetMedia,
use_supported = True),
options.OptionAlwaysShown ("sides", str, "one-sided",
self.cmbJOSides,
self.btnJOResetSides,
combobox_map =
[ "one-sided",
"two-sided-long-edge",
"two-sided-short-edge" ],
use_supported = True),
options.OptionAlwaysShown ("job-hold-until", str,
"no-hold",
self.cmbJOHoldUntil,
self.btnJOResetHoldUntil,
use_supported = True),
options.OptionAlwaysShown ("outputorder", str,
"normal",
self.cmbJOOutputOrder,
self.btnJOResetOutputOrder,
combobox_map =
[ "normal",
"reverse" ]),
options.OptionAlwaysShown ("print-quality", int, 3,
self.cmbJOPrintQuality,
self.btnJOResetPrintQuality,
combobox_map = [ 3, 4, 5 ],
use_supported = True),
options.OptionAlwaysShown ("printer-resolution",
options.IPPResolution,
options.IPPResolution((300,300,3)),
self.cmbJOPrinterResolution,
self.btnJOResetPrinterResolution,
use_supported = True),
options.OptionAlwaysShown ("output-bin", str,
"face-up",
self.cmbJOOutputBin,
self.btnJOResetOutputBin,
use_supported = True),
options.OptionAlwaysShown ("mirror", bool, False,
self.cbJOMirror,
self.btnJOResetMirror),
options.OptionAlwaysShown ("scaling", int, 100,
self.sbJOScaling,
self.btnJOResetScaling),
options.OptionAlwaysShown ("saturation", int, 100,
self.sbJOSaturation,
self.btnJOResetSaturation),
options.OptionAlwaysShown ("hue", int, 0,
self.sbJOHue,
self.btnJOResetHue),
options.OptionAlwaysShown ("gamma", int, 1000,
self.sbJOGamma,
self.btnJOResetGamma),
options.OptionAlwaysShown ("cpi", float, 10.0,
self.sbJOCpi, self.btnJOResetCpi),
options.OptionAlwaysShown ("lpi", float, 6.0,
self.sbJOLpi, self.btnJOResetLpi),
options.OptionAlwaysShown ("page-left", int, 0,
self.sbJOPageLeft,
self.btnJOResetPageLeft),
options.OptionAlwaysShown ("page-right", int, 0,
self.sbJOPageRight,
self.btnJOResetPageRight),
options.OptionAlwaysShown ("page-top", int, 0,
self.sbJOPageTop,
self.btnJOResetPageTop),
options.OptionAlwaysShown ("page-bottom", int, 0,
self.sbJOPageBottom,
self.btnJOResetPageBottom),
options.OptionAlwaysShown ("prettyprint", bool, False,
self.cbJOPrettyPrint,
self.btnJOResetPrettyPrint),
options.OptionAlwaysShown ("wrap", bool, False, self.cbJOWrap,
self.btnJOResetWrap),
options.OptionAlwaysShown ("columns", int, 1,
self.sbJOColumns,
self.btnJOResetColumns),
]
self.job_options_widgets = {}
self.job_options_buttons = {}
for option in opts:
self.job_options_widgets[option.widget] = option
self.job_options_buttons[option.button] = option
self._monitor = None
self._ppdcache = None
self.connect_signals ()
debugprint ("+%s" % self)
def __del__ (self):
debugprint ("-%s" % self)
del self._monitor
def _connect (self, collection, obj, name, handler):
c = self.signal_ids.get (collection, [])
c.append ((obj, obj.connect (name, handler)))
self.signal_ids[collection] = c
def _disconnect (self, collection=None):
if collection:
collection = [collection]
else:
collection = list(self.signal_ids.keys ())
for coll in collection:
if coll in self.signal_ids:
for (obj, signal_id) in self.signal_ids[coll]:
obj.disconnect (signal_id)
del self.signal_ids[coll]
def do_destroy (self):
if self.PrinterPropertiesDialog:
self.PrinterPropertiesDialog.destroy ()
self.PrinterPropertiesDialog = None
def destroy (self):
debugprint ("DESTROY: %s" % self)
self._disconnect ()
self.ppd = None
self.ppd_local = None
self.printer = None
self.emit ('destroy')
def set_monitor (self, monitor):
self._monitor = monitor
if not monitor:
return
self._monitor.connect ('printer-event', self.on_printer_event)
self._monitor.connect ('printer-removed', self.on_printer_removed)
self._monitor.connect ('state-reason-added', self.on_state_reason_added)
self._monitor.connect ('state-reason-removed',
self.on_state_reason_removed)
self._monitor.connect ('cups-connection-error',
self.on_cups_connection_error)
def show (self, name, host=None, encryption=None, parent=None):
self.parent = parent
self._host = host
self._encryption = encryption
if not host:
self._host = cups.getServer()
if not encryption:
self._encryption = cups.getEncryption ()
if self._monitor is None:
self.set_monitor (monitor.Monitor (monitor_jobs=False))
self._ppdcache = self._monitor.get_ppdcache ()
self._disconnect ("newPrinterGUI")
self.newPrinterGUI = newprinter.NewPrinterGUI ()
self._connect ("newPrinterGUI", self.newPrinterGUI,
"printer-modified", self.on_printer_modified)
self._connect ("newPrinterGUI", self.newPrinterGUI,
"dialog-canceled", self.on_printer_not_modified)
if parent:
self.dialog.set_transient_for (parent)
self.load (name, host=host, encryption=encryption, parent=parent)
if not self.printer:
return
for button in [self.btnPrinterPropertiesCancel,
self.btnPrinterPropertiesOK,
self.btnPrinterPropertiesApply]:
if self.printer.discovered:
button.hide ()
else:
button.show ()
if self.printer.discovered:
self.btnPrinterPropertiesClose.show ()
else:
self.btnPrinterPropertiesClose.hide ()
self.setDataButtonState ()
self.btnPrintTestPage.set_tooltip_text(_("CUPS test page"))
self.btnSelfTest.set_tooltip_text(_("Typically shows whether all jets "
"on a print head are functioning "
"and that the print feed mechanisms"
" are working properly."))
treeview = self.tvPrinterProperties
treeview.set_cursor (Gtk.TreePath(), None, False)
host = CUPS_server_hostname ()
self.dialog.set_title (_("Printer Properties - "
"'%s' on %s") % (name, host))
self.dialog.show ()
def printer_properties_response (self, dialog, response):
if not self.printer:
response = Gtk.ResponseType.CANCEL
if response == Gtk.ResponseType.REJECT:
# The Conflict button was pressed.
message = _("There are conflicting options.\n"
"Changes can only be applied after\n"
"these conflicts are resolved.")
message += "\n\n"
for option in self.conflicts:
message += option.option.text + "\n"
dialog = Gtk.MessageDialog(parent=self.dialog,
modal=True, destroy_with_parent=True,
message_type=Gtk.MessageType.WARNING,
buttons=Gtk.ButtonsType.CLOSE,
text=message)
dialog.run()
dialog.destroy()
return
if (response == Gtk.ResponseType.OK or
response == Gtk.ResponseType.APPLY):
if (response == Gtk.ResponseType.OK and len (self.changed) == 0):
failed = False
else:
failed = self.save_printer (self.printer)
if response == Gtk.ResponseType.APPLY and not failed:
try:
self.load (self.printer.name)
except:
pass
self.setDataButtonState ()
if ((response == Gtk.ResponseType.OK and not failed) or
response == Gtk.ResponseType.CANCEL):
self.ppd = None
self.ppd_local = None
self.printer = None
dialog.hide ()
self.emit ('dialog-closed')
if self.newPrinterGUI.NewPrinterWindow.get_property ("visible"):
self.newPrinterGUI.on_NPCancel (None)
# Data handling
def on_delete(self, dialog, event):
self.printer_properties_response (dialog, Gtk.ResponseType.CANCEL)
def on_printer_changed(self, widget):
if isinstance(widget, Gtk.CheckButton):
value = widget.get_active()
elif isinstance(widget, Gtk.Entry):
value = widget.get_text()
elif isinstance(widget, Gtk.RadioButton):
value = widget.get_active()
elif isinstance(widget, Gtk.ComboBox):
model = widget.get_model ()
iter = widget.get_active_iter()
value = model.get_value (iter, 1)
else:
raise ValueError("Widget type not supported (yet)")
p = self.printer
old_values = {
self.entPDescription : p.info,
self.entPLocation : p.location,
self.entPDevice : p.device_uri,
self.chkPEnabled : p.enabled,
self.chkPAccepting : not p.rejecting,
self.chkPShared : p.is_shared,
self.cmbPStartBanner : p.job_sheet_start,
self.cmbPEndBanner : p.job_sheet_end,
self.cmbPErrorPolicy : p.error_policy,
self.cmbPOperationPolicy : p.op_policy,
self.rbtnPAllow: p.default_allow,
}
old_value = old_values[widget]
if old_value == value:
self.changed.discard(widget)
else:
self.changed.add(widget)
self.setDataButtonState()
def option_changed(self, option):
if option.is_changed():
self.changed.add(option)
else:
self.changed.discard(option)
if option.conflicts:
self.conflicts.add(option)
else:
self.conflicts.discard(option)
self.setDataButtonState()
if (self.option_manualfeed and self.option_inputslot and
option == self.option_manualfeed):
if option.get_current_value() == "True":
self.option_inputslot.disable ()
else:
self.option_inputslot.enable ()
# Access control
def getPUsers(self):
"""return list of usernames from the GUI"""
model = self.tvPUsers.get_model()
result = []
model.foreach(lambda model, path, iter, data:
result.append(model.get(iter, 0)[0]), None)
result.sort()
return result
def setPUsers(self, users):
"""write list of usernames into the GUI"""
model = self.tvPUsers.get_model()
model.clear()
for user in users:
model.append((user,))
self.on_entPUser_changed(self.entPUser)
self.on_tvPUsers_cursor_changed(self.tvPUsers)
def checkPUsersChanged(self):
"""check if users in GUI and printer are different
and set self.changed"""
if not self.printer:
return
if self.getPUsers() != self.printer.except_users:
self.changed.add(self.tvPUsers)
else:
self.changed.discard(self.tvPUsers)
self.on_tvPUsers_cursor_changed(self.tvPUsers)
self.setDataButtonState()
def on_btnPAddUser_clicked(self, button):
user = self.entPUser.get_text()
if user:
self.tvPUsers.get_model().insert(0, (user,))
self.entPUser.set_text("")
self.checkPUsersChanged()
def on_btnPDelUser_clicked(self, button):
model, rows = self.tvPUsers.get_selection().get_selected_rows()
rows = [Gtk.TreeRowReference.new (model, row) for row in rows]
for row in rows:
path = row.get_path()
iter = model.get_iter(path)
model.remove(iter)
self.checkPUsersChanged()
def on_entPUser_changed(self, widget):
self.btnPAddUser.set_sensitive(bool(widget.get_text()))
def on_tvPUsers_cursor_changed(self, widget):
selection = widget.get_selection ()
if selection is None:
return
model, rows = selection.get_selected_rows()
self.btnPDelUser.set_sensitive(bool(rows))
# Server side options
def on_job_option_reset(self, button):
option = self.job_options_buttons[button]
option.reset ()
# Remember to set this option for removal in the IPP request.
if option.name in self.server_side_options:
del self.server_side_options[option.name]
if option.is_changed ():
self.changed.add(option)
else:
self.changed.discard(option)
self.setDataButtonState()
def on_job_option_changed(self, widget):
if not self.printer:
return
option = self.job_options_widgets[widget]
option.changed ()
if option.is_changed ():
self.server_side_options[option.name] = option
self.changed.add(option)
else:
if option.name in self.server_side_options:
del self.server_side_options[option.name]
self.changed.discard(option)
self.setDataButtonState()
# Don't set the reset button insensitive if the option hasn't
# changed from the original value: it's still meaningful to
# reset the option to the system default.
def draw_other_job_options (self, editable=True):
n = len (self.other_job_options)
if n == 0:
self.tblJOOther.hide()
return
children = self.tblJOOther.get_children ()
for child in children:
self.tblJOOther.remove (child)
i = 0
for opt in self.other_job_options:
self.tblJOOther.attach (opt.label, 0, i, 1, 1)
opt.label.set_alignment (0.0, 0.5)
self.tblJOOther.attach (opt.selector, 1, i, 1, 1)
opt.selector.set_sensitive (editable)
btn = Gtk.Button.new_from_icon_name (Gtk.STOCK_REMOVE,
Gtk.IconSize.BUTTON)
btn.connect("clicked", self.on_btnJOOtherRemove_clicked)
btn.pyobject = opt
btn.set_sensitive (editable)
self.tblJOOther.attach(btn, 2, i, 1, 1)
i += 1
self.tblJOOther.show_all ()
def add_job_option(self, name, value = "", supported = "", is_new=True,
editable=True):
try:
option = options.OptionWidget(name, value, supported,
self.option_changed)
except ValueError:
# We can't deal with this option type for some reason.
nonfatalException ()
return
option.is_new = is_new
self.other_job_options.append (option)
self.draw_other_job_options (editable=editable)
self.server_side_options[name] = option
if name in self.changed: # was deleted before
option.is_new = False
self.changed.add(option)
self.setDataButtonState()
if is_new:
option.selector.grab_focus ()
def on_btnJOOtherRemove_clicked(self, button):
option = button.pyobject
self.other_job_options.remove (option)
self.draw_other_job_options ()
if option.is_new:
self.changed.discard(option)
else:
# keep name as reminder that option got deleted
self.changed.add(option.name)
del self.server_side_options[option.name]
self.setDataButtonState()
def on_btnNewJobOption_clicked(self, button):
name = self.entNewJobOption.get_text()
self.add_job_option(name)
self.tblJOOther.show_all()
self.entNewJobOption.set_text ('')
self.btnNewJobOption.set_sensitive (False)
self.setDataButtonState()
def on_entNewJobOption_changed(self, widget):
text = self.entNewJobOption.get_text()
active = (len(text) > 0) and text not in self.server_side_options
self.btnNewJobOption.set_sensitive(active)
def on_entNewJobOption_activate(self, widget):
self.on_btnNewJobOption_clicked (widget) # wrong widget but ok
# set buttons sensitivity
def setDataButtonState(self):
try:
attrs = self.printer.other_attributes
formats = attrs.get('document-format-supported', [])
printable = (not bool (self.changed) and
self.printer.enabled and
not self.printer.rejecting)
try:
formats.index ('application/postscript')
testpage = printable
except ValueError:
# PostScript not accepted
testpage = False
self.btnPrintTestPage.set_sensitive (testpage)
adjustable = not (self.printer.discovered or bool (self.changed))
for button in [self.btnChangePPD,
self.btnSelectDevice]:
button.set_sensitive (adjustable)
selftest = False
cleanheads = False
if (printable and
(self.printer.type & cups.CUPS_PRINTER_COMMANDS) != 0):
try:
# Is the command format supported?
formats.index ('application/vnd.cups-command')
# Yes...
commands = attrs.get('printer-commands', [])
for command in commands:
if command == "PrintSelfTestPage":
selftest = True
if cleanheads:
break
elif command == "Clean":
cleanheads = True
if selftest:
break
except ValueError:
# Command format not supported.
pass
for cond, button in [(selftest, self.btnSelfTest),
(cleanheads, self.btnCleanHeads)]:
if cond:
button.show ()
else:
button.hide ()
except:
nonfatalException()
if self.ppd or \
((self.printer.remote or \
((self.printer.device_uri.startswith('dnssd:') or \
self.printer.device_uri.startswith('mdns:')) and \
self.printer.device_uri.endswith('/cups'))) and not \
self.printer.discovered):
self.btnPrintTestPage.show ()
else:
self.btnPrintTestPage.hide ()
installablebold = False
optionsbold = False
if self.conflicts:
debugprint ("Conflicts detected")
self.btnConflict.show()
for option in self.conflicts:
if option.tab_label.get_text () == self.lblPInstallOptions.get_text ():
installablebold = True
else:
optionsbold = True
else:
self.btnConflict.hide()
installabletext = _("Installable Options")
optionstext = _("Printer Options")
if installablebold:
installabletext = "<b>%s</b>" % installabletext
if optionsbold:
optionstext = "<b>%s</b>" % optionstext
self.lblPInstallOptions.set_markup (installabletext)
self.lblPOptions.set_markup (optionstext)
store = self.tvPrinterProperties.get_model ()
if store:
for n in range (self.ntbkPrinter.get_n_pages ()):
page = self.ntbkPrinter.get_nth_page (n)
label = self.ntbkPrinter.get_tab_label (page).get_text ()
try:
if label == self.lblPInstallOptions.get_text():
iter = store.get_iter ((n,))
store.set_value (iter, 0, installabletext)
elif label == self.lblPOptions.get_text ():
iter = store.get_iter ((n,))
store.set_value (iter, 0, optionstext)
except ValueError:
# If we get here, the store has not yet been set
# up (trac #111).
pass
self.btnPrinterPropertiesApply.set_sensitive (len (self.changed) > 0 and
not self.conflicts)
self.btnPrinterPropertiesOK.set_sensitive (not self.conflicts)
def save_printer(self, printer, saveall=False, parent=None):
if parent is None:
parent = self.dialog
class_deleted = False
name = printer.name
if printer.is_class:
self.cups._begin_operation (_("modifying class %s") % name)
else:
self.cups._begin_operation (_("modifying printer %s") % name)
try:
if not printer.is_class and self.ppd:
self.getPrinterSettings()
if self.ppd.nondefaultsMarked() or saveall:
self.cups.addPrinter(name, ppd=self.ppd)
if printer.is_class:
# update member list
new_members = newprinter.getCurrentClassMembers(self.tvClassMembers)
if not new_members:
dialog = Gtk.MessageDialog(
flags=0,
message_type=Gtk.MessageType.WARNING,
buttons=Gtk.ButtonsType.NONE,
text=_("This will delete this class!"))
dialog.format_secondary_text(_("Proceed anyway?"))
dialog.add_buttons (Gtk.STOCK_CANCEL, Gtk.ResponseType.NO,
Gtk.STOCK_DELETE, Gtk.ResponseType.YES)
result = dialog.run()
dialog.destroy()
if result==Gtk.ResponseType.NO:
self.cups._end_operation ()
return True
class_deleted = True
# update member list
old_members = printer.class_members[:]
for member in new_members:
if member in old_members:
old_members.remove(member)
else:
self.cups.addPrinterToClass(member, name)
for member in old_members:
self.cups.deletePrinterFromClass(member, name)
location = self.entPLocation.get_text()
info = self.entPDescription.get_text()
device_uri = self.entPDevice.get_text()
enabled = self.chkPEnabled.get_active()
accepting = self.chkPAccepting.get_active()
shared = self.chkPShared.get_active()
if info!=printer.info or saveall:
self.cups.setPrinterInfo(name, info)
if location!=printer.location or saveall:
self.cups.setPrinterLocation(name, location)
if (not printer.is_class and
(device_uri!=printer.device_uri or saveall)):
self.cups.setPrinterDevice(name, device_uri)
if enabled != printer.enabled or saveall:
printer.setEnabled(enabled)
if accepting == printer.rejecting or saveall:
printer.setAccepting(accepting)
if shared != printer.is_shared or saveall:
printer.setShared(shared)
def get_combo_value (cmb):
model = cmb.get_model ()
iter = cmb.get_active_iter ()
return model.get_value (iter, 1)
job_sheet_start = get_combo_value (self.cmbPStartBanner)
job_sheet_end = get_combo_value (self.cmbPEndBanner)
error_policy = get_combo_value (self.cmbPErrorPolicy)
op_policy = get_combo_value (self.cmbPOperationPolicy)
if (job_sheet_start != printer.job_sheet_start or
job_sheet_end != printer.job_sheet_end) or saveall:
printer.setJobSheets(job_sheet_start, job_sheet_end)
if error_policy != printer.error_policy or saveall:
printer.setErrorPolicy(error_policy)
if op_policy != printer.op_policy or saveall:
printer.setOperationPolicy(op_policy)
default_allow = self.rbtnPAllow.get_active()
except_users = self.getPUsers()
if (default_allow != printer.default_allow or
except_users != printer.except_users) or saveall:
printer.setAccess(default_allow, except_users)
for option in printer.attributes:
if option not in self.server_side_options:
printer.unsetOption(option)
for option in self.server_side_options.values():
if (option.is_changed() or
(saveall and
option.get_current_value () != option.get_default())):
debugprint ("Set %s = %s" % (option.name,
option.get_current_value()))
printer.setOption(option.name, option.get_current_value())
except cups.IPPError as e:
(e, s) = e.args
show_IPP_Error(e, s, parent)
self.cups._end_operation ()
return True
self.cups._end_operation ()
self.changed = set() # of options
if not self.cups._use_pk and "server_settings" not in self.__dict__:
# We can authenticate with the server correctly at this point,
# but we have never fetched the server settings to see whether
# the server is publishing shared printers. Fetch the settings
# now so that we can update the "not published" label if necessary.
self.cups._begin_operation (_("fetching server settings"))
try:
self.server_settings = self.cups.adminGetServerSettings()
except:
nonfatalException()
self.cups._end_operation ()
if not class_deleted:
# Update our copy of the printer's settings.
try:
printer.getAttributes ()
self.updatePrinterProperties ()
except cups.IPPError:
pass
self._monitor.update ()
return False
def getPrinterSettings(self):
#self.ppd.markDefaults()
for option in self.options.values():
option.writeback()
### Printer Properties tree view signal handlers
def on_tvPrinterProperties_selection_changed (self, selection):
# Prevent selection from being de-selected.
(model, iter) = selection.get_selected ()
if iter:
self.printer_properties_last_iter_selected = iter
else:
try:
iter = self.printer_properties_last_iter_selected
except AttributeError:
# Not set yet.
return
if model.iter_is_valid (iter):
selection.select_iter (iter)
def on_tvPrinterProperties_cursor_changed (self, treeview):
# Adjust notebook to reflect selected item.
(path, column) = treeview.get_cursor ()
if path is not None:
model = treeview.get_model ()
iter = model.get_iter (path)
n = model.get_value (iter, 1)
self.ntbkPrinter.set_current_page (n)
# print test page
def printTestPage (self):
self.btnPrintTestPage.clicked ()
def on_btnPrintTestPage_clicked(self, button):
printer = self.printer
if not printer:
# Printer has been deleted meanwhile
return
# if we have a page size specific custom test page, use it;
# otherwise use cups' default one
custom_testpage = None
if self.ppd != False:
opt = self.ppd.findOption ("PageSize")
if opt:
custom_testpage = os.path.join(pkgdata,
'testpage-%s.ps' %
opt.defchoice.lower())
# Connect as the current user so that the test page can be managed
# as a normal job.
user = cups.getUser ()
cups.setUser ('')
try:
c = authconn.Connection (self.parent, try_as_root=False,
host=self._host,
encryption=self._encryption)
except RuntimeError as e:
show_IPP_Error (None, e, self.parent)
return
job_id = None
c._begin_operation (_("printing test page"))
try:
if custom_testpage and os.path.exists(custom_testpage):
debugprint ('Printing custom test page ' + custom_testpage)
job_id = c.printTestPage(printer.name,
file=custom_testpage)
else:
debugprint ('Printing default test page')
job_id = c.printTestPage(printer.name)
except cups.IPPError as e:
(e, msg) = e.args
if (e == cups.IPP_NOT_AUTHORIZED and
self._host != 'localhost' and
self._host[0] != '/'):
show_error_dialog (_("Not possible"),
_("The remote server did not accept "
"the print job, most likely "
"because the printer is not "
"shared."),
self.parent)
else:
show_IPP_Error(e, msg, self.parent)
c._end_operation ()
cups.setUser (user)
if job_id is not None:
show_info_dialog (_("Submitted"),
_("Test page submitted as job %d") % job_id,
parent=self.parent)
def maintenance_command (self, command):
printer = self.printer
if not printer:
# Printer has been deleted meanwhile
return
with tempfile.NamedTemporaryFile(mode='wt') as tmpfile:
tmpfile.write ("#CUPS-COMMAND\n%s\n" % command)
tmpfile.flush()
self.cups._begin_operation (_("sending maintenance command"))
try:
format = "application/vnd.cups-command"
job_id = self.cups.printTestPage (printer.name,
format=format,
file=tmpfile.name,
user=cups.getUser ())
show_info_dialog (_("Submitted"),
_("Maintenance command submitted as "
"job %d") % job_id,
parent=self.parent)
except cups.IPPError as e:
(e, msg) = e.args
if (e == cups.IPP_NOT_AUTHORIZED and
self.printer.name != 'localhost'):
show_error_dialog (_("Not possible"),
_("The remote server did not accept "
"the print job, most likely "
"because the printer is not "
"shared."),
self.parent)
else:
show_IPP_Error(e, msg, self.parent)
self.cups._end_operation ()
def on_btnSelfTest_clicked(self, button):
self.maintenance_command ("PrintSelfTestPage")
def on_btnCleanHeads_clicked(self, button):
self.maintenance_command ("Clean all")
def fillComboBox(self, combobox, values, value, translationdict=None):
if translationdict is None:
translationdict = ppdippstr.TranslationDict ()
model = Gtk.ListStore (str,
str)
combobox.set_model (model)
set_active = False
for nr, val in enumerate(values):
model.append ([(translationdict.get (val)), val])
if val == value:
combobox.set_active(nr)
set_active = True
if not set_active:
combobox.set_active (0)
def load (self, name, host=None, encryption=None, parent=None):
self.changed = set() # of options
self.options = {} # keyword -> Option object
self.conflicts = set() # of options
if not host:
host = cups.getServer()
if not encryption:
encryption = cups.getEncryption ()
c = authconn.Connection (parent=self.dialog,
host=host,
encryption=encryption)
self.cups = c
printer = cupshelpers.Printer (name, self.cups)
self.printer = printer
try:
# CUPS 1.4
publishing = printer.other_attributes['server-is-sharing-printers']
self.server_is_publishing = publishing
except KeyError:
pass
editable = not self.printer.discovered
try:
self.ppd = printer.getPPD()
self.ppd_local = printer.getPPD()
if self.ppd_local != False:
self.ppd_local.localize()
except cups.IPPError as e:
(e, m) = e.args
# We might get IPP_INTERNAL_ERROR if this is a memberless
# class.
if e != cups.IPP_INTERNAL_ERROR:
# Some IPP error other than IPP_NOT_FOUND.
show_IPP_Error(e, m, self.parent)
if e in [cups.IPP_SERVICE_UNAVAILABLE,
cups.IPP_INTERNAL_ERROR]:
show_dialog(_("Raw Queue"),
_("Unable to get queue details. Treating queue "
"as raw."),
Gtk.MessageType.ERROR,
self.parent)
# Treat it as a raw queue.
self.ppd = False
except RuntimeError as e:
# Either the underlying cupsGetPPD2() function returned
# NULL without setting an IPP error (so it'll be something
# like a failed connection), or the PPD could not be parsed.
if str (e).startswith ("ppd"):
show_error_dialog (_("Error"),
_("The PPD file for this queue "
"is damaged."),
self.parent)
else:
show_error_dialog (_("Error"),
_("There was a problem connecting to "
"the CUPS server."),
self.parent)
raise
for widget in (self.entPDescription, self.entPLocation,
self.entPDevice):
widget.set_editable(editable)
for widget in (self.btnSelectDevice, self.btnChangePPD,
self.chkPEnabled, self.chkPAccepting, self.chkPShared,
self.cmbPStartBanner, self.cmbPEndBanner,
self.cmbPErrorPolicy, self.cmbPOperationPolicy,
self.rbtnPAllow, self.rbtnPDeny, self.tvPUsers,
self.entPUser, self.btnPAddUser, self.btnPDelUser):
widget.set_sensitive(editable)
# Description page
self.entPDescription.set_text(printer.info)
self.entPLocation.set_text(printer.location)
uri = printer.device_uri
self.entPDevice.set_text(uri)
self.changed.discard(self.entPDevice)
# Hide make/model and Device URI for classes
for widget in (self.lblPMakeModel2, self.entPMakeModel,
self.btnChangePPD, self.lblPDevice2,
self.entPDevice, self.btnSelectDevice):
if printer.is_class:
widget.hide()
else:
widget.show()
# Policy tab
# ----------
try:
if printer.is_shared:
if self.server_is_publishing:
self.lblNotPublished.hide()
else:
self.lblNotPublished.show_all ()
else:
self.lblNotPublished.hide()
except:
nonfatalException()
self.lblNotPublished.hide()
# Job sheets
self.cmbPStartBanner.set_sensitive(editable)
self.cmbPEndBanner.set_sensitive(editable)
# Policies
self.cmbPErrorPolicy.set_sensitive(editable)
self.cmbPOperationPolicy.set_sensitive(editable)
# Access control
self.entPUser.set_text("")
# Server side options (Job options)
self.server_side_options = {}
for option in self.job_options_widgets.values ():
if option.name == "media" and self.ppd:
# Slightly special case because the 'system default'
# (i.e. what you get when you press Reset) depends
# on the printer's PageSize.
opt = self.ppd.findOption ("PageSize")
if opt:
option.set_default (opt.defchoice)
option_editable = editable
try:
value = self.printer.attributes[option.name]
except KeyError:
option.reinit (None)
else:
try:
if option.name in self.printer.possible_attributes:
supported = self.printer.\
possible_attributes[option.name][1]
# Set the option widget.
# In CUPS 1.3.x the orientation-requested-default
# attribute may have the value None; this means there
# is no value set. This suits our needs here, as None
# resets the option to the system default and makes the
# Reset button insensitive.
option.reinit (value, supported=supported)
else:
option.reinit (value)
self.server_side_options[option.name] = option
except:
nonfatalException()
option_editable = False
debugprint ("Option '%s' has value '%s' and cannot be edited." % (option.name, value))
option.widget.set_sensitive (option_editable)
if not editable:
option.button.set_sensitive (False)
self.other_job_options = []
self.draw_other_job_options (editable=editable)
for option in self.printer.attributes.keys ():
if option in self.server_side_options:
continue
if option == "output-mode" or option == "media-col":
# Not settable
continue
value = self.printer.attributes[option]
if option in self.printer.possible_attributes:
supported = self.printer.possible_attributes[option][1]
else:
if isinstance (value, bool):
supported = ["true", "false"]
value = str (value).lower ()
else:
supported = ""
value = str (value)
self.add_job_option (option, value=value,
supported=supported, is_new=False,
editable=editable)
self.entNewJobOption.set_text ('')
self.entNewJobOption.set_sensitive (editable)
self.btnNewJobOption.set_sensitive (False)
if printer.is_class:
# remove InstallOptions tab
tab_nr = self.ntbkPrinter.page_num(self.swPInstallOptions)
if tab_nr != -1:
self.ntbkPrinter.remove_page(tab_nr)
self.fillClassMembers(editable)
else:
# real Printer
self.fillPrinterOptions(name, editable)
self.updateMarkerLevels()
self.updateStateReasons()
self.updatePrinterPropertiesTreeView()
self.changed = set() # of options
self.updatePrinterProperties ()
self.setDataButtonState()
def updatePrinterPropertiesTreeView (self):
# Now update the tree view (which we use instead of the notebook tabs).
store = Gtk.ListStore (str, int)
self.ntbkPrinter.set_show_tabs (False)
for n in range (self.ntbkPrinter.get_n_pages ()):
page = self.ntbkPrinter.get_nth_page (n)
label = self.ntbkPrinter.get_tab_label (page)
iter = store.append (None)
store.set_value (iter, 0, label.get_text ())
store.set_value (iter, 1, n)
sel = self.tvPrinterProperties.get_selection ()
self.tvPrinterProperties.set_model (store)
def updateMarkerLevels (self):
printer = self.printer
if not printer:
# Printer has been deleted meanwhile
return
# Marker levels
for widget in self.vboxMarkerLevels.get_children ():
self.vboxMarkerLevels.remove (widget)
marker_info = dict()
num_markers = 0
for (attr, typ) in [('marker-colors', str),
('marker-names', str),
('marker-types', str),
('marker-levels', float)]:
val = printer.other_attributes.get (attr, [])
if typ != str and len (val) > 0:
try:
# Can the value be coerced into the right type?
typ (val[0])
except TypeError as s:
debugprint ("%s value not coercible to %s: %s" %
(attr, typ, s))
val = [0.0 for x in val]
marker_info[attr] = [0.0 if (typ != str and x < 0) \
else x for x in val]
if num_markers == 0 or len (val) < num_markers:
num_markers = len (val)
for attr in ['marker-colors', 'marker-names',
'marker-types', 'marker-levels']:
if len (marker_info[attr]) > num_markers:
debugprint ("Trimming %s from %s" %
(marker_info[attr][num_markers:], attr))
del marker_info[attr][num_markers:]
markers = list(map (lambda color, name, type, level:
(color, name, type, level),
marker_info['marker-colors'],
marker_info['marker-names'],
marker_info['marker-types'],
marker_info['marker-levels']))
debugprint (markers)
can_refresh = (printer.type & cups.CUPS_PRINTER_COMMANDS) != 0
if can_refresh:
self.btnRefreshMarkerLevels.show ()
else:
self.btnRefreshMarkerLevels.hide ()
if len (markers) == 0:
label = Gtk.Label(label=_("Marker levels are not reported "
"for this printer."))
label.set_line_wrap (True)
label.set_alignment (0.0, 0.0)
self.vboxMarkerLevels.pack_start (label, False, False, 0)
else:
num_markers = 0
cols = len (markers)
rows = 1 + (cols - 1) / 4
if cols > 4:
cols = 4
grid = Gtk.Grid()
grid.set_column_homogeneous(True)
grid.set_row_homogeneous(True)
grid.set_column_spacing (6)
grid.set_row_spacing (12)
self.vboxMarkerLevels.pack_start (grid, False, False, 0)
for color, name, marker_type, level in markers:
if name is None:
name = ''
elif self.ppd != False:
localized_name = self.ppd.localizeMarkerName(name)
if localized_name is not None:
name = localized_name
row = num_markers / 4
col = num_markers % 4
vbox = Gtk.Box (spacing=6)
subhbox = Gtk.Box ()
inklevel = gtkinklevel.GtkInkLevel (color, level)
inklevel.set_tooltip_text ("%d%%" % level)
subhbox.pack_start (inklevel, True, False, 0)
vbox.pack_start (subhbox, False, False, 0)
label = Gtk.Label(label=name)
label.set_width_chars (10)
label.set_line_wrap (True)
vbox.pack_start (label, False, False, 0)
grid.attach (vbox, col, row, 1, 1)
num_markers += 1
self.vboxMarkerLevels.show_all ()
def on_btnRefreshMarkerLevels_clicked (self, button):
self.maintenance_command ("ReportLevels")
def updateStateReasons (self):
printer = self.printer
reasons = printer.other_attributes.get ('printer-state-reasons', [])
store = Gtk.ListStore (str, str)
any = False
for reason in reasons:
if reason == "none":
break
any = True
iter = store.append (None)
r = statereason.StateReason (printer.name, reason, self._ppdcache)
if r.get_reason () == "paused":
icon = Gtk.STOCK_MEDIA_PAUSE
else:
icon = statereason.StateReason.LEVEL_ICON[r.get_level ()]
store.set_value (iter, 0, icon)
(title, text) = r.get_description ()
store.set_value (iter, 1, text)
self.tvPrinterStateReasons.set_model (store)
page = 0
if any:
page = 1
self.ntbkPrinterStateReasons.set_current_page (page)
def set_printer_state_reason_icon (self, column, cell, model, iter, *data):
icon = model.get_value (iter, 0)
theme = Gtk.IconTheme.get_default ()
try:
pixbuf = theme.load_icon (icon, 22, 0)
cell.set_property ("pixbuf", pixbuf)
except GLib.GError:
pass # Couldn't load icon
def set_printer_state_reason_text (self, column, cell, model, iter, *data):
cell.set_property ("text", model.get_value (iter, 1))
def updatePrinterProperties(self):
debugprint ("update printer properties")
printer = self.printer
self.entPDevice.set_text(printer.device_uri)
self.entPMakeModel.set_text(printer.make_and_model)
state = self.printer_states.get (printer.state,
_("Unknown"))
reason = printer.other_attributes.get ('printer-state-message', '')
if len (reason) > 0:
state += ' - ' + reason
self.entPState.set_text(state)
if len (self.changed) == 0:
debugprint ("no changes yet: full printer properties update")
# State
self.chkPEnabled.set_active(printer.enabled)
self.chkPAccepting.set_active(not printer.rejecting)
self.chkPShared.set_active(printer.is_shared)
# Job sheets
self.fillComboBox(self.cmbPStartBanner,
printer.job_sheets_supported,
printer.job_sheet_start,
ppdippstr.job_sheets)
self.fillComboBox(self.cmbPEndBanner, printer.job_sheets_supported,
printer.job_sheet_end,
ppdippstr.job_sheets)
# Policies
self.fillComboBox(self.cmbPErrorPolicy,
printer.error_policy_supported,
printer.error_policy,
ppdippstr.printer_error_policy)
self.fillComboBox(self.cmbPOperationPolicy,
printer.op_policy_supported,
printer.op_policy,
ppdippstr.printer_op_policy)
# Access control
self.rbtnPAllow.set_active(printer.default_allow)
self.rbtnPDeny.set_active(not printer.default_allow)
self.setPUsers(printer.except_users)
# Marker levels
self.updateMarkerLevels ()
self.updateStateReasons ()
self.updatePrinterPropertiesTreeView ()
def fillPrinterOptions(self, name, editable):
# remove Class membership tab
tab_nr = self.ntbkPrinter.page_num(self.vbClassMembers)
if tab_nr != -1:
self.ntbkPrinter.remove_page(tab_nr)
# clean Installable Options Tab
for widget in self.vbPInstallOptions.get_children():
self.vbPInstallOptions.remove(widget)
# clean Options Tab
for widget in self.vbPOptions.get_children():
self.vbPOptions.remove(widget)
# insert Options Tab
if self.ntbkPrinter.page_num(self.swPOptions) == -1:
self.ntbkPrinter.insert_page(
self.swPOptions, self.lblPOptions, self.static_tabs)
if not self.ppd:
tab_nr = self.ntbkPrinter.page_num(self.swPInstallOptions)
if tab_nr != -1:
self.ntbkPrinter.remove_page(tab_nr)
tab_nr = self.ntbkPrinter.page_num(self.swPOptions)
if tab_nr != -1:
self.ntbkPrinter.remove_page(tab_nr)
return
ppd = self.ppd
ppd.markDefaults()
self.ppd_local.markDefaults()
hasInstallableOptions = False
# build option tabs
for group in self.ppd_local.optionGroups:
if group.name == "InstallableOptions":
hasInstallableOptions = True
container = self.vbPInstallOptions
tab_nr = self.ntbkPrinter.page_num(self.swPInstallOptions)
if tab_nr == -1:
self.ntbkPrinter.insert_page(self.swPInstallOptions,
Gtk.Label(label=group.text),
self.static_tabs)
tab_label = self.lblPInstallOptions
else:
group_name = ppdippstr.ppd.get (group.text)
frame = Gtk.Frame(label="<b>%s</b>" % html.escape (group_name))
frame.get_label_widget().set_use_markup(True)
frame.set_shadow_type (Gtk.ShadowType.NONE)
self.vbPOptions.pack_start (frame, False, False, 0)
container = Gtk.Alignment.new(0.5, 0.5, 1.0, 1.0)
# We want a left padding of 12, but there is a Table with
# spacing 6, and the left-most column of it (the conflict
# icon) is normally hidden, so just use 6 here.
container.set_padding (6, 12, 6, 0)
frame.add (container)
tab_label = self.lblPOptions
grid = Gtk.Grid()
grid.set_column_spacing(6)
grid.set_row_spacing(6)
container.add(grid)
rows = 0
# InputSlot and ManualFeed need special handling. With
# libcups, if ManualFeed is True, InputSlot gets unset.
# Likewise, if InputSlot is set, ManualFeed becomes False.
# We handle it by toggling the sensitivity of InputSlot
# based on ManualFeed.
self.option_inputslot = self.option_manualfeed = None
for nr, option in enumerate(group.options):
if option.keyword == "PageRegion":
continue
rows += 1
o = OptionWidget(option, ppd, self, tab_label=tab_label)
grid.attach(o.conflictIcon, 0, nr, 1, 1)
hbox = Gtk.Box()
if o.label:
a = Gtk.Alignment.new(0.5, 0.5, 1.0, 1.0)
a.set_padding (0, 0, 0, 6)
a.add (o.label)
grid.attach(a, 1, nr, 1, 1)
grid.attach(hbox, 2, nr, 1, 1)
else:
grid.attach(hbox, 1, nr, 2, 1)
hbox.pack_start(o.selector, False, False, 0)
self.options[option.keyword] = o
o.selector.set_sensitive(editable)
if option.keyword == "InputSlot":
self.option_inputslot = o
elif option.keyword == "ManualFeed":
self.option_manualfeed = o
# remove Installable Options tab if not needed
if not hasInstallableOptions:
tab_nr = self.ntbkPrinter.page_num(self.swPInstallOptions)
if tab_nr != -1:
self.ntbkPrinter.remove_page(tab_nr)
# check for conflicts
for option in self.options.values():
conflicts = option.checkConflicts()
if conflicts:
self.conflicts.add(option)
self.swPInstallOptions.show_all()
self.swPOptions.show_all()
# Class members
def fillClassMembers(self, editable):
self.btnClassAddMember.set_sensitive(editable)
self.btnClassDelMember.set_sensitive(editable)
# remove Options tab
tab_nr = self.ntbkPrinter.page_num(self.swPOptions)
if tab_nr != -1:
self.ntbkPrinter.remove_page(tab_nr)
# insert Member Tab
if self.ntbkPrinter.page_num(self.vbClassMembers) == -1:
self.ntbkPrinter.insert_page(
self.vbClassMembers, self.lblClassMembers,
self.static_tabs)
model_members = self.tvClassMembers.get_model()
model_not_members = self.tvClassNotMembers.get_model()
model_members.clear()
model_not_members.clear()
names = list (self._monitor.get_printers ())
names.sort ()
for name in names:
if name != self.printer.name:
if name in self.printer.class_members:
model_members.append((name, ))
elif not self.printer.type & cups.CUPS_PRINTER_CLASS:
model_not_members.append((name, ))
def on_btnClassAddMember_clicked(self, button):
newprinter.moveClassMembers(self.tvClassNotMembers,
self.tvClassMembers)
if newprinter.getCurrentClassMembers(self.tvClassMembers) != self.printer.class_members:
self.changed.add(self.tvClassMembers)
else:
self.changed.discard(self.tvClassMembers)
self.setDataButtonState()
def on_btnClassDelMember_clicked(self, button):
newprinter.moveClassMembers(self.tvClassMembers,
self.tvClassNotMembers)
if newprinter.getCurrentClassMembers(self.tvClassMembers) != self.printer.class_members:
self.changed.add(self.tvClassMembers)
else:
self.changed.discard(self.tvClassMembers)
self.setDataButtonState()
def sensitise_new_printer_widgets (self, sensitive=True):
sensitive = (sensitive and
self.printer is not None and
not (self.printer.discovered or
bool (self.changed)))
for button in [self.btnChangePPD,
self.btnSelectDevice]:
button.set_sensitive (sensitive)
def desensitise_new_printer_widgets (self):
self.sensitise_new_printer_widgets (False)
# change device
def on_btnSelectDevice_clicked(self, button):
busy (self.dialog)
self.desensitise_new_printer_widgets ()
if not self.newPrinterGUI.init("device", device_uri=self.printer.device_uri,
name=self.printer.name,
host=self._host,
encryption=self._encryption,
parent=self.dialog):
self.sensitise_new_printer_widgets ()
ready (self.dialog)
# change PPD
def on_btnChangePPD_clicked(self, button):
busy (self.dialog)
self.desensitise_new_printer_widgets ()
if not self.newPrinterGUI.init("ppd", device_uri=self.printer.device_uri,
ppd=self.ppd,
name=self.printer.name,
host=self._host,
encryption=self._encryption,
parent=self.dialog):
self.sensitise_new_printer_widgets ()
ready (self.dialog)
# NewPrinterGUI signal handlers
def on_printer_modified (self, obj, name, ppd_has_changed):
debugprint ("on_printer_modified called")
self.sensitise_new_printer_widgets ()
if self.dialog.get_property ('visible') and self.printer:
try:
self.printer.getAttributes ()
if ppd_has_changed:
self.load (name)
else:
self.updatePrinterProperties ()
except cups.IPPError:
pass
def on_printer_not_modified (self, obj):
self.sensitise_new_printer_widgets ()
# Monitor signal handlers
def on_printer_event (self, mon, printer, eventname, event):
self.on_printer_modified (None, printer, False)
def on_printer_removed (self, mon, printer):
if (self.dialog.get_property ('visible') and
self.printer and self.printer.name == printer):
self.dialog.response (Gtk.ResponseType.CANCEL)
if self.printer and self.printer.name == printer:
self.printer = None
def on_state_reason_added (self, mon, reason):
if (self.dialog.get_property ('visible') and
self.printer and self.printer.name == reason.get_printer ()):
try:
self.printer.getAttributes ()
self.updatePrinterProperties ()
except cups.IPPError:
pass
def on_state_reason_removed (self, mon, reason):
if (self.dialog.get_property ('visible') and
self.printer and self.printer.name == reason.get_printer ()):
try:
self.printer.getAttributes ()
self.updatePrinterProperties ()
except cups.IPPError:
pass
def on_cups_connection_error (self, mon):
# FIXME: figure out how to handle this
pass
if __name__ == '__main__':
import sys
if len (sys.argv) < 2:
print("Specify queue name")
sys.exit (1)
set_debugging (True)
os.environ["SYSTEM_CONFIG_PRINTER_UI"] = "ui"
locale.setlocale (locale.LC_ALL, "")
ppdippstr.init ()
loop = GObject.MainLoop ()
def on_dialog_closed (obj):
obj.destroy ()
loop.quit ()
properties = PrinterPropertiesDialog ()
properties.connect ('dialog-closed', on_dialog_closed)
properties.show (sys.argv[1])
loop.run ()