2025-12-28 21:38:21 +01:00
|
|
|
|
2025-12-30 21:20:27 +01:00
|
|
|
from PyQt6.QtWidgets import (
|
|
|
|
|
QMainWindow, QWidget, QVBoxLayout, QHBoxLayout,
|
|
|
|
|
QListWidget, QListWidgetItem, QComboBox, QLabel, QLineEdit, QPushButton,
|
|
|
|
|
QFileDialog, QMenuBar, QMenu, QMessageBox, QStatusBar, QProgressBar, QApplication
|
|
|
|
|
)
|
2025-12-28 21:57:44 +01:00
|
|
|
|
|
|
|
|
# ... (imports remain)
|
|
|
|
|
|
|
|
|
|
# ... (imports remain)
|
2025-12-28 21:38:21 +01:00
|
|
|
from PyQt6.QtGui import QAction, QKeyEvent
|
|
|
|
|
from PyQt6.QtCore import Qt
|
|
|
|
|
|
|
|
|
|
from .io import load_t42, save_t42
|
|
|
|
|
from .renderer import TeletextCanvas
|
2025-12-30 21:20:27 +01:00
|
|
|
import sys
|
|
|
|
|
import os
|
2025-12-28 21:38:21 +01:00
|
|
|
from .models import TeletextService, Page, Packet
|
|
|
|
|
|
|
|
|
|
class MainWindow(QMainWindow):
|
|
|
|
|
def __init__(self):
|
|
|
|
|
super().__init__()
|
|
|
|
|
self.setWindowTitle("Teletext Editor")
|
|
|
|
|
self.resize(1024, 768)
|
|
|
|
|
|
|
|
|
|
self.service = TeletextService()
|
|
|
|
|
self.current_page: Page = None
|
|
|
|
|
|
|
|
|
|
# UI Components
|
|
|
|
|
self.central_widget = QWidget()
|
|
|
|
|
self.setCentralWidget(self.central_widget)
|
|
|
|
|
|
|
|
|
|
self.layout = QHBoxLayout(self.central_widget)
|
|
|
|
|
|
2025-12-28 21:41:09 +01:00
|
|
|
# Left Panel: Page List
|
|
|
|
|
left_layout = QVBoxLayout()
|
|
|
|
|
left_label = QLabel("Pages")
|
|
|
|
|
left_layout.addWidget(left_label)
|
2025-12-28 21:38:21 +01:00
|
|
|
|
2025-12-28 21:41:09 +01:00
|
|
|
self.page_list = QListWidget()
|
|
|
|
|
self.page_list.setFixedWidth(150)
|
|
|
|
|
self.page_list.itemClicked.connect(self.on_page_selected)
|
|
|
|
|
left_layout.addWidget(self.page_list)
|
|
|
|
|
|
2025-12-28 21:57:44 +01:00
|
|
|
# Hex Inspector
|
|
|
|
|
hex_layout = QVBoxLayout()
|
|
|
|
|
hex_label = QLabel("Hex Value:")
|
|
|
|
|
self.hex_input = QLineEdit()
|
|
|
|
|
self.hex_input.setMaxLength(2)
|
|
|
|
|
self.hex_input.setPlaceholderText("00")
|
|
|
|
|
self.hex_input.returnPressed.connect(self.on_hex_entered)
|
|
|
|
|
hex_layout.addWidget(hex_label)
|
|
|
|
|
hex_layout.addWidget(self.hex_input)
|
|
|
|
|
left_layout.addLayout(hex_layout)
|
|
|
|
|
|
|
|
|
|
left_layout.addStretch()
|
|
|
|
|
|
2025-12-28 21:41:09 +01:00
|
|
|
self.layout.addLayout(left_layout)
|
|
|
|
|
|
|
|
|
|
# Center Area Layout (Top Bar + Canvas)
|
|
|
|
|
center_layout = QVBoxLayout()
|
|
|
|
|
|
|
|
|
|
# Top Bar: Subpage Selector
|
|
|
|
|
top_bar = QHBoxLayout()
|
|
|
|
|
self.subpage_label = QLabel("Subpage:")
|
|
|
|
|
self.subpage_combo = QComboBox()
|
|
|
|
|
self.subpage_combo.setMinimumWidth(250)
|
|
|
|
|
self.subpage_combo.currentIndexChanged.connect(self.on_subpage_changed)
|
2025-12-30 21:07:12 +01:00
|
|
|
|
|
|
|
|
self.btn_prev_sub = QPushButton("<")
|
|
|
|
|
self.btn_prev_sub.setFixedWidth(30)
|
|
|
|
|
self.btn_prev_sub.clicked.connect(self.prev_subpage)
|
|
|
|
|
|
|
|
|
|
self.btn_next_sub = QPushButton(">")
|
|
|
|
|
self.btn_next_sub.setFixedWidth(30)
|
|
|
|
|
self.btn_next_sub.clicked.connect(self.next_subpage)
|
|
|
|
|
|
2025-12-28 21:41:09 +01:00
|
|
|
top_bar.addWidget(self.subpage_label)
|
2025-12-30 21:07:12 +01:00
|
|
|
top_bar.addWidget(self.btn_prev_sub)
|
2025-12-28 21:41:09 +01:00
|
|
|
top_bar.addWidget(self.subpage_combo)
|
2025-12-30 21:07:12 +01:00
|
|
|
top_bar.addWidget(self.btn_next_sub)
|
2025-12-28 21:41:09 +01:00
|
|
|
top_bar.addStretch()
|
|
|
|
|
|
|
|
|
|
center_layout.addLayout(top_bar)
|
|
|
|
|
|
2025-12-28 21:57:44 +01:00
|
|
|
# Color Shortcuts
|
|
|
|
|
color_layout = QHBoxLayout()
|
|
|
|
|
colors = [
|
|
|
|
|
("Red", 0x01, "#FF0000"),
|
|
|
|
|
("Green", 0x02, "#00FF00"),
|
|
|
|
|
("Yellow", 0x03, "#FFFF00"),
|
|
|
|
|
("Blue", 0x04, "#0000FF"),
|
|
|
|
|
("Magenta", 0x05, "#FF00FF"),
|
|
|
|
|
("Cyan", 0x06, "#00FFFF"),
|
|
|
|
|
("White", 0x07, "#FFFFFF"),
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
for name, code, hex_color in colors:
|
|
|
|
|
btn = QPushButton(name)
|
|
|
|
|
btn.setStyleSheet(f"background-color: {hex_color}; font-weight: bold; color: black;")
|
|
|
|
|
btn.clicked.connect(lambda checked, c=code: self.insert_char(c))
|
|
|
|
|
color_layout.addWidget(btn)
|
|
|
|
|
|
|
|
|
|
color_layout.addStretch()
|
|
|
|
|
center_layout.addLayout(color_layout)
|
|
|
|
|
|
2025-12-28 21:41:09 +01:00
|
|
|
# Canvas
|
2025-12-28 21:38:21 +01:00
|
|
|
self.canvas = TeletextCanvas()
|
2025-12-28 21:57:44 +01:00
|
|
|
self.canvas.cursorChanged.connect(self.on_cursor_changed)
|
2025-12-28 21:41:09 +01:00
|
|
|
center_layout.addWidget(self.canvas, 1) # Expand
|
|
|
|
|
|
|
|
|
|
self.layout.addLayout(center_layout, 1)
|
2025-12-28 21:38:21 +01:00
|
|
|
|
|
|
|
|
# Menus
|
|
|
|
|
self.create_menus()
|
|
|
|
|
|
|
|
|
|
def create_menus(self):
|
|
|
|
|
menu_bar = self.menuBar()
|
|
|
|
|
|
|
|
|
|
file_menu = menu_bar.addMenu("File")
|
|
|
|
|
|
|
|
|
|
open_action = QAction("Open T42...", self)
|
|
|
|
|
open_action.triggered.connect(self.open_file)
|
|
|
|
|
file_menu.addAction(open_action)
|
|
|
|
|
|
|
|
|
|
save_action = QAction("Save T42...", self)
|
|
|
|
|
save_action.triggered.connect(self.save_file)
|
|
|
|
|
file_menu.addAction(save_action)
|
|
|
|
|
|
2025-12-30 21:05:25 +01:00
|
|
|
close_action = QAction("Close File", self)
|
|
|
|
|
close_action.triggered.connect(self.close_file)
|
|
|
|
|
file_menu.addAction(close_action)
|
|
|
|
|
|
|
|
|
|
file_menu.addSeparator()
|
|
|
|
|
|
2025-12-28 21:38:21 +01:00
|
|
|
exit_action = QAction("Exit", self)
|
|
|
|
|
exit_action.triggered.connect(self.close)
|
|
|
|
|
file_menu.addAction(exit_action)
|
|
|
|
|
|
|
|
|
|
view_menu = menu_bar.addMenu("View")
|
|
|
|
|
|
|
|
|
|
lang_menu = view_menu.addMenu("Language")
|
|
|
|
|
langs = ["English", "German", "Swedish/Finnish", "Italian", "French", "Portuguese/Spanish", "Turkish", "Romania"]
|
|
|
|
|
for i, lang in enumerate(langs):
|
|
|
|
|
action = QAction(lang, self)
|
|
|
|
|
action.setData(i)
|
|
|
|
|
action.triggered.connect(self.set_language)
|
|
|
|
|
lang_menu.addAction(action)
|
|
|
|
|
|
|
|
|
|
def set_language(self):
|
|
|
|
|
action = self.sender()
|
|
|
|
|
if action:
|
|
|
|
|
idx = action.data()
|
|
|
|
|
self.canvas.subset_idx = idx
|
|
|
|
|
self.canvas.redraw()
|
|
|
|
|
self.canvas.update()
|
|
|
|
|
|
2025-12-30 21:07:12 +01:00
|
|
|
def prev_subpage(self):
|
|
|
|
|
count = self.subpage_combo.count()
|
|
|
|
|
if count <= 1: return
|
|
|
|
|
|
|
|
|
|
current = self.subpage_combo.currentIndex()
|
|
|
|
|
new_index = (current - 1) % count
|
|
|
|
|
self.subpage_combo.setCurrentIndex(new_index)
|
|
|
|
|
|
|
|
|
|
def next_subpage(self):
|
|
|
|
|
count = self.subpage_combo.count()
|
|
|
|
|
if count <= 1: return
|
|
|
|
|
|
|
|
|
|
current = self.subpage_combo.currentIndex()
|
|
|
|
|
new_index = (current + 1) % count
|
|
|
|
|
self.subpage_combo.setCurrentIndex(new_index)
|
|
|
|
|
|
2025-12-30 21:05:25 +01:00
|
|
|
def close_file(self):
|
|
|
|
|
# Reset everything
|
|
|
|
|
self.service = TeletextService()
|
|
|
|
|
self.current_page = None
|
2025-12-30 21:20:27 +01:00
|
|
|
self.current_file_path = None
|
2025-12-30 21:05:25 +01:00
|
|
|
self.populate_list()
|
|
|
|
|
self.subpage_combo.clear()
|
|
|
|
|
self.page_groups = {}
|
|
|
|
|
# Clear canvas
|
|
|
|
|
self.canvas.set_page(None)
|
|
|
|
|
# Maybe reset text of hex input
|
|
|
|
|
self.hex_input.clear()
|
|
|
|
|
QMessageBox.information(self, "Closed", "File closed.")
|
2025-12-30 21:20:27 +01:00
|
|
|
self.status_label.setText("Ready")
|
2025-12-30 21:05:25 +01:00
|
|
|
|
2025-12-28 21:38:21 +01:00
|
|
|
def open_file(self):
|
|
|
|
|
fname, _ = QFileDialog.getOpenFileName(self, "Open T42", "", "Teletext Files (*.t42);;All Files (*)")
|
|
|
|
|
if fname:
|
|
|
|
|
try:
|
2025-12-30 21:20:27 +01:00
|
|
|
self.progress_bar.setVisible(True)
|
|
|
|
|
self.status_label.setText(f"Loading {os.path.basename(fname)}...")
|
|
|
|
|
self.progress_bar.setValue(0)
|
|
|
|
|
|
|
|
|
|
self.service = load_t42(fname, progress_callback=self.update_progress)
|
|
|
|
|
self.current_file_path = fname
|
2025-12-28 21:41:09 +01:00
|
|
|
self.populate_list()
|
2025-12-30 21:20:27 +01:00
|
|
|
|
|
|
|
|
self.progress_bar.setVisible(False)
|
|
|
|
|
self.status_label.setText(f"Loaded {len(self.service.pages)} pages from {os.path.basename(fname)}")
|
2025-12-28 21:38:21 +01:00
|
|
|
except Exception as e:
|
2025-12-30 21:20:27 +01:00
|
|
|
self.progress_bar.setVisible(False)
|
2025-12-28 21:38:21 +01:00
|
|
|
QMessageBox.critical(self, "Error", f"Failed to load file: {e}")
|
2025-12-30 21:20:27 +01:00
|
|
|
self.status_label.setText("Error loading file")
|
2025-12-28 21:38:21 +01:00
|
|
|
|
|
|
|
|
def save_file(self):
|
2025-12-30 21:20:27 +01:00
|
|
|
if not self.current_file_path:
|
|
|
|
|
# Logic for "Save As" if path not known, but for T42 we usually overwrite or ask.
|
|
|
|
|
# To keep it simple, ask every time or track path.
|
|
|
|
|
# Let's ask.
|
|
|
|
|
fname, _ = QFileDialog.getSaveFileName(self, "Save T42", "", "Teletext Files (*.t42)")
|
|
|
|
|
if not fname: return
|
|
|
|
|
self.current_file_path = fname
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
self.progress_bar.setVisible(True)
|
|
|
|
|
self.status_label.setText(f"Saving {os.path.basename(self.current_file_path)}...")
|
|
|
|
|
|
|
|
|
|
save_t42(self.current_file_path, self.service, progress_callback=self.update_progress)
|
|
|
|
|
|
|
|
|
|
self.progress_bar.setVisible(False)
|
|
|
|
|
self.status_label.setText(f"Saved {len(self.service.pages)} pages to {os.path.basename(self.current_file_path)}")
|
|
|
|
|
except Exception as e:
|
|
|
|
|
self.progress_bar.setVisible(False)
|
|
|
|
|
QMessageBox.critical(self, "Error", f"Failed to save file: {e}")
|
|
|
|
|
self.status_label.setText("Error saving file")
|
2025-12-28 21:38:21 +01:00
|
|
|
|
2025-12-28 21:41:09 +01:00
|
|
|
def populate_list(self):
|
|
|
|
|
self.page_list.clear()
|
|
|
|
|
|
|
|
|
|
# Group pages by Mag+PageNum
|
|
|
|
|
# We want unique list items
|
|
|
|
|
self.page_groups = {} # Key: (mag, page_num) -> List[Page]
|
2025-12-28 21:38:21 +01:00
|
|
|
|
|
|
|
|
for p in self.service.pages:
|
2025-12-28 21:41:09 +01:00
|
|
|
key = (p.magazine, p.page_number)
|
|
|
|
|
if key not in self.page_groups:
|
|
|
|
|
self.page_groups[key] = []
|
|
|
|
|
self.page_groups[key].append(p)
|
2025-12-28 21:38:21 +01:00
|
|
|
|
2025-12-28 21:41:09 +01:00
|
|
|
# Sort keys
|
|
|
|
|
sorted_keys = sorted(self.page_groups.keys())
|
|
|
|
|
|
|
|
|
|
for mag, pnum in sorted_keys:
|
|
|
|
|
label = f"{mag}{pnum:02d}"
|
|
|
|
|
item = QListWidgetItem(label)
|
|
|
|
|
item.setData(Qt.ItemDataRole.UserRole, (mag, pnum))
|
|
|
|
|
self.page_list.addItem(item)
|
|
|
|
|
|
|
|
|
|
def on_page_selected(self, item):
|
|
|
|
|
mag, pnum = item.data(Qt.ItemDataRole.UserRole)
|
|
|
|
|
pages = self.page_groups.get((mag, pnum), [])
|
|
|
|
|
|
|
|
|
|
# Populate Subpage Combo
|
|
|
|
|
self.subpage_combo.blockSignals(True)
|
|
|
|
|
self.subpage_combo.clear()
|
|
|
|
|
|
|
|
|
|
for i, p in enumerate(pages):
|
|
|
|
|
# Display format: Index or Subcode?
|
|
|
|
|
# Subcode is often 0000. Index 1/N is clearer for editing.
|
|
|
|
|
label = f"{i+1}/{len(pages)} (Sub {p.sub_code:04X})"
|
|
|
|
|
self.subpage_combo.addItem(label, p)
|
2025-12-28 21:38:21 +01:00
|
|
|
|
2025-12-28 21:41:09 +01:00
|
|
|
self.subpage_combo.blockSignals(False)
|
|
|
|
|
|
|
|
|
|
if pages:
|
|
|
|
|
self.subpage_combo.setCurrentIndex(0)
|
|
|
|
|
# Trigger update (manual because blockSignals)
|
|
|
|
|
self.on_subpage_changed(0)
|
2025-12-28 21:38:21 +01:00
|
|
|
|
2025-12-28 21:41:09 +01:00
|
|
|
def on_subpage_changed(self, index):
|
|
|
|
|
if index < 0: return
|
|
|
|
|
page = self.subpage_combo.itemData(index)
|
2025-12-28 21:38:21 +01:00
|
|
|
if isinstance(page, Page):
|
|
|
|
|
self.current_page = page
|
|
|
|
|
self.canvas.set_page(page)
|
|
|
|
|
self.canvas.setFocus()
|
2025-12-28 21:57:44 +01:00
|
|
|
|
|
|
|
|
def insert_char(self, char_code):
|
|
|
|
|
self.canvas.set_byte_at_cursor(char_code)
|
|
|
|
|
# Advance cursor
|
|
|
|
|
self.canvas.move_cursor(1, 0)
|
|
|
|
|
self.canvas.setFocus()
|
|
|
|
|
|
|
|
|
|
def on_cursor_changed(self, x, y, val):
|
|
|
|
|
self.hex_input.setText(f"{val:02X}")
|
|
|
|
|
|
|
|
|
|
def on_hex_entered(self):
|
|
|
|
|
text = self.hex_input.text()
|
|
|
|
|
try:
|
|
|
|
|
val = int(text, 16)
|
|
|
|
|
if 0 <= val <= 255:
|
|
|
|
|
# Update canvas input
|
|
|
|
|
# We can call handle_input with char, OR set byte directly.
|
|
|
|
|
# Direct byte set is safer for non-printable.
|
|
|
|
|
self.canvas.set_byte_at_cursor(val)
|
|
|
|
|
self.canvas.setFocus() # Return focus to canvas
|
|
|
|
|
except ValueError:
|
|
|
|
|
pass # Ignore invalid hex
|
2025-12-28 21:38:21 +01:00
|
|
|
|
|
|
|
|
# Input Handling (Editor Logic)
|
|
|
|
|
def keyPressEvent(self, event: QKeyEvent):
|
|
|
|
|
if not self.current_page:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
key = event.key()
|
|
|
|
|
text = event.text()
|
|
|
|
|
|
|
|
|
|
# Navigation
|
|
|
|
|
if key == Qt.Key.Key_Up:
|
|
|
|
|
self.canvas.move_cursor(0, -1)
|
|
|
|
|
elif key == Qt.Key.Key_Down:
|
|
|
|
|
self.canvas.move_cursor(0, 1)
|
|
|
|
|
elif key == Qt.Key.Key_Left:
|
|
|
|
|
self.canvas.move_cursor(-1, 0)
|
|
|
|
|
elif key == Qt.Key.Key_Right:
|
|
|
|
|
self.canvas.move_cursor(1, 0)
|
2025-12-28 21:57:44 +01:00
|
|
|
elif key == Qt.Key.Key_Return or key == Qt.Key.Key_Enter:
|
|
|
|
|
# Move to start of next line
|
|
|
|
|
self.canvas.cursor_x = 0
|
|
|
|
|
self.canvas.move_cursor(0, 1)
|
2025-12-28 21:38:21 +01:00
|
|
|
else:
|
|
|
|
|
# Typing
|
2025-12-30 21:05:25 +01:00
|
|
|
# Allow wider range of chars for national support
|
|
|
|
|
if text and len(text) == 1 and ord(text) >= 32:
|
2025-12-28 21:38:21 +01:00
|
|
|
self.canvas.handle_input(text)
|
|
|
|
|
elif key == Qt.Key.Key_Backspace:
|
|
|
|
|
# Move back and delete
|
|
|
|
|
self.canvas.move_cursor(-1, 0)
|
|
|
|
|
self.canvas.handle_input(' ')
|
|
|
|
|
self.canvas.move_cursor(-1, 0) # Compensate for the auto-advance logic if any
|
|
|
|
|
|