优化ucd调用结构
This commit is contained in:
@@ -1 +1,3 @@
|
||||
from app.services.pattern_service import PatternService, PatternSession
|
||||
from app.ucd import PatternService, PatternSession
|
||||
|
||||
__all__ = ["PatternService", "PatternSession"]
|
||||
|
||||
@@ -1,216 +0,0 @@
|
||||
from __future__ import annotations
|
||||
|
||||
import copy
|
||||
from dataclasses import dataclass
|
||||
|
||||
from app.data_range_converter import convert_pattern_params
|
||||
from app.pq.pq_config import get_pattern
|
||||
|
||||
|
||||
@dataclass
|
||||
class PatternSession:
|
||||
mode: str
|
||||
test_type: str
|
||||
active_config: object
|
||||
pattern_params: list[list[int]]
|
||||
total_patterns: int
|
||||
display_names: list[str]
|
||||
|
||||
|
||||
class PatternService:
|
||||
def __init__(self, app):
|
||||
self.app = app
|
||||
|
||||
def _build_apply_config_error(self, test_type):
|
||||
timing = self.app.config.current_test_types.get(test_type, {}).get("timing", "-")
|
||||
detail = ""
|
||||
err = self.app.signal_service.last_error
|
||||
if err:
|
||||
detail = f", detail={err}"
|
||||
return f"UCD profile apply_config failed for {test_type}, timing={timing}{detail}"
|
||||
|
||||
def prepare_session(self, mode, *, test_type=None, log_details=False):
|
||||
test_type = test_type or self.app.config.current_test_type
|
||||
if hasattr(self.app.config, "set_current_test_type"):
|
||||
self.app.config.set_current_test_type(test_type)
|
||||
if not self.app.config.set_current_pattern(mode):
|
||||
raise ValueError(f"未知的图案模式: {mode}")
|
||||
|
||||
active_config = self.app.config
|
||||
source_params = self._get_source_pattern_params(mode)
|
||||
|
||||
if test_type == "screen_module":
|
||||
screen_cfg = self.app.config.current_test_types.get("screen_module", {})
|
||||
color_space = (
|
||||
self.app.screen_module_color_space_var.get()
|
||||
if hasattr(self.app, "screen_module_color_space_var")
|
||||
else screen_cfg.get("colorimetry", "sRGB")
|
||||
)
|
||||
data_range = (
|
||||
self.app.screen_module_data_range_var.get()
|
||||
if hasattr(self.app, "screen_module_data_range_var")
|
||||
else screen_cfg.get("data_range", "Full")
|
||||
)
|
||||
bit_depth = (
|
||||
self.app.screen_module_bit_depth_var.get()
|
||||
if hasattr(self.app, "screen_module_bit_depth_var")
|
||||
else f"{int(screen_cfg.get('bpc', 8))}bit"
|
||||
)
|
||||
output_format = (
|
||||
self.app.screen_module_output_format_var.get()
|
||||
if hasattr(self.app, "screen_module_output_format_var")
|
||||
else screen_cfg.get("color_format", "RGB")
|
||||
)
|
||||
|
||||
if log_details:
|
||||
self._log("=" * 50, "separator")
|
||||
self._log("设置屏模组信号格式:", "info")
|
||||
self._log("=" * 50, "separator")
|
||||
for label, value in [
|
||||
("色彩空间", color_space),
|
||||
("色彩格式", output_format),
|
||||
("数据范围", data_range),
|
||||
("编码位深", bit_depth),
|
||||
("Timing", self.app.config.current_test_types[test_type]["timing"]),
|
||||
]:
|
||||
self._log(f" {label}: {value}", "info")
|
||||
if not self.app.signal_service.apply_config(active_config):
|
||||
raise RuntimeError(self._build_apply_config_error(test_type))
|
||||
success = self.app.signal_service.update_signal_format(
|
||||
color_space=color_space,
|
||||
data_range=data_range,
|
||||
bit_depth=bit_depth,
|
||||
output_format=output_format,
|
||||
)
|
||||
if log_details:
|
||||
self._log(
|
||||
f"屏模组信号格式设置{'成功' if success else '失败'}",
|
||||
"success" if success else "error",
|
||||
)
|
||||
|
||||
elif test_type == "sdr_movie":
|
||||
data_range = self.app.sdr_data_range_var.get()
|
||||
if log_details:
|
||||
self._log("=" * 50, "separator")
|
||||
self._log("设置 SDR 信号格式:", "info")
|
||||
self._log("=" * 50, "separator")
|
||||
for label, value in [
|
||||
("色彩空间", self.app.sdr_color_space_var.get()),
|
||||
("色彩格式", self.app.sdr_output_format_var.get()),
|
||||
("Gamma", self.app.sdr_gamma_type_var.get()),
|
||||
("数据范围", data_range),
|
||||
("编码位深", self.app.sdr_bit_depth_var.get()),
|
||||
]:
|
||||
self._log(f" {label}: {value}", "info")
|
||||
converted_params = convert_pattern_params(
|
||||
source_params, data_range=data_range, verbose=False
|
||||
)
|
||||
active_config = self.app.config.get_temp_config_with_converted_params(
|
||||
mode=mode, converted_params=converted_params
|
||||
)
|
||||
if hasattr(active_config, "set_current_test_type"):
|
||||
active_config.set_current_test_type(test_type)
|
||||
if not self.app.signal_service.apply_config(active_config):
|
||||
raise RuntimeError(self._build_apply_config_error(test_type))
|
||||
success = self.app.signal_service.update_signal_format(
|
||||
color_space=self.app.sdr_color_space_var.get(),
|
||||
data_range=data_range,
|
||||
bit_depth=self.app.sdr_bit_depth_var.get(),
|
||||
output_format=self.app.sdr_output_format_var.get(),
|
||||
)
|
||||
if log_details:
|
||||
self._log(f"SDR 信号格式设置{'成功' if success else '失败'}", "success" if success else "error")
|
||||
self._log(f"图案参数已设置,共 {len(converted_params)} 个图案", "success")
|
||||
|
||||
elif test_type == "hdr_movie":
|
||||
data_range = self.app.hdr_data_range_var.get()
|
||||
if log_details:
|
||||
self._log("=" * 50, "separator")
|
||||
self._log("设置 HDR 信号格式:", "info")
|
||||
self._log("=" * 50, "separator")
|
||||
for label, value in [
|
||||
("色彩空间", self.app.hdr_color_space_var.get()),
|
||||
("色彩格式", self.app.hdr_output_format_var.get()),
|
||||
("数据范围", data_range),
|
||||
("编码位深", self.app.hdr_bit_depth_var.get()),
|
||||
("MaxCLL", self.app.hdr_maxcll_var.get()),
|
||||
("MaxFALL", self.app.hdr_maxfall_var.get()),
|
||||
]:
|
||||
self._log(f" {label}: {value}", "info")
|
||||
converted_params = convert_pattern_params(
|
||||
source_params, data_range=data_range, verbose=False
|
||||
)
|
||||
active_config = self.app.config.get_temp_config_with_converted_params(
|
||||
mode=mode, converted_params=converted_params
|
||||
)
|
||||
if hasattr(active_config, "set_current_test_type"):
|
||||
active_config.set_current_test_type(test_type)
|
||||
if not self.app.signal_service.apply_config(active_config):
|
||||
raise RuntimeError(self._build_apply_config_error(test_type))
|
||||
success = self.app.signal_service.update_signal_format(
|
||||
color_space=self.app.hdr_color_space_var.get(),
|
||||
data_range=data_range,
|
||||
bit_depth=self.app.hdr_bit_depth_var.get(),
|
||||
output_format=self.app.hdr_output_format_var.get(),
|
||||
max_cll=self.app.hdr_maxcll_var.get(),
|
||||
max_fall=self.app.hdr_maxfall_var.get(),
|
||||
)
|
||||
if log_details:
|
||||
self._log(f"HDR 信号格式设置{'成功' if success else '失败'}", "success" if success else "error")
|
||||
self._log(f"图案参数已设置,共 {len(converted_params)} 个图案", "success")
|
||||
|
||||
else:
|
||||
raise ValueError(f"不支持的测试类型: {test_type}")
|
||||
|
||||
pattern_params = copy.deepcopy(active_config.current_pattern["pattern_params"])
|
||||
return PatternSession(
|
||||
mode=mode,
|
||||
test_type=test_type,
|
||||
active_config=active_config,
|
||||
pattern_params=pattern_params,
|
||||
total_patterns=len(pattern_params),
|
||||
display_names=self._get_display_names(mode, len(pattern_params)),
|
||||
)
|
||||
|
||||
def send_session_pattern(self, session, index):
|
||||
if index < 0 or index >= session.total_patterns:
|
||||
raise IndexError(f"pattern 索引越界: {index}")
|
||||
|
||||
pattern_param = session.pattern_params[index]
|
||||
if not self.app.signal_service.send_pattern_params(pattern_param):
|
||||
raise RuntimeError(f"发送 pattern 失败: {index}")
|
||||
return pattern_param
|
||||
|
||||
def send_rgb(self, rgb, *, session=None, test_type=None):
|
||||
active_session = session or self.prepare_session(
|
||||
"rgb",
|
||||
test_type=test_type,
|
||||
log_details=False,
|
||||
)
|
||||
converted_rgb = self._convert_rgb_for_test_type(rgb, active_session.test_type)
|
||||
self.app.signal_service.send_solid_rgb(converted_rgb)
|
||||
return True
|
||||
|
||||
def _get_source_pattern_params(self, mode):
|
||||
return copy.deepcopy(get_pattern(mode)["pattern_params"])
|
||||
|
||||
def _get_display_names(self, mode, total_patterns):
|
||||
if mode == "accuracy":
|
||||
return self.app.config.get_accuracy_color_names()
|
||||
if mode == "custom" and hasattr(self.app.config, "get_temp_pattern_names"):
|
||||
return self.app.config.get_temp_pattern_names()
|
||||
return [f"P {index + 1}" for index in range(total_patterns)]
|
||||
|
||||
def _convert_rgb_for_test_type(self, rgb, test_type):
|
||||
if test_type == "sdr_movie":
|
||||
data_range = self.app.sdr_data_range_var.get()
|
||||
elif test_type == "hdr_movie":
|
||||
data_range = self.app.hdr_data_range_var.get()
|
||||
else:
|
||||
data_range = "Full"
|
||||
|
||||
return convert_pattern_params([list(rgb)], data_range=data_range, verbose=False)[0]
|
||||
|
||||
def _log(self, message, level):
|
||||
if hasattr(self.app, "log_gui"):
|
||||
self.app.log_gui.log(message, level=level)
|
||||
@@ -1,219 +0,0 @@
|
||||
"""UCD 信号 / 图案应用服务层。
|
||||
|
||||
服务层是 GUI ↔ Driver 的唯一通道,负责:
|
||||
- 将 UI 字符串("BT.709"、"10bit"、"YCbCr 4:4:4" 等)翻译成 :class:`SignalFormat`;
|
||||
- 将各 panel 的 timing 字符串翻译成 :class:`TimingSpec`;
|
||||
- 协调 :meth:`IUcdDevice.configure` / ``set_pattern`` / ``apply`` 的调用顺序;
|
||||
- 通过 :class:`EventBus` 让 GUI 订阅状态变化,而非主动轮询。
|
||||
|
||||
本层不直接 import UniTAP,也不读取 :mod:`tkinter` 变量;
|
||||
所有输入都是显式参数,便于单测。
|
||||
|
||||
线程安全由 :class:`UCD323Device` 的设备锁统一保证,本层不再重复加锁。
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
import logging
|
||||
|
||||
from app.ucd_domain import (
|
||||
Colorimetry,
|
||||
DynamicRange,
|
||||
EventBus,
|
||||
PatternKind,
|
||||
PatternSpec,
|
||||
SignalFormat,
|
||||
TimingSpec,
|
||||
UcdError,
|
||||
UcdState,
|
||||
bit_depth_str_to_bpc,
|
||||
color_space_to_colorimetry,
|
||||
data_range_to_dynamic_range,
|
||||
output_format_to_color_format,
|
||||
parse_timing_str,
|
||||
)
|
||||
from drivers.ucd_driver import IUcdDevice
|
||||
|
||||
log = logging.getLogger(__name__)
|
||||
|
||||
|
||||
# ─── 视图字符串 → 值对象 转换工具 ────────────────────────────────
|
||||
|
||||
|
||||
def build_signal_format(
|
||||
*,
|
||||
color_space: str,
|
||||
output_format: str,
|
||||
bit_depth: str,
|
||||
data_range: str = "Full",
|
||||
) -> SignalFormat:
|
||||
"""根据下拉框字符串组装 :class:`SignalFormat`。
|
||||
|
||||
各参数解析失败抛 :class:`UcdConfigError`。
|
||||
"""
|
||||
return SignalFormat(
|
||||
color_format=output_format_to_color_format(output_format),
|
||||
colorimetry=color_space_to_colorimetry(color_space),
|
||||
bpc=bit_depth_str_to_bpc(bit_depth),
|
||||
dynamic_range=data_range_to_dynamic_range(data_range),
|
||||
)
|
||||
|
||||
|
||||
def build_signal_format_from_profile(
|
||||
*,
|
||||
color_space: str,
|
||||
color_format: str,
|
||||
bpc: int,
|
||||
data_range: str = "Full",
|
||||
) -> SignalFormat:
|
||||
"""从 PQConfig test_type 条目组装 :class:`SignalFormat`。"""
|
||||
bit_depth = f"{int(bpc)}bit"
|
||||
return build_signal_format(
|
||||
color_space=color_space,
|
||||
output_format=color_format,
|
||||
bit_depth=bit_depth,
|
||||
data_range=data_range,
|
||||
)
|
||||
|
||||
|
||||
def build_timing(timing_str: str) -> TimingSpec:
|
||||
"""``"DMT 3840x2160@60Hz"`` → :class:`TimingSpec`。"""
|
||||
return parse_timing_str(timing_str)
|
||||
|
||||
|
||||
def solid_rgb_pattern(rgb: tuple[int, int, int] | list[int]) -> PatternSpec:
|
||||
r, g, b = rgb[0], rgb[1], rgb[2]
|
||||
return PatternSpec(kind=PatternKind.SOLID, solid_rgb=(int(r), int(g), int(b)))
|
||||
|
||||
|
||||
def image_pattern(path: str) -> PatternSpec:
|
||||
return PatternSpec(kind=PatternKind.IMAGE, image_path=path)
|
||||
|
||||
|
||||
# ─── 服务 ────────────────────────────────────────────────────────
|
||||
|
||||
|
||||
class SignalService:
|
||||
"""协调 SignalFormat / Timing / Pattern 的写入与提交。"""
|
||||
|
||||
def __init__(self, device: IUcdDevice, bus: EventBus):
|
||||
self._dev = device
|
||||
self._bus = bus
|
||||
|
||||
# -- 高层接口 ------------------------------------------------
|
||||
|
||||
def apply(
|
||||
self,
|
||||
*,
|
||||
signal: SignalFormat,
|
||||
timing: TimingSpec,
|
||||
pattern: PatternSpec,
|
||||
) -> bool:
|
||||
"""一次性提交信号格式 + timing + 图案。
|
||||
|
||||
Returns:
|
||||
``format_changed``——本次相对上一次 :meth:`apply` 是否变化。
|
||||
"""
|
||||
log.info(
|
||||
"SignalService.apply signal=%s timing=%s pattern=%s",
|
||||
signal,
|
||||
timing,
|
||||
pattern.kind.value,
|
||||
)
|
||||
changed = self._dev.configure(signal, timing)
|
||||
self._dev.set_pattern(pattern)
|
||||
self._dev.apply()
|
||||
return changed
|
||||
|
||||
def send_pattern(self, pattern: PatternSpec) -> None:
|
||||
"""在已 configure 的信号上仅更新图案后 apply。"""
|
||||
log.info("SignalService.send_pattern pattern=%s", pattern.kind.value)
|
||||
self._dev.set_pattern(pattern)
|
||||
self._dev.apply()
|
||||
|
||||
def send_solid_rgb(self, rgb: tuple[int, int, int] | list[int]) -> None:
|
||||
self.send_pattern(solid_rgb_pattern(rgb))
|
||||
|
||||
def send_image(self, path: str) -> None:
|
||||
self.send_pattern(image_pattern(path))
|
||||
|
||||
def update_signal_format(
|
||||
self,
|
||||
*,
|
||||
color_space: str,
|
||||
output_format: str,
|
||||
bit_depth: str,
|
||||
data_range: str = "Full",
|
||||
max_cll: int | None = None,
|
||||
max_fall: int | None = None,
|
||||
) -> bool:
|
||||
"""仅将信号格式提交到 SDK(沿用上一次的 timing),不切换图案。
|
||||
|
||||
UI 字符串先经域层解析做参数校验;解析失败抛 :class:`UcdConfigError`。
|
||||
"""
|
||||
_ = build_signal_format(
|
||||
color_space=color_space,
|
||||
output_format=output_format,
|
||||
bit_depth=bit_depth,
|
||||
data_range=data_range,
|
||||
)
|
||||
return self._dev.apply_signal_format(
|
||||
color_space=color_space,
|
||||
color_format=output_format,
|
||||
bit_depth=bit_depth,
|
||||
data_range=data_range,
|
||||
max_cll=max_cll,
|
||||
max_fall=max_fall,
|
||||
)
|
||||
|
||||
# -- 透传给上层的查询 ---------------------------------------
|
||||
|
||||
@property
|
||||
def device(self) -> IUcdDevice:
|
||||
return self._dev
|
||||
|
||||
def current_resolution(self) -> tuple[int, int]:
|
||||
return self._dev.current_resolution()
|
||||
|
||||
@property
|
||||
def is_connected(self) -> bool:
|
||||
"""UCD 设备是否已打开。供 GUI 做前置校验。"""
|
||||
return self._dev.state != UcdState.CLOSED
|
||||
|
||||
@property
|
||||
def format_changed(self) -> bool:
|
||||
"""最近一次视频模式提交是否相对上次发生变化。"""
|
||||
return self._dev.format_changed
|
||||
|
||||
@property
|
||||
def last_error(self) -> str | None:
|
||||
return self._dev.last_error
|
||||
|
||||
def apply_config(self, config) -> bool:
|
||||
"""按 :class:`PQConfig` 写入色彩 / Timing / 当前 Pattern(不 apply 输出)。"""
|
||||
return bool(self._dev.set_ucd_params(config))
|
||||
|
||||
def send_pattern_params(self, params) -> bool:
|
||||
"""以 ``params`` 更新当前 pattern 的参数并 apply。"""
|
||||
return bool(self._dev.send_current_pattern_params(params))
|
||||
|
||||
def apply_and_run(self, config, pattern_params) -> bool:
|
||||
"""``set_ucd_params`` + ``set_pattern`` + ``run`` 的复合操作。"""
|
||||
return bool(self._dev.apply_config_and_run(config, pattern_params))
|
||||
|
||||
|
||||
__all__ = [
|
||||
"SignalService",
|
||||
"build_signal_format",
|
||||
"build_signal_format_from_profile",
|
||||
"build_timing",
|
||||
"solid_rgb_pattern",
|
||||
"image_pattern",
|
||||
"SignalFormat",
|
||||
"TimingSpec",
|
||||
"PatternSpec",
|
||||
"PatternKind",
|
||||
"Colorimetry",
|
||||
"DynamicRange",
|
||||
"UcdError",
|
||||
]
|
||||
Reference in New Issue
Block a user