19 changed files with 392 additions and 163 deletions
@ -0,0 +1,15 @@
|
||||
Summary of the main pyTermTk Threading (26/Oct/2025) [8148cf0a] |
||||
```text |
||||
|
||||
MainThread (mainlop) |
||||
│ └─▶ Wait on inputQueue ─> Signal ┬─▶ inputEvent |
||||
│ ▲ └─▶ pasteEvent |
||||
│ │ |
||||
│ (push to inputQueue) |
||||
├─▶ TTkInput Thread │ |
||||
│ └─▶ Wait on select(Stdin) |
||||
│ |
||||
└─▶ TTk (Draw) |
||||
└─▶ Wait on timeout (~65 fps) ──▶ Refresh Screen |
||||
|
||||
``` |
||||
@ -0,0 +1,36 @@
|
||||
import threading |
||||
import traceback |
||||
|
||||
def custom_thread_excepthook(args): |
||||
""" |
||||
Args is a named tuple with: |
||||
- exc_type: Exception class |
||||
- exc_value: Exception instance |
||||
- exc_traceback: Traceback object |
||||
- thread: Thread object where exception occurred |
||||
""" |
||||
print(f"Thread {args.thread.name} crashed!") |
||||
print(f"Exception: {args.exc_type.__name__}: {args.exc_value}") |
||||
|
||||
# Print full traceback like default handler |
||||
print("\nFull traceback:") |
||||
traceback.print_exception(args.exc_type, args.exc_value, args.exc_traceback, colorize=True) |
||||
|
||||
# Or to match default format exactly: |
||||
# ßsys.stderr.write(f"\n\nPIPPO\n\nException in thread {args.thread.name}:\n") |
||||
# traceback.print_exception(args.exc_type, args.exc_value, args.exc_traceback, file=sys.stderr) |
||||
|
||||
# Set custom handler |
||||
threading.excepthook = custom_thread_excepthook |
||||
|
||||
# Example thread that will crash |
||||
def worker(): |
||||
raise ValueError("Something went wrong!") |
||||
|
||||
thread = threading.Thread(name='Pippo', target=worker) |
||||
try: |
||||
thread.start() |
||||
thread.join() |
||||
except Exception as e: |
||||
print(e) |
||||
print('EUGENIO') |
||||
@ -0,0 +1,70 @@
|
||||
#!/usr/bin/env python3 |
||||
# MIT License |
||||
# |
||||
# Copyright (c) 2025 Eugenio Parodi <ceccopierangiolieugenio AT googlemail DOT com> |
||||
# |
||||
# Permission is hereby granted, free of charge, to any person obtaining a copy |
||||
# of this software and associated documentation files (the "Software"), to deal |
||||
# in the Software without restriction, including without limitation the rights |
||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
||||
# copies of the Software, and to permit persons to whom the Software is |
||||
# furnished to do so, subject to the following conditions: |
||||
# |
||||
# The above copyright notice and this permission notice shall be included in all |
||||
# copies or substantial portions of the Software. |
||||
# |
||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
||||
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
||||
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
||||
# SOFTWARE. |
||||
|
||||
import sys, os |
||||
import threading |
||||
|
||||
sys.path.append(os.path.join(sys.path[0],'../../libs/pyTermTk')) |
||||
import TermTk as ttk |
||||
|
||||
ttk.TTkLog.use_default_file_logging() |
||||
|
||||
@ttk.pyTTkSlot() |
||||
def _raise(): |
||||
raise Exception('_raise FAIL!!!') |
||||
|
||||
class Fail(ttk.TTkWidget): |
||||
fail_state:bool = False |
||||
def mousePressEvent(self, evt): |
||||
Fail.fail_state = True |
||||
self.update() |
||||
return True |
||||
def paintEvent(self, canvas): |
||||
if Fail.fail_state: |
||||
raise Exception('Fail FAIL!!!') |
||||
canvas.fill(color=ttk.TTkColor.BG_RED) |
||||
canvas.drawText(text='X', pos=(2,1)) |
||||
|
||||
timer = ttk.TTkTimer() |
||||
timer.timeout.connect(_raise) |
||||
|
||||
root = ttk.TTk() |
||||
|
||||
# simulate error |
||||
|
||||
# Fail in the main Thread |
||||
ttk.TTkButton(parent=root,text=' X ',border=True).clicked.connect(_raise) |
||||
# Fail in the draw Thread |
||||
Fail(parent=root,pos=(0,3), size=(5,3)) |
||||
# Generic Failure on a generic thread |
||||
ttk.TTkButton(parent=root,pos=(0,6),text=' X ',border=True).clicked.connect(lambda:threading.Thread(target=_raise).start()) |
||||
# Generic Failure on a TTkTimer |
||||
ttk.TTkButton(parent=root,pos=(0,9),text=' X ',border=True).clicked.connect(lambda:timer.start()) |
||||
|
||||
# Generic Quit |
||||
ttk.TTkButton(parent=root,pos=(0,13),text=' X ',border=True).clicked.connect(root.quit) |
||||
ttk.TTkButton(parent=root,pos=(0,16),text=' X ',border=True).clicked.connect(ttk.TTkHelper.quit) |
||||
|
||||
win=ttk.TTkWindow(parent=root, pos=(5,0), size=(100,30), border=True, layout=ttk.TTkGridLayout()) |
||||
ttk.TTkLogViewer(parent=win) |
||||
root.mainloop() |
||||
Loading…
Reference in new issue