refactor: update camera control signals and improve live view handling
This commit is contained in:
@@ -112,7 +112,7 @@ class MainController:
|
||||
@Slot()
|
||||
def on_camera_started(self):
|
||||
"""Updates UI when the camera stream starts."""
|
||||
# self.split_view.show_live_view()
|
||||
self.split_view.toggle_live_view()
|
||||
self.welcome_view.set_button_text("Stop Camera")
|
||||
# Re-route button click to stop the camera
|
||||
self.welcome_view.camera_start_btn.clicked.disconnect()
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
from PySide6.QtCore import QObject, QTimer, Signal, Slot, QMutex, QMutexLocker
|
||||
from PySide6.QtCore import QObject, QTimer, Signal, Slot, QMutex, QMutexLocker, QThread
|
||||
from PySide6.QtGui import QImage, QPixmap
|
||||
import cv2
|
||||
|
||||
@@ -9,6 +9,7 @@ class CameraWorker(QObject):
|
||||
frame_ready = Signal(QPixmap)
|
||||
photo_ready = Signal(QPixmap)
|
||||
error_occurred = Signal(str)
|
||||
camera_ready = Signal(bool)
|
||||
|
||||
def __init__(self, parent: QObject | None = None) -> None:
|
||||
super().__init__(parent)
|
||||
@@ -44,8 +45,10 @@ class CameraWorker(QObject):
|
||||
|
||||
if self.camera.connect():
|
||||
self.is_connected = True
|
||||
self.camera_ready.emit(True)
|
||||
else:
|
||||
self.is_connected = False
|
||||
self.camera_ready.emit(False)
|
||||
self.error_occurred.emit(self.camera.get_error_msg())
|
||||
|
||||
@Slot()
|
||||
@@ -98,14 +101,13 @@ class CameraWorker(QObject):
|
||||
pixmap = QPixmap.fromImage(qimg)
|
||||
self.frame_ready.emit(pixmap)
|
||||
|
||||
def isConnected(self):
|
||||
return self.is_connected
|
||||
|
||||
|
||||
class CameraController(QObject):
|
||||
frame_ready = Signal(QPixmap)
|
||||
photo_ready = Signal(QPixmap)
|
||||
error_occurred = Signal(str)
|
||||
camera_ready = Signal(bool)
|
||||
|
||||
# Signals to command the worker
|
||||
_set_camera_requested = Signal(BaseCamera, int)
|
||||
@@ -125,6 +127,7 @@ class CameraController(QObject):
|
||||
self._worker.frame_ready.connect(self.frame_ready)
|
||||
self._worker.photo_ready.connect(self.photo_ready)
|
||||
self._worker.error_occurred.connect(self.error_occurred)
|
||||
self._worker.camera_ready.connect(self.camera_ready)
|
||||
|
||||
# Connect controller's command signals to worker's slots
|
||||
self._set_camera_requested.connect(self._worker.set_camera)
|
||||
@@ -157,5 +160,3 @@ class CameraController(QObject):
|
||||
def stop_stream(self) -> None:
|
||||
self._stop_stream_requested.emit()
|
||||
|
||||
def is_connected(self):
|
||||
return self._worker.isConnected()
|
||||
@@ -1,4 +1,4 @@
|
||||
from PySide6.QtCore import QObject, Signal, QRunnable, QThreadPool, QThread
|
||||
from PySide6.QtCore import QObject, Signal
|
||||
from PySide6.QtGui import QPixmap
|
||||
|
||||
from .camera_controller import CameraController
|
||||
@@ -7,149 +7,108 @@ from .opencv_camera import OpenCvCamera
|
||||
from .base_camera import BaseCamera
|
||||
|
||||
|
||||
class CameraDetectionWorker(QRunnable):
|
||||
class CameraManager(QObject):
|
||||
"""
|
||||
Worker thread for detecting cameras to avoid blocking the GUI.
|
||||
Zarządza wszystkimi operacjami związanymi z kamerami,
|
||||
stanowiąc fasadę dla reszty aplikacji.
|
||||
"""
|
||||
class WorkerSignals(QObject):
|
||||
finished = Signal(list)
|
||||
error = Signal(str)
|
||||
frame_ready = Signal(QPixmap)
|
||||
error_occurred = Signal(str)
|
||||
cameras_detected = Signal(list)
|
||||
camera_started = Signal()
|
||||
camera_stopped = Signal()
|
||||
|
||||
def __init__(self):
|
||||
super().__init__()
|
||||
self.signals = self.WorkerSignals()
|
||||
def __init__(self, parent: QObject | None = None) -> None:
|
||||
super().__init__(parent)
|
||||
self._camera_controller = CameraController()
|
||||
self._detected_cameras: list[dict] = []
|
||||
self._active_camera: BaseCamera | None = None
|
||||
self._active_camera_info: dict | None = None
|
||||
|
||||
def run(self) -> None:
|
||||
"""The main work of the worker."""
|
||||
detected_cameras = []
|
||||
# Przekazywanie sygnałów z kontrolera kamery na zewnątrz
|
||||
self._camera_controller.frame_ready.connect(self.frame_ready)
|
||||
self._camera_controller.error_occurred.connect(self.error_occurred)
|
||||
self._camera_controller.camera_ready.connect(self.start_liveview)
|
||||
|
||||
def detect_cameras(self) -> None:
|
||||
"""Wykrywa wszystkie dostępne kamery (GPhoto i OpenCV)."""
|
||||
self._detected_cameras.clear()
|
||||
|
||||
# Wykryj kamery GPhoto
|
||||
try:
|
||||
gphoto_cameras = GPhotoCamera.detect()
|
||||
for index, info in gphoto_cameras.items():
|
||||
detected_cameras.append({
|
||||
self._detected_cameras.append({
|
||||
"id": f"gphoto_{index}",
|
||||
"name": f"{info['name']} ({info['port']})",
|
||||
"type": "gphoto",
|
||||
"index": index
|
||||
})
|
||||
except Exception as e:
|
||||
self.signals.error.emit(f"Błąd podczas wykrywania kamer GPhoto: {e}")
|
||||
self.error_occurred.emit(
|
||||
f"Błąd podczas wykrywania kamer GPhoto: {e}")
|
||||
|
||||
# Wykryj kamery OpenCV
|
||||
try:
|
||||
opencv_cameras = OpenCvCamera.detect()
|
||||
for index, info in opencv_cameras.items():
|
||||
detected_cameras.append({
|
||||
self._detected_cameras.append({
|
||||
"id": f"opencv_{index}",
|
||||
"name": f"OpenCV: {info['name']}",
|
||||
"type": "opencv",
|
||||
"index": index
|
||||
})
|
||||
except Exception as e:
|
||||
self.signals.error.emit(f"Błąd podczas wykrywania kamer OpenCV: {e}")
|
||||
self.error_occurred.emit(
|
||||
f"Błąd podczas wykrywania kamer OpenCV: {e}")
|
||||
|
||||
self.signals.finished.emit(detected_cameras)
|
||||
|
||||
|
||||
class CameraManager(QObject):
|
||||
"""
|
||||
Zarządza wszystkimi operacjami związanymi z kamerami,
|
||||
stanowiąc fasadę dla reszty aplikacji.
|
||||
"""
|
||||
# --- Public API Signals ---
|
||||
frame_ready = Signal(QPixmap)
|
||||
error_occurred = Signal(str)
|
||||
detection_started = Signal()
|
||||
cameras_detected = Signal(list)
|
||||
camera_started = Signal()
|
||||
camera_stopped = Signal()
|
||||
|
||||
# --- Internal signals to communicate with worker thread ---
|
||||
_request_set_camera = Signal(BaseCamera, int)
|
||||
_request_start_camera = Signal()
|
||||
_request_stop_camera = Signal()
|
||||
_request_start_stream = Signal()
|
||||
_request_stop_stream = Signal()
|
||||
|
||||
def __init__(self, parent: QObject | None = None) -> None:
|
||||
super().__init__(parent)
|
||||
|
||||
self._camera_thread = QThread()
|
||||
self._camera_controller = CameraController()
|
||||
self._camera_controller.moveToThread(self._camera_thread)
|
||||
|
||||
# --- Connections ---
|
||||
# Connect signals from controller to be re-emitted by manager
|
||||
self._camera_controller.frame_ready.connect(self.frame_ready)
|
||||
self._camera_controller.error_occurred.connect(self.error_occurred)
|
||||
|
||||
# Connect internal requests to controller slots
|
||||
self._request_set_camera.connect(self._camera_controller.set_camera)
|
||||
self._request_start_camera.connect(self._camera_controller.start_camera)
|
||||
self._request_stop_camera.connect(self._camera_controller.stop_camera)
|
||||
self._request_start_stream.connect(self._camera_controller.start_stream)
|
||||
self._request_stop_stream.connect(self._camera_controller.stop_stream)
|
||||
|
||||
# Connect thread management
|
||||
self._camera_thread.started.connect(self._camera_controller.run)
|
||||
|
||||
self._camera_thread.start()
|
||||
|
||||
self._detected_cameras: list[dict] = []
|
||||
self._active_camera_info: dict | None = None
|
||||
self.thread_pool = QThreadPool.globalInstance() # For detection worker
|
||||
|
||||
def detect_cameras(self) -> None:
|
||||
self.detection_started.emit()
|
||||
worker = CameraDetectionWorker()
|
||||
worker.signals.finished.connect(self._on_detection_finished)
|
||||
worker.signals.error.connect(self.error_occurred)
|
||||
self.thread_pool.start(worker)
|
||||
|
||||
def _on_detection_finished(self, detected_cameras: list):
|
||||
self._detected_cameras = detected_cameras
|
||||
self.cameras_detected.emit(self._detected_cameras)
|
||||
|
||||
def get_detected_cameras(self) -> list[dict]:
|
||||
return self._detected_cameras
|
||||
|
||||
def start_camera(self, camera_id: str, fps: int = 15) -> None:
|
||||
if self._active_camera_info and self._active_camera_info['id'] == camera_id:
|
||||
return
|
||||
|
||||
if self._active_camera_info:
|
||||
"""Uruchamia wybraną kamerę."""
|
||||
if self._active_camera:
|
||||
self.stop_camera()
|
||||
|
||||
camera_info = next((c for c in self._detected_cameras if c['id'] == camera_id), None)
|
||||
camera_info = next(
|
||||
(c for c in self._detected_cameras if c['id'] == camera_id), None)
|
||||
|
||||
if not camera_info:
|
||||
self.error_occurred.emit(f"Nie znaleziono kamery o ID: {camera_id}")
|
||||
self.error_occurred.emit(
|
||||
f"Nie znaleziono kamery o ID: {camera_id}")
|
||||
return
|
||||
|
||||
camera_type = camera_info['type']
|
||||
camera_index = camera_info['index']
|
||||
|
||||
camera_instance: BaseCamera | None = None
|
||||
if camera_type == "gphoto":
|
||||
camera_instance = GPhotoCamera()
|
||||
self._active_camera = GPhotoCamera()
|
||||
elif camera_type == "opencv":
|
||||
camera_instance = OpenCvCamera()
|
||||
self._active_camera = OpenCvCamera()
|
||||
else:
|
||||
self.error_occurred.emit(f"Nieznany typ kamery: {camera_type}")
|
||||
return
|
||||
|
||||
self._active_camera_info = camera_info
|
||||
|
||||
# Emit signals to trigger slots in the worker thread
|
||||
self._request_set_camera.emit(camera_instance, fps)
|
||||
self._request_start_camera.emit()
|
||||
self._request_start_stream.emit()
|
||||
self._camera_controller.set_camera(self._active_camera, fps)
|
||||
self._camera_controller.start_camera()
|
||||
|
||||
def start_liveview(self, connected):
|
||||
if connected:
|
||||
self._camera_controller.start_stream()
|
||||
self.camera_started.emit()
|
||||
else:
|
||||
self._active_camera = None
|
||||
self._active_camera_info = None
|
||||
|
||||
def stop_camera(self) -> None:
|
||||
if self._active_camera_info:
|
||||
# Emit signals to trigger slots in the worker thread
|
||||
self._request_stop_stream.emit()
|
||||
self._request_stop_camera.emit()
|
||||
"""Zatrzymuje aktywną kamerę."""
|
||||
if self._active_camera:
|
||||
self._camera_controller.stop_camera()
|
||||
self._active_camera = None
|
||||
self._active_camera_info = None
|
||||
self.camera_stopped.emit()
|
||||
|
||||
@@ -157,9 +116,6 @@ class CameraManager(QObject):
|
||||
return self._active_camera_info
|
||||
|
||||
def shutdown(self) -> None:
|
||||
if self._camera_thread.isRunning():
|
||||
self._camera_thread.quit()
|
||||
if not self._camera_thread.wait(5000):
|
||||
print("Camera thread did not finish gracefully, terminating.")
|
||||
self._camera_thread.terminate()
|
||||
self._camera_thread.wait()
|
||||
"""Zamyka kontroler kamery i jego wątek."""
|
||||
self.stop_camera()
|
||||
self._camera_controller.stop()
|
||||
|
||||
Reference in New Issue
Block a user