mirror of
https://github.com/virt-manager/virt-manager.git
synced 2025-01-25 06:03:55 +03:00
95ca39aae8
Signed-off-by: Cole Robinson <crobinso@redhat.com>
246 lines
7.4 KiB
Python
246 lines
7.4 KiB
Python
# This work is licensed under the GNU GPLv2 or later.
|
|
# See the COPYING file in the top-level directory.
|
|
|
|
# pylint: disable=wrong-import-order,ungrouped-imports
|
|
import os
|
|
|
|
import gi
|
|
|
|
from virtinst import log
|
|
|
|
# We can use either gtksourceview3 or gtksourceview4
|
|
have_gtksourceview = True
|
|
try:
|
|
gi.require_version("GtkSource", "4")
|
|
log.debug("Using GtkSource 4")
|
|
except ValueError: # pragma: no cover
|
|
try:
|
|
gi.require_version("GtkSource", "3.0")
|
|
log.debug("Using GtkSource 3.0")
|
|
except ValueError:
|
|
log.debug("Not using GtkSource")
|
|
have_gtksourceview = False
|
|
|
|
if "VIRTINST_TEST_SUITE_FAKE_NO_SOURCEVIEW" in os.environ:
|
|
log.debug("Faking missing GtkSource for test suite")
|
|
have_gtksourceview = False
|
|
|
|
if have_gtksourceview:
|
|
from gi.repository import GtkSource
|
|
else:
|
|
# if GtkSourceView is not available, just use a plain TextView. This will
|
|
# only disable auto-indent and syntax highlighting.
|
|
from gi.repository import Gtk
|
|
|
|
from .lib import uiutil
|
|
from .baseclass import vmmGObjectUI
|
|
|
|
_PAGE_DETAILS = 0
|
|
_PAGE_XML = 1
|
|
|
|
|
|
class vmmXMLEditor(vmmGObjectUI):
|
|
__gsignals__ = {
|
|
"changed": (vmmGObjectUI.RUN_FIRST, None, []),
|
|
"xml-requested": (vmmGObjectUI.RUN_FIRST, None, []),
|
|
"xml-reset": (vmmGObjectUI.RUN_FIRST, None, []),
|
|
}
|
|
|
|
def __init__(self, builder, topwin, parent_container, details_widget):
|
|
super().__init__("xmleditor.ui", None,
|
|
builder=builder, topwin=topwin)
|
|
|
|
parent_container.remove(details_widget)
|
|
parent_container.add(self.widget("xml-notebook"))
|
|
self.widget("xml-details-box").add(details_widget)
|
|
|
|
self._curpage = _PAGE_DETAILS
|
|
self._srcxml = ""
|
|
self._srcview = None
|
|
self._srcbuff = None
|
|
self._init_ui()
|
|
|
|
self.details_changed = False
|
|
|
|
self.add_gsettings_handle(
|
|
self.config.on_xmleditor_enabled_changed(
|
|
self._xmleditor_enabled_changed_cb))
|
|
|
|
|
|
def _cleanup(self):
|
|
self._srcview.destroy()
|
|
self._srcbuff = None
|
|
|
|
|
|
###########
|
|
# UI init #
|
|
###########
|
|
|
|
def _set_xmleditor_enabled_from_config(self):
|
|
enabled = self.config.get_xmleditor_enabled()
|
|
self._srcview.set_editable(enabled)
|
|
uiutil.set_grid_row_visible(self.widget("xml-warning-box"),
|
|
not enabled)
|
|
|
|
def _init_ui(self):
|
|
if have_gtksourceview:
|
|
self._srcview = GtkSource.View()
|
|
self._srcbuff = self._srcview.get_buffer()
|
|
self._srcview.set_auto_indent(True)
|
|
lang = GtkSource.LanguageManager.get_default().get_language("xml")
|
|
self._srcbuff.set_language(lang)
|
|
self._srcbuff.set_highlight_syntax(True)
|
|
else:
|
|
self._srcview = Gtk.TextView()
|
|
self._srcbuff = self._srcview.get_buffer()
|
|
|
|
self._srcview.set_monospace(True)
|
|
self._srcview.get_accessible().set_name("XML editor")
|
|
|
|
self._srcbuff.connect("changed", self._buffer_changed_cb)
|
|
|
|
self.widget("xml-notebook").connect("switch-page",
|
|
self._before_page_changed_cb)
|
|
self.widget("xml-notebook").connect("notify::page",
|
|
self._after_page_changed_cb)
|
|
|
|
self._srcview.show_all()
|
|
self.widget("xml-scroll").add(self._srcview)
|
|
self._set_xmleditor_enabled_from_config()
|
|
|
|
|
|
####################
|
|
# Internal helpers #
|
|
####################
|
|
|
|
def _reselect_page(self, pagenum):
|
|
# Setting _curpage first will shortcircuit our page changed callback
|
|
self._curpage = pagenum
|
|
self.widget("xml-notebook").set_current_page(pagenum)
|
|
|
|
def _reset_xml(self):
|
|
self.set_xml("")
|
|
self.emit("xml-reset")
|
|
|
|
def _reset_cursor(self):
|
|
# Put cursor at the start of the second line. Starting on the
|
|
# first means XML open/close tags are highlighted which is weird
|
|
# starting visual
|
|
startiter = self._srcbuff.get_start_iter()
|
|
startiter.forward_line()
|
|
self._srcbuff.place_cursor(startiter)
|
|
|
|
def _detials_unapplied_changes(self):
|
|
if not self.details_changed:
|
|
return False
|
|
|
|
ret = self.err.yes_no(
|
|
_("There are unapplied changes."),
|
|
_("Your changes will be lost if you leave this tab. "
|
|
"Really leave this tab?"))
|
|
if ret:
|
|
self.details_changed = False
|
|
|
|
return not ret
|
|
|
|
def _xml_unapplied_changes(self):
|
|
if self._srcxml == self.get_xml():
|
|
return False
|
|
|
|
ret = self.err.yes_no(
|
|
_("There are unapplied changes."),
|
|
_("Your XML changes will be lost if you leave this tab. "
|
|
"Really leave this tab?"))
|
|
|
|
return not ret
|
|
|
|
|
|
|
|
|
|
##############
|
|
# Public API #
|
|
##############
|
|
|
|
def reset_state(self):
|
|
"""
|
|
Clear XML and select the details page. Used when callers do
|
|
their own reset_state
|
|
"""
|
|
self._reset_xml()
|
|
return self.widget("xml-notebook").set_current_page(_PAGE_DETAILS)
|
|
|
|
def get_xml(self):
|
|
"""
|
|
Return the XML from the editor UI
|
|
"""
|
|
return self._srcbuff.get_property("text")
|
|
|
|
def set_xml(self, xml):
|
|
"""
|
|
Set the editor UI XML to the passed string
|
|
"""
|
|
try:
|
|
self._srcbuff.disconnect_by_func(self._buffer_changed_cb)
|
|
self._srcxml = xml or ""
|
|
self._srcbuff.set_text(self._srcxml)
|
|
self._reset_cursor()
|
|
finally:
|
|
self._srcbuff.connect("changed", self._buffer_changed_cb)
|
|
|
|
def set_xml_from_libvirtobject(self, libvirtobject):
|
|
"""
|
|
Set the editor UI XML to the inactive XML from the passed
|
|
vmmLibvirtObject. If the XML UI isn't visible, we don't set
|
|
anything, which lets callers use this on every page refresh
|
|
"""
|
|
if not self.is_xml_selected():
|
|
return
|
|
xml = ""
|
|
if libvirtobject:
|
|
xml = libvirtobject.get_xml_to_define()
|
|
self.set_xml(xml)
|
|
|
|
def is_xml_selected(self):
|
|
"""
|
|
Return True if the XML page is selected
|
|
"""
|
|
return self._curpage == _PAGE_XML
|
|
|
|
|
|
#############
|
|
# Listeners #
|
|
#############
|
|
|
|
def _buffer_changed_cb(self, buf):
|
|
self.emit("changed")
|
|
|
|
def _before_page_changed_cb(self, notebook, widget, pagenum):
|
|
if self._curpage == pagenum:
|
|
return
|
|
prevpage = self._curpage
|
|
self._curpage = pagenum
|
|
|
|
if pagenum == _PAGE_XML:
|
|
if not self._detials_unapplied_changes():
|
|
# If the XML page is clicked, emit xml-requested signal which
|
|
# expects the user to call set_xml/set_libvirtobject. This saves
|
|
# having to fetch inactive XML up front, and gives users like
|
|
# a hook to actually serialize the final XML to return
|
|
self.emit("xml-requested")
|
|
return
|
|
else:
|
|
if not self._xml_unapplied_changes():
|
|
self._reset_xml()
|
|
return
|
|
|
|
# I can't find anyway to make the notebook stay on the current page
|
|
# So set an idle callback to switch back to the XML page. It causes
|
|
# a visual UI blip unfortunately
|
|
self.idle_add(self._reselect_page, prevpage)
|
|
|
|
def _after_page_changed_cb(self, notebook, gparam):
|
|
self._curpage = notebook.get_current_page()
|
|
|
|
def _xmleditor_enabled_changed_cb(self):
|
|
self._set_xmleditor_enabled_from_config()
|