qtconsoleapp.py
386 lines
| 13.6 KiB
| text/x-python
|
PythonLexer
epatters
|
r2801 | """ A minimal application using the Qt console-style IPython frontend. | ||
MinRK
|
r4021 | |||
This is not a complete console app, as subprocess will not be able to receive | ||||
input, there is no real readline support, among other limitations. | ||||
epatters
|
r2758 | """ | ||
Dimitry Kloper
|
r16520 | # Copyright (c) IPython Development Team. | ||
# Distributed under the terms of the Modified BSD License. | ||||
epatters
|
r2961 | #----------------------------------------------------------------------------- | ||
# Imports | ||||
#----------------------------------------------------------------------------- | ||||
MinRK
|
r3971 | # stdlib imports | ||
import os | ||||
import signal | ||||
import sys | ||||
Mark Wiebe
|
r6168 | # If run on Windows, install an exception hook which pops up a | ||
# message box. Pythonw.exe hides the console, so without this | ||||
# the application silently fails to load. | ||||
# | ||||
# We always install this handler, because the expectation is for | ||||
# qtconsole to bring up a GUI even if called from the console. | ||||
# The old handler is called, so the exception is printed as well. | ||||
# If desired, check for pythonw with an additional condition | ||||
# (sys.executable.lower().find('pythonw.exe') >= 0). | ||||
if os.name == 'nt': | ||||
old_excepthook = sys.excepthook | ||||
Thomas Kluyver
|
r13587 | # Exclude this from our autogenerated API docs. | ||
undoc = lambda func: func | ||||
@undoc | ||||
Mark Wiebe
|
r6168 | def gui_excepthook(exctype, value, tb): | ||
try: | ||||
import ctypes, traceback | ||||
Thomas Kluyver
|
r13352 | MB_ICONERROR = 0x00000010 | ||
Mark Wiebe
|
r6168 | title = u'Error starting IPython QtConsole' | ||
msg = u''.join(traceback.format_exception(exctype, value, tb)) | ||||
ctypes.windll.user32.MessageBoxW(0, msg, title, MB_ICONERROR) | ||||
finally: | ||||
# Also call the old exception hook to let it do | ||||
# its thing too. | ||||
old_excepthook(exctype, value, tb) | ||||
sys.excepthook = gui_excepthook | ||||
MinRK
|
r3971 | # System library imports | ||
MinRK
|
r5609 | from IPython.external.qt import QtCore, QtGui | ||
Evan Patterson
|
r3304 | |||
epatters
|
r2758 | # Local imports | ||
Helen ST
|
r19171 | from IPython.config.application import boolean_flag | ||
Thomas Kluyver
|
r11131 | from IPython.config.application import catch_config_error | ||
MinRK
|
r4024 | from IPython.core.application import BaseIPythonApplication | ||
Fernando Perez
|
r11022 | from IPython.qt.console.ipython_widget import IPythonWidget | ||
from IPython.qt.console.rich_ipython_widget import RichIPythonWidget | ||||
from IPython.qt.console import styles | ||||
from IPython.qt.console.mainwindow import MainWindow | ||||
from IPython.qt.client import QtKernelClient | ||||
from IPython.qt.manager import QtKernelManager | ||||
MinRK
|
r3971 | from IPython.utils.traitlets import ( | ||
Thomas Kluyver
|
r11131 | Dict, Unicode, CBool, Any | ||
MinRK
|
r3971 | ) | ||
Paul Ivanov
|
r5604 | |||
Fernando Perez
|
r11022 | from IPython.consoleapp import ( | ||
MinRK
|
r5618 | IPythonConsoleApp, app_aliases, app_flags, flags, aliases | ||
Paul Ivanov
|
r5603 | ) | ||
MinRK
|
r3971 | |||
epatters
|
r2961 | #----------------------------------------------------------------------------- | ||
MinRK
|
r3144 | # Network Constants | ||
epatters
|
r2961 | #----------------------------------------------------------------------------- | ||
MinRK
|
r12591 | from IPython.utils.localinterfaces import is_local_ip | ||
epatters
|
r2823 | |||
epatters
|
r2961 | #----------------------------------------------------------------------------- | ||
Brian Granger
|
r4216 | # Globals | ||
#----------------------------------------------------------------------------- | ||||
_examples = """ | ||||
Matthias BUSSONNIER
|
r11784 | ipython qtconsole # start the qtconsole | ||
ipython qtconsole --matplotlib=inline # start with matplotlib inline plotting mode | ||||
Brian Granger
|
r4216 | """ | ||
#----------------------------------------------------------------------------- | ||||
MinRK
|
r3971 | # Aliases and Flags | ||
epatters
|
r2961 | #----------------------------------------------------------------------------- | ||
epatters
|
r2758 | |||
MinRK
|
r5610 | # start with copy of flags | ||
flags = dict(flags) | ||||
MinRK
|
r4247 | qt_flags = { | ||
MinRK
|
r6806 | 'plain' : ({'IPythonQtConsoleApp' : {'plain' : True}}, | ||
MinRK
|
r3971 | "Disable rich text support."), | ||
MinRK
|
r4247 | } | ||
Helen ST
|
r19171 | qt_flags.update(boolean_flag( | ||
'banner', 'IPythonQtConsoleApp.display_banner', | ||||
Helen ST
|
r19190 | "Display a banner upon starting the QtConsole.", | ||
"Don't display a banner upon starting the QtConsole." | ||||
Helen ST
|
r19171 | )) | ||
Matthias BUSSONNIER
|
r7389 | |||
MinRK
|
r5610 | # and app_flags from the Console Mixin | ||
qt_flags.update(app_flags) | ||||
# add frontend flags to the full set | ||||
MinRK
|
r4247 | flags.update(qt_flags) | ||
MinRK
|
r3971 | |||
MinRK
|
r5610 | # start with copy of front&backend aliases list | ||
aliases = dict(aliases) | ||||
MinRK
|
r4247 | qt_aliases = dict( | ||
MinRK
|
r3971 | style = 'IPythonWidget.syntax_style', | ||
stylesheet = 'IPythonQtConsoleApp.stylesheet', | ||||
colors = 'ZMQInteractiveShell.colors', | ||||
editor = 'IPythonWidget.editor', | ||||
MinRK
|
r4222 | paging = 'ConsoleWidget.paging', | ||
MinRK
|
r4247 | ) | ||
MinRK
|
r5610 | # and app_aliases from the Console Mixin | ||
qt_aliases.update(app_aliases) | ||||
Matthias BUSSONNIER
|
r7389 | qt_aliases.update({'gui-completion':'ConsoleWidget.gui_completion'}) | ||
MinRK
|
r5610 | # add frontend aliases to the full set | ||
MinRK
|
r4247 | aliases.update(qt_aliases) | ||
MinRK
|
r4222 | |||
MinRK
|
r5610 | # get flags&aliases into sets, and remove a couple that | ||
# shouldn't be scrubbed from backend flags: | ||||
qt_aliases = set(qt_aliases.keys()) | ||||
qt_aliases.remove('colors') | ||||
qt_flags = set(qt_flags.keys()) | ||||
MinRK
|
r5136 | #----------------------------------------------------------------------------- | ||
# Classes | ||||
#----------------------------------------------------------------------------- | ||||
MinRK
|
r3971 | |||
#----------------------------------------------------------------------------- | ||||
# IPythonQtConsole | ||||
#----------------------------------------------------------------------------- | ||||
Brian Granger
|
r4215 | |||
MinRK
|
r5618 | class IPythonQtConsoleApp(BaseIPythonApplication, IPythonConsoleApp): | ||
MinRK
|
r3971 | name = 'ipython-qtconsole' | ||
Matthias BUSSONNIER
|
r5060 | |||
MinRK
|
r4021 | description = """ | ||
The IPython QtConsole. | ||||
This launches a Console-style application using Qt. It is not a full | ||||
MinRK
|
r4222 | console, in that launched terminal subprocesses will not be able to accept | ||
input. | ||||
The QtConsole supports various extra features beyond the Terminal IPython | ||||
shell, such as inline plotting with matplotlib, via: | ||||
Matthias BUSSONNIER
|
r11784 | ipython qtconsole --matplotlib=inline | ||
MinRK
|
r4021 | |||
MinRK
|
r4222 | as well as saving your session as HTML, and printing the output. | ||
MinRK
|
r4021 | |||
""" | ||||
Brian Granger
|
r4216 | examples = _examples | ||
Brian Granger
|
r4215 | |||
MinRK
|
r7291 | classes = [IPythonWidget] + IPythonConsoleApp.classes | ||
MinRK
|
r3971 | flags = Dict(flags) | ||
aliases = Dict(aliases) | ||||
MinRK
|
r5620 | frontend_flags = Any(qt_flags) | ||
frontend_aliases = Any(qt_aliases) | ||||
MinRK
|
r10288 | kernel_client_class = QtKernelClient | ||
MinRK
|
r10312 | kernel_manager_class = QtKernelManager | ||
MinRK
|
r3971 | |||
stylesheet = Unicode('', config=True, | ||||
help="path to a custom CSS stylesheet") | ||||
Dan Kilman
|
r10960 | hide_menubar = CBool(False, config=True, | ||
help="Start the console window with the menu bar hidden.") | ||||
maximize = CBool(False, config=True, | ||||
help="Start the console window maximized.") | ||||
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 | |||
Helen ST
|
r19171 | display_banner = CBool(True, config=True, | ||
Helen ST
|
r19191 | help="Whether to display a banner upon starting the QtConsole." | ||
Helen ST
|
r19171 | ) | ||
MinRK
|
r6806 | def _plain_changed(self, name, old, new): | ||
kind = 'plain' if new else 'rich' | ||||
MinRK
|
r3971 | self.config.ConsoleWidget.kind = kind | ||
MinRK
|
r6806 | if new: | ||
MinRK
|
r3971 | self.widget_factory = IPythonWidget | ||
MinRK
|
r3173 | else: | ||
MinRK
|
r3971 | self.widget_factory = RichIPythonWidget | ||
# the factory for creating a widget | ||||
widget_factory = Any(RichIPythonWidget) | ||||
def parse_command_line(self, argv=None): | ||||
super(IPythonQtConsoleApp, self).parse_command_line(argv) | ||||
MinRK
|
r5620 | self.build_kernel_argv(argv) | ||
Paul Ivanov
|
r5604 | |||
MinRK
|
r3971 | |||
MinRK
|
r5135 | def new_frontend_master(self): | ||
""" Create and return new frontend attached to new kernel, launched on localhost. | ||||
Matthias BUSSONNIER
|
r5039 | """ | ||
MinRK
|
r5849 | kernel_manager = self.kernel_manager_class( | ||
MinRK
|
r5073 | connection_file=self._new_connection_file(), | ||
MinRK
|
r11064 | parent=self, | ||
MinRK
|
r10312 | autorestart=True, | ||
Matthias BUSSONNIER
|
r5035 | ) | ||
# start the kernel | ||||
Min RK
|
r18933 | kwargs = {} | ||
# FIXME: remove special treatment of IPython kernels | ||||
if self.kernel_manager.ipython_kernel: | ||||
kwargs['extra_arguments'] = self.kernel_argv | ||||
Matthias BUSSONNIER
|
r5039 | kernel_manager.start_kernel(**kwargs) | ||
MinRK
|
r10312 | kernel_manager.client_factory = self.kernel_client_class | ||
kernel_client = kernel_manager.client() | ||||
kernel_client.start_channels(shell=True, iopub=True) | ||||
Matthias BUSSONNIER
|
r5035 | widget = self.widget_factory(config=self.config, | ||
MinRK
|
r5073 | local_kernel=True) | ||
MinRK
|
r6056 | self.init_colors(widget) | ||
Matthias BUSSONNIER
|
r5035 | widget.kernel_manager = kernel_manager | ||
MinRK
|
r10289 | widget.kernel_client = kernel_client | ||
MinRK
|
r5138 | widget._existing = False | ||
widget._may_close = True | ||||
widget._confirm_exit = self.confirm_exit | ||||
Helen ST
|
r19327 | widget._display_banner = self.display_banner | ||
MinRK
|
r5135 | return widget | ||
def new_frontend_slave(self, current_widget): | ||||
"""Create and return a new frontend attached to an existing kernel. | ||||
Parameters | ||||
---------- | ||||
current_widget : IPythonWidget | ||||
The IPythonWidget whose kernel this frontend is to share | ||||
""" | ||||
MinRK
|
r10288 | kernel_client = self.kernel_client_class( | ||
connection_file=current_widget.kernel_client.connection_file, | ||||
MinRK
|
r5073 | config = self.config, | ||
Matthias BUSSONNIER
|
r5035 | ) | ||
MinRK
|
r10288 | kernel_client.load_connection_file() | ||
kernel_client.start_channels() | ||||
Matthias BUSSONNIER
|
r5035 | widget = self.widget_factory(config=self.config, | ||
MinRK
|
r5073 | local_kernel=False) | ||
MinRK
|
r6056 | self.init_colors(widget) | ||
MinRK
|
r5138 | widget._existing = True | ||
widget._may_close = False | ||||
widget._confirm_exit = False | ||||
Helen ST
|
r19327 | widget._display_banner = self.display_banner | ||
MinRK
|
r10288 | widget.kernel_client = kernel_client | ||
MinRK
|
r10312 | widget.kernel_manager = current_widget.kernel_manager | ||
MinRK
|
r5135 | return widget | ||
MinRK
|
r3971 | |||
MinRK
|
r10312 | def init_qt_app(self): | ||
# separate from qt_elements, because it must run first | ||||
self.app = QtGui.QApplication([]) | ||||
MinRK
|
r3971 | def init_qt_elements(self): | ||
# Create the widget. | ||||
Matthias BUSSONNIER
|
r5053 | |||
base_path = os.path.abspath(os.path.dirname(__file__)) | ||||
icon_path = os.path.join(base_path, 'resources', 'icon', 'IPythonConsole.svg') | ||||
Matthias BUSSONNIER
|
r5055 | self.app.icon = QtGui.QIcon(icon_path) | ||
QtGui.QApplication.setWindowIcon(self.app.icon) | ||||
Matthias BUSSONNIER
|
r5028 | |||
MinRK
|
r12110 | ip = self.ip | ||
MinRK
|
r12591 | local_kernel = (not self.existing) or is_local_ip(ip) | ||
MinRK
|
r3971 | self.widget = self.widget_factory(config=self.config, | ||
local_kernel=local_kernel) | ||||
MinRK
|
r6056 | self.init_colors(self.widget) | ||
MinRK
|
r5138 | self.widget._existing = self.existing | ||
self.widget._may_close = not self.existing | ||||
self.widget._confirm_exit = self.confirm_exit | ||||
Helen ST
|
r19171 | self.widget._display_banner = self.display_banner | ||
Matthias BUSSONNIER
|
r5038 | |||
MinRK
|
r3971 | self.widget.kernel_manager = self.kernel_manager | ||
MinRK
|
r10288 | self.widget.kernel_client = self.kernel_client | ||
MinRK
|
r5138 | self.window = MainWindow(self.app, | ||
MinRK
|
r5135 | confirm_exit=self.confirm_exit, | ||
new_frontend_factory=self.new_frontend_master, | ||||
slave_frontend_factory=self.new_frontend_slave, | ||||
) | ||||
Matthias BUSSONNIER
|
r5038 | self.window.log = self.log | ||
Matthias BUSSONNIER
|
r5051 | self.window.add_tab_with_frontend(self.widget) | ||
Dimitry Kloper
|
r16481 | self.window.init_magic_helper() | ||
Dimitry Kloper
|
r16482 | self.window.init_menu_bar() | ||
Matthias BUSSONNIER
|
r5393 | |||
Dan Kilman
|
r10960 | # Ignore on OSX, where there is always a menu bar | ||
if sys.platform != 'darwin' and self.hide_menubar: | ||||
self.window.menuBar().setVisible(False) | ||||
MinRK
|
r6806 | self.window.setWindowTitle('IPython') | ||
MinRK
|
r3971 | |||
MinRK
|
r6056 | def init_colors(self, widget): | ||
MinRK
|
r3971 | """Configure the coloring of the widget""" | ||
# Note: This will be dramatically simplified when colors | ||||
# are removed from the backend. | ||||
# parse the colors arg down to current known labels | ||||
MinRK
|
r12796 | cfg = self.config | ||
colors = cfg.ZMQInteractiveShell.colors if 'ZMQInteractiveShell.colors' in cfg else None | ||||
style = cfg.IPythonWidget.syntax_style if 'IPythonWidget.syntax_style' in cfg else None | ||||
sheet = cfg.IPythonWidget.style_sheet if 'IPythonWidget.style_sheet' in cfg else None | ||||
MinRK
|
r3971 | |||
# 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 | ||
MinRK
|
r6056 | # Configure the style | ||
MinRK
|
r3971 | 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: | ||
MinRK
|
r6056 | # use a default dark/light/bw style | ||
MinRK
|
r3171 | widget.set_default_style(colors=colors) | ||
MinRK
|
r3170 | |||
MinRK
|
r3971 | if self.stylesheet: | ||
MinRK
|
r6056 | # we got an explicit stylesheet | ||
MinRK
|
r3971 | if os.path.isfile(self.stylesheet): | ||
with open(self.stylesheet) as f: | ||||
MinRK
|
r3170 | sheet = f.read() | ||
else: | ||||
MinRK
|
r6056 | raise IOError("Stylesheet %r not found." % self.stylesheet) | ||
if sheet: | ||||
widget.style_sheet = sheet | ||||
widget._style_sheet_changed() | ||||
MinRK
|
r3971 | |||
MinRK
|
r5609 | def init_signal(self): | ||
"""allow clean shutdown on sigint""" | ||||
signal.signal(signal.SIGINT, lambda sig, frame: self.exit(-2)) | ||||
# need a timer, so that QApplication doesn't block until a real | ||||
# Qt event fires (can require mouse movement) | ||||
# timer trick from http://stackoverflow.com/q/4938723/938949 | ||||
timer = QtCore.QTimer() | ||||
# Let the interpreter run each 200 ms: | ||||
timer.timeout.connect(lambda: None) | ||||
timer.start(200) | ||||
# hold onto ref, so the timer doesn't get cleaned up | ||||
self._sigint_timer = timer | ||||
MinRK
|
r5214 | @catch_config_error | ||
MinRK
|
r3971 | def initialize(self, argv=None): | ||
MinRK
|
r10312 | self.init_qt_app() | ||
MinRK
|
r3971 | super(IPythonQtConsoleApp, self).initialize(argv) | ||
MinRK
|
r5618 | IPythonConsoleApp.initialize(self,argv) | ||
MinRK
|
r3971 | self.init_qt_elements() | ||
MinRK
|
r5609 | self.init_signal() | ||
MinRK
|
r3971 | |||
def start(self): | ||||
# draw the window | ||||
Dan Kilman
|
r10960 | if self.maximize: | ||
self.window.showMaximized() | ||||
else: | ||||
self.window.show() | ||||
Matthias BUSSONNIER
|
r5712 | self.window.raise_() | ||
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() | ||||