ipythonqt.py
402 lines
| 15.1 KiB
| text/x-python
|
PythonLexer
epatters
|
r2801 | """ A minimal application using the Qt console-style IPython frontend. | ||
epatters
|
r2758 | """ | ||
epatters
|
r2961 | #----------------------------------------------------------------------------- | ||
# Imports | ||||
#----------------------------------------------------------------------------- | ||||
MinRK
|
r3971 | # stdlib imports | ||
import os | ||||
import signal | ||||
import sys | ||||
# System library imports | ||||
Evan Patterson
|
r3304 | from IPython.external.qt import QtGui | ||
MinRK
|
r3170 | from pygments.styles import get_all_styles | ||
Evan Patterson
|
r3304 | |||
epatters
|
r2758 | # Local imports | ||
MinRK
|
r3983 | from IPython.config.application import boolean_flag | ||
MinRK
|
r3971 | from IPython.core.newapplication import ProfileDir, BaseIPythonApplication | ||
epatters
|
r2801 | from IPython.frontend.qt.console.frontend_widget import FrontendWidget | ||
from IPython.frontend.qt.console.ipython_widget import IPythonWidget | ||||
from IPython.frontend.qt.console.rich_ipython_widget import RichIPythonWidget | ||||
MinRK
|
r3170 | from IPython.frontend.qt.console import styles | ||
epatters
|
r2758 | from IPython.frontend.qt.kernelmanager import QtKernelManager | ||
MinRK
|
r3971 | from IPython.utils.traitlets import ( | ||
MinRK
|
r3983 | Dict, List, Unicode, Int, CaselessStrEnum, CBool, Any | ||
MinRK
|
r3971 | ) | ||
from IPython.zmq.ipkernel import ( | ||||
flags as ipkernel_flags, | ||||
aliases as ipkernel_aliases, | ||||
IPKernelApp | ||||
) | ||||
from IPython.zmq.zmqshell import ZMQInteractiveShell | ||||
epatters
|
r2758 | |||
epatters
|
r2961 | #----------------------------------------------------------------------------- | ||
MinRK
|
r3144 | # Network Constants | ||
epatters
|
r2961 | #----------------------------------------------------------------------------- | ||
MinRK
|
r3144 | from IPython.utils.localinterfaces import LOCALHOST, LOCAL_IPS | ||
epatters
|
r2823 | |||
epatters
|
r2961 | #----------------------------------------------------------------------------- | ||
# Classes | ||||
#----------------------------------------------------------------------------- | ||||
class MainWindow(QtGui.QMainWindow): | ||||
#--------------------------------------------------------------------------- | ||||
# 'object' interface | ||||
#--------------------------------------------------------------------------- | ||||
MinRK
|
r3983 | def __init__(self, app, frontend, existing=False, may_close=True, | ||
confirm_exit=True): | ||||
epatters
|
r2961 | """ Create a MainWindow for the specified FrontendWidget. | ||
MinRK
|
r3100 | |||
MinRK
|
r3105 | The app is passed as an argument to allow for different | ||
closing behavior depending on whether we are the Kernel's parent. | ||||
MinRK
|
r3142 | If existing is True, then this Console does not own the Kernel. | ||
If may_close is True, then this Console is permitted to close the kernel | ||||
epatters
|
r2961 | """ | ||
super(MainWindow, self).__init__() | ||||
MinRK
|
r3104 | self._app = app | ||
epatters
|
r2961 | self._frontend = frontend | ||
MinRK
|
r3090 | self._existing = existing | ||
MinRK
|
r3144 | if existing: | ||
MinRK
|
r3142 | self._may_close = may_close | ||
else: | ||||
self._may_close = True | ||||
epatters
|
r2961 | self._frontend.exit_requested.connect(self.close) | ||
MinRK
|
r3983 | self._confirm_exit = confirm_exit | ||
epatters
|
r2961 | self.setCentralWidget(frontend) | ||
#--------------------------------------------------------------------------- | ||||
# QWidget interface | ||||
#--------------------------------------------------------------------------- | ||||
def closeEvent(self, event): | ||||
Erik Tollerud
|
r3190 | """ Close the window and the kernel (if necessary). | ||
This will prompt the user if they are finished with the kernel, and if | ||||
so, closes the kernel cleanly. Alternatively, if the exit magic is used, | ||||
it closes without prompt. | ||||
epatters
|
r2961 | """ | ||
Erik Tollerud
|
r3185 | keepkernel = None #Use the prompt by default | ||
if hasattr(self._frontend,'_keep_kernel_on_exit'): #set by exit magic | ||||
keepkernel = self._frontend._keep_kernel_on_exit | ||||
epatters
|
r2982 | kernel_manager = self._frontend.kernel_manager | ||
Erik Tollerud
|
r3183 | |||
MinRK
|
r3983 | if keepkernel is None and not self._confirm_exit: | ||
# don't prompt, just terminate the kernel if we own it | ||||
# or leave it alone if we don't | ||||
keepkernel = not self._existing | ||||
Erik Tollerud
|
r3185 | if keepkernel is None: #show prompt | ||
Erik Tollerud
|
r3183 | if kernel_manager and kernel_manager.channels_running: | ||
title = self.window().windowTitle() | ||||
cancel = QtGui.QMessageBox.Cancel | ||||
okay = QtGui.QMessageBox.Ok | ||||
if self._may_close: | ||||
msg = "You are closing this Console window." | ||||
info = "Would you like to quit the Kernel and all attached Consoles as well?" | ||||
justthis = QtGui.QPushButton("&No, just this Console", self) | ||||
justthis.setShortcut('N') | ||||
closeall = QtGui.QPushButton("&Yes, quit everything", self) | ||||
closeall.setShortcut('Y') | ||||
Evan Patterson
|
r3304 | box = QtGui.QMessageBox(QtGui.QMessageBox.Question, | ||
title, msg) | ||||
Erik Tollerud
|
r3183 | box.setInformativeText(info) | ||
box.addButton(cancel) | ||||
box.addButton(justthis, QtGui.QMessageBox.NoRole) | ||||
box.addButton(closeall, QtGui.QMessageBox.YesRole) | ||||
box.setDefaultButton(closeall) | ||||
box.setEscapeButton(cancel) | ||||
reply = box.exec_() | ||||
if reply == 1: # close All | ||||
kernel_manager.shutdown_kernel() | ||||
#kernel_manager.stop_channels() | ||||
event.accept() | ||||
elif reply == 0: # close Console | ||||
if not self._existing: | ||||
# Have kernel: don't quit, just close the window | ||||
self._app.setQuitOnLastWindowClosed(False) | ||||
self.deleteLater() | ||||
event.accept() | ||||
else: | ||||
event.ignore() | ||||
MinRK
|
r3142 | else: | ||
Erik Tollerud
|
r3183 | reply = QtGui.QMessageBox.question(self, title, | ||
"Are you sure you want to close this Console?"+ | ||||
"\nThe Kernel and other Consoles will remain active.", | ||||
okay|cancel, | ||||
defaultButton=okay | ||||
) | ||||
if reply == okay: | ||||
event.accept() | ||||
else: | ||||
event.ignore() | ||||
Erik Tollerud
|
r3185 | elif keepkernel: #close console but leave kernel running (no prompt) | ||
Erik Tollerud
|
r3183 | if kernel_manager and kernel_manager.channels_running: | ||
if not self._existing: | ||||
# I have the kernel: don't quit, just close the window | ||||
self._app.setQuitOnLastWindowClosed(False) | ||||
event.accept() | ||||
Erik Tollerud
|
r3185 | else: #close console and kernel (no prompt) | ||
Erik Tollerud
|
r3183 | if kernel_manager and kernel_manager.channels_running: | ||
kernel_manager.shutdown_kernel() | ||||
event.accept() | ||||
epatters
|
r2961 | |||
#----------------------------------------------------------------------------- | ||||
MinRK
|
r3971 | # Aliases and Flags | ||
epatters
|
r2961 | #----------------------------------------------------------------------------- | ||
epatters
|
r2758 | |||
MinRK
|
r3971 | flags = dict(ipkernel_flags) | ||
flags.update({ | ||||
'existing' : ({'IPythonQtConsoleApp' : {'existing' : True}}, | ||||
"Connect to an existing kernel."), | ||||
'pure' : ({'IPythonQtConsoleApp' : {'pure' : True}}, | ||||
"Use a pure Python kernel instead of an IPython kernel."), | ||||
'plain' : ({'ConsoleWidget' : {'kind' : 'plain'}}, | ||||
"Disable rich text support."), | ||||
}) | ||||
MinRK
|
r3983 | flags.update(boolean_flag( | ||
'gui-completion', 'ConsoleWidget.gui_completion', | ||||
"use a GUI widget for tab completion", | ||||
"use plaintext output for completion" | ||||
)) | ||||
flags.update(boolean_flag( | ||||
'confirm-exit', 'IPythonQtConsoleApp.confirm_exit', | ||||
"""Set to display confirmation dialog on exit. You can always use 'exit' or 'quit', | ||||
to force a direct exit without any confirmation. | ||||
""", | ||||
"""Don't prompt the user when exiting. This will terminate the kernel | ||||
if it is owned by the frontend, and leave it alive if it is external. | ||||
""" | ||||
)) | ||||
# the flags that are specific to the frontend | ||||
# these must be scrubbed before being passed to the kernel, | ||||
# or it will raise an error on unrecognized flags | ||||
qt_flags = ['existing', 'pure', 'plain', 'gui-completion', 'no-gui-completion', | ||||
'confirm-exit', 'no-confirm-exit'] | ||||
MinRK
|
r3971 | |||
aliases = dict(ipkernel_aliases) | ||||
aliases.update(dict( | ||||
hb = 'IPythonQtConsoleApp.hb_port', | ||||
shell = 'IPythonQtConsoleApp.shell_port', | ||||
iopub = 'IPythonQtConsoleApp.iopub_port', | ||||
stdin = 'IPythonQtConsoleApp.stdin_port', | ||||
ip = 'IPythonQtConsoleApp.ip', | ||||
plain = 'IPythonQtConsoleApp.plain', | ||||
pure = 'IPythonQtConsoleApp.pure', | ||||
MinRK
|
r3983 | gui_completion = 'ConsoleWidget.gui_completion', | ||
MinRK
|
r3971 | style = 'IPythonWidget.syntax_style', | ||
stylesheet = 'IPythonQtConsoleApp.stylesheet', | ||||
colors = 'ZMQInteractiveShell.colors', | ||||
editor = 'IPythonWidget.editor', | ||||
pi = 'IPythonWidget.in_prompt', | ||||
po = 'IPythonWidget.out_prompt', | ||||
si = 'IPythonWidget.input_sep', | ||||
so = 'IPythonWidget.output_sep', | ||||
so2 = 'IPythonWidget.output_sep2', | ||||
)) | ||||
#----------------------------------------------------------------------------- | ||||
# IPythonQtConsole | ||||
#----------------------------------------------------------------------------- | ||||
class IPythonQtConsoleApp(BaseIPythonApplication): | ||||
name = 'ipython-qtconsole' | ||||
default_config_file_name='ipython_config.py' | ||||
classes = [IPKernelApp, IPythonWidget, ZMQInteractiveShell, ProfileDir] | ||||
flags = Dict(flags) | ||||
aliases = Dict(aliases) | ||||
kernel_argv = List(Unicode) | ||||
# connection info: | ||||
ip = Unicode(LOCALHOST, config=True, | ||||
help="""Set the kernel\'s IP address [default localhost]. | ||||
If the IP address is something other than localhost, then | ||||
Consoles on other machines will be able to connect | ||||
to the Kernel, so be careful!""" | ||||
) | ||||
hb_port = Int(0, config=True, | ||||
help="set the heartbeat port [default: random]") | ||||
shell_port = Int(0, config=True, | ||||
help="set the shell (XREP) port [default: random]") | ||||
iopub_port = Int(0, config=True, | ||||
help="set the iopub (PUB) port [default: random]") | ||||
stdin_port = Int(0, config=True, | ||||
help="set the stdin (XREQ) port [default: random]") | ||||
MinRK
|
r3983 | existing = CBool(False, config=True, | ||
MinRK
|
r3971 | help="Whether to connect to an already running Kernel.") | ||
stylesheet = Unicode('', config=True, | ||||
help="path to a custom CSS stylesheet") | ||||
MinRK
|
r3983 | pure = CBool(False, config=True, | ||
MinRK
|
r3971 | help="Use a pure Python kernel instead of an IPython kernel.") | ||
MinRK
|
r3983 | plain = CBool(False, config=True, | ||
MinRK
|
r3976 | help="Use a plaintext widget instead of rich text (plain can't print/save).") | ||
MinRK
|
r3971 | |||
def _pure_changed(self, name, old, new): | ||||
kind = 'plain' if self.plain else 'rich' | ||||
self.config.ConsoleWidget.kind = kind | ||||
if self.pure: | ||||
self.widget_factory = FrontendWidget | ||||
elif self.plain: | ||||
self.widget_factory = IPythonWidget | ||||
MinRK
|
r3173 | else: | ||
MinRK
|
r3971 | self.widget_factory = RichIPythonWidget | ||
_plain_changed = _pure_changed | ||||
MinRK
|
r3983 | confirm_exit = CBool(True, config=True, | ||
help=""" | ||||
Set to display confirmation dialog on exit. You can always use 'exit' or 'quit', | ||||
to force a direct exit without any confirmation.""", | ||||
) | ||||
MinRK
|
r3971 | # the factory for creating a widget | ||
widget_factory = Any(RichIPythonWidget) | ||||
def parse_command_line(self, argv=None): | ||||
super(IPythonQtConsoleApp, self).parse_command_line(argv) | ||||
if argv is None: | ||||
argv = sys.argv[1:] | ||||
self.kernel_argv = list(argv) # copy | ||||
# scrub frontend-specific flags | ||||
for a in argv: | ||||
if a.startswith('--') and a[2:] in qt_flags: | ||||
self.kernel_argv.remove(a) | ||||
def init_kernel_manager(self): | ||||
# Don't let Qt or ZMQ swallow KeyboardInterupts. | ||||
signal.signal(signal.SIGINT, signal.SIG_DFL) | ||||
# Create a KernelManager and start a kernel. | ||||
self.kernel_manager = QtKernelManager( | ||||
MinRK
|
r3974 | shell_address=(self.ip, self.shell_port), | ||
MinRK
|
r3971 | sub_address=(self.ip, self.iopub_port), | ||
MinRK
|
r3974 | stdin_address=(self.ip, self.stdin_port), | ||
MinRK
|
r3971 | hb_address=(self.ip, self.hb_port) | ||
) | ||||
# start the kernel | ||||
if not self.existing: | ||||
kwargs = dict(ip=self.ip, ipython=not self.pure) | ||||
kwargs['extra_arguments'] = self.kernel_argv | ||||
self.kernel_manager.start_kernel(**kwargs) | ||||
self.kernel_manager.start_channels() | ||||
def init_qt_elements(self): | ||||
# Create the widget. | ||||
self.app = QtGui.QApplication([]) | ||||
local_kernel = (not self.existing) or self.ip in LOCAL_IPS | ||||
self.widget = self.widget_factory(config=self.config, | ||||
local_kernel=local_kernel) | ||||
self.widget.kernel_manager = self.kernel_manager | ||||
self.window = MainWindow(self.app, self.widget, self.existing, | ||||
MinRK
|
r3983 | may_close=local_kernel, | ||
confirm_exit=self.confirm_exit) | ||||
MinRK
|
r3971 | self.window.setWindowTitle('Python' if self.pure else 'IPython') | ||
def init_colors(self): | ||||
"""Configure the coloring of the widget""" | ||||
# Note: This will be dramatically simplified when colors | ||||
# are removed from the backend. | ||||
if self.pure: | ||||
# only IPythonWidget supports styling | ||||
return | ||||
# parse the colors arg down to current known labels | ||||
try: | ||||
colors = self.config.ZMQInteractiveShell.colors | ||||
except AttributeError: | ||||
colors = None | ||||
try: | ||||
style = self.config.IPythonWidget.colors | ||||
except AttributeError: | ||||
style = None | ||||
# find the value for colors: | ||||
if colors: | ||||
colors=colors.lower() | ||||
if colors in ('lightbg', 'light'): | ||||
colors='lightbg' | ||||
elif colors in ('dark', 'linux'): | ||||
colors='linux' | ||||
else: | ||||
colors='nocolor' | ||||
elif style: | ||||
if style=='bw': | ||||
colors='nocolor' | ||||
elif styles.dark_style(style): | ||||
colors='linux' | ||||
else: | ||||
colors='lightbg' | ||||
MinRK
|
r3173 | else: | ||
MinRK
|
r3971 | colors=None | ||
# Configure the style. | ||||
widget = self.widget | ||||
if style: | ||||
widget.style_sheet = styles.sheet_from_template(style, colors) | ||||
widget.syntax_style = style | ||||
MinRK
|
r3170 | widget._syntax_style_changed() | ||
widget._style_sheet_changed() | ||||
MinRK
|
r3171 | elif colors: | ||
# use a default style | ||||
widget.set_default_style(colors=colors) | ||||
MinRK
|
r3170 | else: | ||
# this is redundant for now, but allows the widget's | ||||
# defaults to change | ||||
MinRK
|
r3171 | widget.set_default_style() | ||
MinRK
|
r3170 | |||
MinRK
|
r3971 | if self.stylesheet: | ||
MinRK
|
r3170 | # we got an expicit stylesheet | ||
MinRK
|
r3971 | if os.path.isfile(self.stylesheet): | ||
with open(self.stylesheet) as f: | ||||
MinRK
|
r3170 | sheet = f.read() | ||
widget.style_sheet = sheet | ||||
widget._style_sheet_changed() | ||||
else: | ||||
MinRK
|
r3971 | raise IOError("Stylesheet %r not found."%self.stylesheet) | ||
def initialize(self, argv=None): | ||||
super(IPythonQtConsoleApp, self).initialize(argv) | ||||
self.init_kernel_manager() | ||||
self.init_qt_elements() | ||||
self.init_colors() | ||||
def start(self): | ||||
# draw the window | ||||
self.window.show() | ||||
MinRK
|
r3170 | |||
MinRK
|
r3971 | # Start the application main loop. | ||
self.app.exec_() | ||||
epatters
|
r2841 | |||
MinRK
|
r3971 | #----------------------------------------------------------------------------- | ||
# Main entry point | ||||
#----------------------------------------------------------------------------- | ||||
def main(): | ||||
app = IPythonQtConsoleApp() | ||||
app.initialize() | ||||
app.start() | ||||
epatters
|
r2758 | |||
if __name__ == '__main__': | ||||
main() | ||||