qubes-installer-qubes-os/firstboot/firstboot/interface.py

505 lines
20 KiB
Python

#
# Chris Lumens <clumens@redhat.com>
#
# Copyright 2007 Red Hat, Inc.
#
# This copyrighted material is made available to anyone wishing to use, modify,
# copy, or redistribute it subject to the terms and conditions of the GNU
# General Public License v.2. This program is distributed in the hope that it
# will be useful, but WITHOUT ANY WARRANTY expressed or implied, including the
# implied warranties 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. Any Red Hat
# trademarks that are incorporated in the source code or documentation are not
# subject to the GNU General Public License and may only be used or replicated
# with the express permission of Red Hat, Inc.
#
import gtk
import logging, os
from firstboot.config import *
from firstboot.constants import *
from firstboot.functions import *
from firstboot.moduleset import *
import gettext
_ = lambda x: gettext.ldgettext("firstboot", x)
class Control:
def __init__(self):
self.currentPage = 0
self.history = []
self.moduleList = []
class Interface(object):
def __init__(self, autoscreenshot=False, moduleList=[], testing=False):
"""Create a new Interface instance. Instance attributes:
autoscreenshot -- If true, a screenshot will be taken after every
module is run.
moduleList -- A list of references to all the loaded modules.
This is not typically used, but is required to
make the interface work.
testing -- Is firstboot running under testing mode, where no
changes will be made to the disk?
"""
self._screenshotDir = "/root/firstboot-screenshots"
self._screenshotIndex = 0
# This is needed for ModuleSet to work. We maintain a stack of control
# states, creating a new state for navigation when we enter into a
# ModuleSet, then popping it off when we leave.
self._controlStack = [Control()]
self._control = self._controlStack[0]
self.moduleList = moduleList
self._x_size = gtk.gdk.screen_width()
self._y_size = gtk.gdk.screen_height()
self.autoscreenshot = autoscreenshot
self.testing = testing
def _setModuleList(self, moduleList):
self._control.moduleList = moduleList
moduleList = property(lambda s: s._control.moduleList,
lambda s, v: s._setModuleList(v))
def _backClicked(self, *args):
# If there's nowhere to go back to, we're either at the first page in
# the module set or something went wrong (Back is enabled on the very
# first page). In the former case, revert back to the enclosing
# control state for what page to display next.
if len(self._control.history) == 0:
if len(self._controlStack) == 1:
logging.error(_("Attempted to go back, but history is empty."))
return
else:
self._controlStack.pop()
self._control = self._controlStack[-1]
# If we were previously on the last page, we need to set the Next
# button's label back to normal.
if self.nextButton.get_label() == _("_Finish"):
self.nextButton.set_label("gtk-go-forward")
self._control.currentPage = self._control.history.pop()
self.moveToPage(pageNum=self._control.currentPage)
def _keyRelease(self, window, event):
if event.keyval == gtk.keysyms.F12:
self.nextButton.clicked()
elif event.keyval == gtk.keysyms.F11:
self.backButton.clicked()
elif event.keyval == gtk.keysyms.Print and event.state & gtk.gdk.SHIFT_MASK:
self.takeScreenshot()
def _nextClicked(self, *args):
if self.autoscreenshot:
self.takeScreenshot()
self.advance()
def _setBackSensitivity(self):
self.backButton.set_sensitive(not(self._control.currentPage == 0 and len(self._controlStack) == 1))
def _setPointer(self, number):
# The sidebar pointer only works in terms of the top-level module list
# as we don't display anything on the side for a ModuleSet and making
# the pointer move around then would be confusing.
for i in range(len(self.sidebar.get_children())):
(alignment, label) = self.sidebar.get_children()[i].get_children()
pix = alignment.get_children()[0]
if i == number:
f = "%s/%s" % (config.themeDir, "pointer-white.png")
if os.access(f, os.F_OK):
pix.set_from_file(f)
else:
pix.set_from_file("%s/%s" % (config.defaultThemeDir, "pointer-white.png"))
else:
f = "%s/%s" % (config.themeDir, "pointer-blank.png")
if os.access(f, os.F_OK):
pix.set_from_file(f)
else:
pix.set_from_file("%s/%s" % (config.defaultThemeDir, "pointer-blank.png"))
def _sidebarExposed(self, eb, event):
pixbuf = self.sidebarBg.scale_simple(int(self._y_size * self.aspectRatio),
self._y_size, gtk.gdk.INTERP_BILINEAR)
cairo_context = eb.window.cairo_create()
cairo_context.set_source_pixbuf(pixbuf, 0, self._y_size-pixbuf.get_height())
cairo_context.paint()
return False
def advance(self):
"""Call the apply method on the currently displayed page, add it to
the history, and move to the next page. It is not safe to call this
method from within firstboot modules.
"""
module = self.moduleList[self._control.currentPage]
# This could fail, in which case the exception will propagate up to the
# interface which will know the proper way to handle it.
result = module.apply(self, self.testing)
# If something went wrong in the module, don't advance.
if result == RESULT_FAILURE:
return
# If the apply action from the current page jumped us to another page,
# don't try to jump again.
if result != RESULT_JUMP:
self.moveToPage(pageNum=self._control.currentPage+1)
# if we are on the last page overall (not just the last page of a
# ModuleSet), it's time to kill the interface.
if len(self._controlStack) == 1:
if self._control.currentPage == len(self.moduleList)-1:
self.nextButton.set_label(_("_Finish"))
elif self._control.currentPage == len(self.moduleList):
self.checkReboot()
self.destroy()
def checkReboot(self):
"""Check to see if any module requires a reboot for changes to take
effect, displaying a dialog if so. This method immediately reboots
the system.
"""
needReboot = False
for module in self.moduleList:
if module.needsReboot():
needReboot = True
break
if not needReboot or self.testing:
return
dlg = gtk.MessageDialog(None, 0, gtk.MESSAGE_INFO, gtk.BUTTONS_OK,
_("The system must now reboot for some of your selections to take effect."))
dlg.set_position(gtk.WIN_POS_CENTER)
dlg.show_all()
dlg.run()
dlg.destroy()
os.system("/sbin/reboot")
def fit_window_to_screen(self):
# need this to get the monitor
self.win.show()
screen = self.win.get_screen()
monitor = screen.get_monitor_at_window(self.win.get_window())
geometry = screen.get_monitor_geometry(monitor)
self._x_size = geometry.width
self._y_size = geometry.height
logging.info("Setting size to %sx%s" % (self._x_size, self._y_size))
self.win.set_size_request(self._x_size, self._y_size)
def createMainWindow(self):
"""Create and initialize the main window. This includes switching to
fullscreen mode if necessary, adding buttons, displaying artwork,
and other functions. This method should also create the UI elements
that make up the sidebar. This method returns a gtk.Window.
"""
# Create the initial window and a vbox to fill it with.
self.win = gtk.Window()
self.win.set_position(gtk.WIN_POS_NONE)
self.win.set_decorated(False)
# we don't set border width here so that the sidebar will meet
# the edge of the screen
self.fit_window_to_screen()
# Create a box that will hold all other widgets.
self.mainHBox = gtk.HBox(False, 10)
# Create the sidebar box.
self.sidebar = gtk.VBox()
self.sidebar.set_border_width(24)
# Load this background now so we can figure out how big to make
# the left side.
try:
self.sidebarBg = loadPixbuf("%s/%s" % (config.themeDir, "firstboot-left.png"))
except:
self.sidebarBg = loadPixbuf("%s/%s" % (config.defaultThemeDir, "firstboot-left.png"))
self.aspectRatio = (1.0 * self.sidebarBg.get_width()) / (1.0 * self.sidebarBg.get_height())
# leftEventBox exists only so we have somewhere to paint an image.
self.leftEventBox = gtk.EventBox()
self.leftEventBox.add(self.sidebar)
self.sidebar.connect("expose-event", self._sidebarExposed)
# Create the box for the right side of the screen. This holds the
# display for the current module and the button box.
self.rightBox = gtk.VBox()
self.rightBox.set_border_width(24)
leftWidth = int(self._y_size * self.aspectRatio)
self.leftEventBox.set_size_request(leftWidth, self._y_size)
self.win.fullscreen()
# Create a button box to handle navigation.
self.buttonBox = gtk.HButtonBox()
self.buttonBox.set_layout(gtk.BUTTONBOX_END)
self.buttonBox.set_spacing(10)
self.buttonBox.set_border_width(10)
# Create the Back button, marking it insensitive by default since we
# start at the first page.
self.backButton = gtk.Button(use_underline=True, stock="gtk-go-back",
label=_("_Back"))
self._setBackSensitivity()
self.backButton.connect("clicked", self._backClicked)
self.buttonBox.pack_start(self.backButton)
# Create the Forward button.
self.nextButton = gtk.Button(use_underline=True, stock="gtk-go-forward",
label=_("_Forward"))
self.nextButton.connect("clicked", self._nextClicked)
self.buttonBox.pack_start(self.nextButton)
# Add the widgets into the right side.
self.rightBox.pack_end(self.buttonBox, expand=False)
# Add the widgets into the main hbox widget.
self.mainHBox.pack_start(self.leftEventBox, expand=False, fill=False)
self.mainHBox.pack_start(self.rightBox, expand=True, fill=True)
self.win.add(self.mainHBox)
self.win.connect("destroy", self.destroy)
self.win.connect("key-release-event", self._keyRelease)
return self.win
def createScreens(self):
"""Call the createScreen method on all loaded modules. This loads the
UI elements for each page and stuffs the rendered page into a UI
wrapper containing the module's title and icon.
"""
loaded_modules = []
for module in self.moduleList:
try:
module.createScreen()
except Exception as e:
logging.error(_("Module %s raised an exception while loading: %s" % (module.title, e)))
continue
if isinstance(module, Module) and module.vbox is None:
logging.error(_("Module %s did not set up its UI properly.") % module.title)
continue
try:
module.renderModule(self)
except Exception as e:
logging.error(_("Module %s raised an exception while rendering: %s" % (module.title, e)))
continue
loaded_modules.append(module)
self.moduleList = loaded_modules
def createSidebar(self):
"""Add the sidebarTitle from every module to the sidebar."""
for module in self.moduleList:
hbox = gtk.HBox(False, 5)
label = gtk.Label("")
label.set_markup("<span foreground='#FFFFFF'><b>%s</b></span>" % _(module.sidebarTitle))
label.set_alignment(0.0, 0.5)
# Wrap the sidebar title if it's too long.
label.set_line_wrap(True)
(w, h) = self.leftEventBox.get_size_request()
label.set_size_request((int)(w*0.7), -1)
# Make sure the arrow is at the top of any wrapped line.
alignment = gtk.Alignment(yalign=0.2)
try:
alignment.add(loadToImage("%s/%s" % (config.themeDir, "pointer-blank.png")))
except:
alignment.add(loadToImage("%s/%s" % (config.defaultThemeDir, "pointer-blank.png")))
hbox.pack_start(alignment, False)
hbox.pack_end(label, True)
self.sidebar.pack_start(hbox, False, True, 3)
# Initialize sidebar pointer
self._setPointer(0)
def destroy(self, *args):
"""Destroy the UI, but do not take any other action to quit firstboot."""
try:
gtk.main_quit()
except RuntimeError:
os._exit(1)
def displayModule(self):
"""Display the current module on the main portion of the screen. This
method should take into account that a module might be displayed
already and should be removed first.
"""
# Remove any module that was already being displayed.
if len(self.rightBox.get_children()) == 2:
oldModule = self.rightBox.get_children()[0]
self.rightBox.remove(oldModule)
# Initialize the module's UI (sync up with the state of some file on
# disk, or whatever) and then pack it into the right side of the
# screen for display.
currentModule = self.moduleList[self._control.currentPage]
currentModule.initializeUI()
if currentModule.vbox is None:
err = _("Module %s did not setup its UI properly") % currentModule.title
logging.error(err)
raise RuntimeError, err
self.rightBox.pack_start(currentModule.vbox)
currentModule.focus()
self.win.show_all()
def moveToPage(self, moduleTitle=None, pageNum=None):
"""Move to and display the page given either by title or page number.
This method raises SystemError if neither is provided, or if no
page is found. It is safe to call this method from within the apply
method of modules, unlike advance().
"""
if moduleTitle is None and pageNum is None:
logging.error(_("moveToPage must be given a module title or page number."))
raise SystemError, _("moveToPage must be given a module title or page number.")
# If we were given a moduleTitle, look up the corresponding pageNum.
# Everything else in firstboot is indexed by number.
if moduleTitle is not None:
pageNum = self.titleToPageNum(moduleTitle, self.moduleList)
# If we're at the end of a ModuleSet's module list, pop off the control
# structure and set up to move to the next page after the set. If
# we're already at the top level, it's easy.
if pageNum == len(self.moduleList):
if len(self._controlStack) > 1:
oldFrame = self._controlStack.pop()
self._control = self._controlStack[-1]
# Put the ModuleSet's history into the object so we can keep
# the history should we go back into the set later on. This
# is kind of a hack.
oldPage = self.moduleList[self._control.currentPage]
if isinstance(oldPage, ModuleSet):
# Add the last page in the ModuleSet, since it will
# otherwise be forgotten. We don't append to the history
# until later in this method.
oldPage._history = oldFrame.history + [oldFrame.currentPage]
self.moveToPage(pageNum=self._control.currentPage+1)
return
else:
self._control.currentPage += 1
return
# Only add the current page to the history if we are moving forward.
# Adding it when we're going backwards traps us at the first page of
# a ModuleSet.
if pageNum > self._control.currentPage and not self._control.currentPage in self._control.history:
self._control.history.append(self._control.currentPage)
# Set this regardless so we know where we are on the way back out of
# a ModuleSet.
self._control.currentPage = pageNum
if isinstance(self.moduleList[pageNum], ModuleSet):
newControl = Control()
newControl.currentPage = 0
newControl.moduleList = self.moduleList[pageNum].moduleList
# If we are stepping back into a ModuleSet from somewhere after it, try
# to load any saved history. This preserves the UI flow that you'd
# expect.
if hasattr(self.moduleList[pageNum], "_history"):
newControl.history = self.moduleList[pageNum]._history
else:
newControl.history = []
self._controlStack.append(newControl)
self._control = newControl
if len(newControl.history) > 0:
self.moveToPage(pageNum=self._control.history.pop())
return
else:
self._setBackSensitivity()
self._setPointer(self._controlStack[0].currentPage)
self.displayModule()
def run(self):
"""Given an interface that has had all its UI components loaded and
initialized, run the interface. Because this method must call into
the UI toolkit to handle events and such, it is assumed that this
method does not exit until the UI is unloaded. From this point on,
all interaction must take place in callbacks.
"""
self.displayModule()
self.win.present()
self.nextButton.grab_focus()
gtk.main()
def takeScreenshot(self):
"""Take a screenshot."""
if not os.access(self._screenshotDir, os.R_OK):
try:
os.mkdir(self._screenshotDir)
except:
logging.error(_("Unable to create the screenshot dir; skipping."))
return
screenshot = gtk.gdk.Pixbuf(gtk.gdk.COLORSPACE_RGB, False, 8,
self._x_size, self._y_size)
screenshot.get_from_drawable(gtk.gdk.get_default_root_window(),
gtk.gdk.colormap_get_system(),
0, 0, 0, 0, self._x_size, self._y_size)
if not screenshot:
return
while True:
sname = "screenshot-%04d.png" % self._screenshotIndex
if not os.access("%s/%s" % (self._screenshotDir, sname), os.R_OK):
break
self._screenshotIndex += 1
screenshot.save("%s/%s" % (self._screenshotDir, sname), "png")
self._screenshotIndex += 1
def titleToPageNum(self, moduleTitle, moduleList):
"""Lookup the given title in the given module list. Returns the page
number on success. This only works on the given moduleList, so
for a ModuleSet it would only find the page if it exists in the
set.
"""
if moduleTitle is None:
return None
pageNum = 0
while True:
try:
if moduleList[pageNum].title == moduleTitle:
break
pageNum += 1
except IndexError:
logging.error(_("No module exists with the title %s.") % moduleTitle)
raise SystemError, _("No module exists with the title %s.") % moduleTitle
return pageNum