text_console
text_console is a customizable, Tkinter-based interactive shell widget that lets users type commands into a console pane and have them evaluated in your application’s Python environment. It supports arrows and command history. It’s designed for embedding in GUI apps, debugging, scripting, and educational tools as an API Playground.
It can also be used as a standalone Python command interpreter.
This program reflects the core functionality of wxPython Shell, offering an even richer feature set.
Key Features
Live code execution
Send single- or multi-line Python code to the interpreter and see results immediately.
Integrated application context
Access and modify your app’s variables and functions via the console_locals namespace.
- Advanced editing
- Keyboard shortcuts: all standard keyboard shortcut are allowed
- Multiline editing: Single line and multiline editing is allowed, also with copy/paste features. When pressing enter within an edited line, a popup appears to ask the requested action (execute the command, add a new line, abort). Shift-Enter is also allowed.
- Prompt Protection: The prompt area (
>>>
or...
) is protected. The cursor cannot move into or before the prompt, and editing actions (insertion, deletion) are blocked in the prompt area. - Smart Arrow Navigation: Left and right arrow keys skip over prompt tags and any protected regions, ensuring the cursor only lands in editable areas. Arrow navigation also respects line boundaries and prompt positions.
- Home/End Navigation: The
Home
andEnd
keys move the cursor to the beginning or end of the current line, but never into the prompt area. - Undo/Redo Support: Full undo/redo support is enabled (
Ctrl+Z
/Ctrl+Y
), with fine-grained control for character-by-character undo. - Tab and Shift+Tab: Pressing
Tab
inserts four spaces. PressingShift+Tab
removes up to four spaces. - Selection Awareness: Editing and navigation actions are aware of text selection. For example, custom arrow key logic is bypassed when a selection is active.
- Clear Console: The console can be cleared with a single command, automatically restoring the prompt and positioning the cursor for new input.
Command history
Navigate previous commands with ↑/↓ arrows; history is saved to a file you choose.
Cut/Copy/Paste/Clear
Right-click context menu (and customizable via context_menu_items) for text editing.
Customizable UI
The package provides flexibility to customize:
history_file
: Change the location of the history fileconsole_locals
: Add custom variables and functions to the console’s namespacecontext_menu_items
: Modify the right-click context menushow_about_message
: Customize the about dialog contentshow_help_content
: Customize the help window contentcreate_menu
: Override to completely customize the menu bar
Subclass-friendly
Extend the TextConsole class and override any of the above to fit your needs.
Keyboard shortcuts
Shortcut | Description |
---|---|
Return | Execute current command; if editing mid-line or multiline, show modal allowing to select either to run the code or insert a linefeed in the cursor position. |
Shift Return | Insert a new line within the edited multiline command. |
Control Return | Move the cursor to the end of the last line of input. |
Tab | Indent code (up to 4 spaces); if selection, indent all selected lines. |
Shift Tab | Un-indent (remove up to 4 spaces before cursor). |
Down Arrow | Recall next command in history. |
Up Arrow | Recall previous command in history. |
Left Arrow | Move to previous non-tagged character. |
Right Arrow | Move to next non-tagged character. |
Control Left Arrow | Move to previous word. |
Control Right Arrow | Move to next word. |
Escape | Jump to last blank command (empty input) in history. |
BackSpace | Delete character before cursor. |
Control R | Open Command History panel / Start reverse history search. |
Control C | Copy selected code, removing prompts first. |
Control V | Paste text from clipboard, handling prompts and multiline input. |
Button-3 (Right Click) | Show context menu (Cut, Copy, Paste, Clear). |
Control Z | Undo last edit (safe, ignores errors). |
Control Y | Redo last undone edit (safe, ignores errors). |
Control K | Remove the current element from the history. |
Home | Move cursor to start of current line. |
End | Move cursor to end of current line. |
Control + | Increase font size. |
Control - | Decrease font size. |
Control 0 | Reset font size. |
In the History Panel:
Shortcut | Description |
---|---|
Control N | In history panel: Find next search match |
Control B | In history panel: Find previous search match |
Control S | In history panel: Load selected command to main window |
Double-click | Recall command to main window from history panel |
Esc | In history panel: Close Command History panel / Cancel search |
Installation
pip install text-console
Playground
python -m text_console
Available options:
Python Console [-h] [-V]
optional arguments:
-h, --help show this help message and exit
-V, --version Print version and exit
A customizable Tkinter-based text console widget.
Running the pre-built GUI executable
The text_console.zip archive in the Releases folder incudes the text_console.exe executable asset; the ZIP archive is auto-generated by a GitHub Action. text_console.exe is a Windows GUI that can be directly executed.
Basic usage with default settings
import tkinter as tk
from text_console import TextConsole
class TkConsole(tk.Tk):
def __init__(self):
super().__init__()
self.title("Python Console")
self.geometry("800x400")
# Initialize the TextConsole widget
console = TextConsole(self, self)
console.pack(fill='both', expand=True)
# Configure grid resizing for the main window
self.grid_rowconfigure(0, weight=1)
self.grid_columnconfigure(0, weight=1)
app = TkConsole()
app.mainloop()
Invoking TextConsole from a Master widget
import tkinter as tk
from text_console import TextConsole
class TkConsole(tk.Tk):
def __init__(self):
super().__init__()
self.title("Python Console")
self.geometry("100x70")
# Add a button to launch the TextConsole
run_console_button = tk.Button(
self,
text="Debug Console",
command=self.run_text_console
)
run_console_button.pack(pady=20) # Add some spacing around the button
# Configure grid resizing for the main window
self.grid_rowconfigure(0, weight=1)
self.grid_columnconfigure(0, weight=1)
def run_text_console(self):
"""Launches the TextConsole in a new Toplevel window."""
console_window = tk.Toplevel(self)
console_window.title("Debug Console")
console_window.geometry("800x400")
# Initialize the TextConsole widget
console = TextConsole(self, console_window)
console.pack(fill='both', expand=True)
app = TkConsole()
app.mainloop()
Customized console through subclassing
from text_console import TextConsole
class MyCustomConsole(TextConsole):
# Override class attributes
history_file = "my_custom_history.txt"
console_locals = {
"my_var": 42,
"my_function": lambda x: x * 2
}
context_menu_items = [
("Custom Action", "custom_action"),
"-", # separator
("Clear", "clear")
]
show_about_message = "My Custom Console v1.0"
show_help_content = "This is my custom console help content"
def custom_action(self):
print("Custom action executed!")
def create_menu(self, master):
# Override to create a custom menu
super().create_menu(main, master)
# Add "Web Site" to the Help menu
menu_bar = master.nametowidget(master.cget('menu')) # Get the menu widget
help_menu = list(menu_bar.children.values())[2] # Access the Help menu (third = 2)
help_menu.insert_command(
help_menu.index("end"),
label="Web Site",
command=self.new_action
)
# Override to create a custom menu
menu_bar = Menu(master)
master.config(menu=menu_bar)
# Custom menu items
custom_menu = Menu(menu_bar, tearoff=0)
custom_menu.add_command(label="My Action", command=self.custom_action)
menu_bar.add_cascade(label="Custom", menu=custom_menu)
def new_action(self):
pass
""" Alternatively, override create_menu:
def create_menu(self, master):
# Override to create a custom menu
menu_bar = Menu(master)
master.config(menu=menu_bar)
# Custom menu items
custom_menu = Menu(menu_bar, tearoff=0)
custom_menu.add_command(label="My Action", command=self.custom_action)
menu_bar.add_cascade(label="Custom", menu=custom_menu)
"""
# Use the custom console
text_console = MyCustomConsole(main, master)
Limitations
text_console does not support Ctrl+C to abort long‑running commands. Because the console is designed as an embeddable in-process API playground within a Tkinter process, it cannot leverage multiprocessing to provide real interrupt semantics, which would fork a separate Python environment, breaking the API. Likewise, multithreading cannot forcibly terminate an executing function due to Python’s Global Interpreter Lock (GIL) and the absence of a built-in thread‑kill API. Furthermore, Tkinter itself offers no mechanism to cancel an in‑progress widget callback or command/code evaluation and it runs inside an event loop rather than the standard Python REPL. Consequently, users must wait for blocking operations to finish.