691 lines
25 KiB
Python
691 lines
25 KiB
Python
import sys
|
||
import os
|
||
from dataclasses import dataclass
|
||
from typing import Optional, Tuple
|
||
import numpy as np
|
||
|
||
from PyQt5 import QtCore, QtGui, QtWidgets
|
||
|
||
|
||
# -----------------------------
|
||
# 数据模型与枚举
|
||
# -----------------------------
|
||
|
||
class PixelFormat:
|
||
# Packed YUV422
|
||
YUYV = "YUYV" # Y0 U0 Y1 V0
|
||
YVYU = "YVYU" # Y0 V0 Y1 U0
|
||
UYVY = "UYVY" # U0 Y0 V0 Y1
|
||
VYUY = "VYUY" # V0 Y0 U0 Y1
|
||
|
||
# Semi-planar / planar variants
|
||
NV12 = "NV12" # Y + interleaved UV (4:2:0), UV order = U,V
|
||
NV21 = "NV21" # Y + interleaved VU (4:2:0), UV order = V,U
|
||
NV16 = "NV16" # Y + interleaved UV (4:2:2), UV order = U,V
|
||
NV61 = "NV61" # Y + interleaved VU (4:2:2), UV order = V,U
|
||
|
||
GRAY = "GRAY" # Only Y
|
||
|
||
# YUV444
|
||
YUV444_PLANAR = "YUV444_PLANAR" # YYYY... UUUU... VVVV...
|
||
YUV444_INTERLEAVED_UV = "YUV444_INTERLEAVED_UV" # Y U V Y U V ... (UV顺序=U,V)
|
||
YUV444_INTERLEAVED_VU = "YUV444_INTERLEAVED_VU" # Y V U Y V U ... (UV顺序=V,U)
|
||
|
||
# RGB/BGR formats
|
||
RGB565 = "RGB565" # 16-bit RGB: RRRRR GGGGGG BBBBB
|
||
BGR565 = "BGR565" # 16-bit BGR: BBBBB GGGGGG RRRRR
|
||
RGB888 = "RGB888" # 24-bit RGB: R G B
|
||
BGR888 = "BGR888" # 24-bit BGR: B G R
|
||
RGB8888 = "RGB8888" # 32-bit RGB: R G B (alpha ignored)
|
||
BGR8888 = "BGR8888" # 32-bit BGR: B G R (alpha ignored)
|
||
NRGB8888 = "NRGB8888" # 32-bit native RGB (no alpha)
|
||
NBGR8888 = "NBGR8888" # 32-bit native BGR (no alpha)
|
||
|
||
|
||
SUPPORTED_FORMATS = [
|
||
PixelFormat.YUYV, PixelFormat.YVYU, PixelFormat.UYVY, PixelFormat.VYUY,
|
||
PixelFormat.NV12, PixelFormat.NV21, PixelFormat.NV16, PixelFormat.NV61,
|
||
PixelFormat.GRAY,
|
||
PixelFormat.YUV444_PLANAR, PixelFormat.YUV444_INTERLEAVED_UV, PixelFormat.YUV444_INTERLEAVED_VU,
|
||
PixelFormat.RGB565, PixelFormat.BGR565,
|
||
PixelFormat.RGB888, PixelFormat.BGR888,
|
||
PixelFormat.RGB8888, PixelFormat.BGR8888,
|
||
PixelFormat.NRGB8888, PixelFormat.NBGR8888
|
||
]
|
||
|
||
# 预定义的尺寸选项
|
||
PREDEFINED_SIZES = [
|
||
"640x480 (VGA)",
|
||
"800x600 (SVGA)",
|
||
"1024x768 (XGA)",
|
||
"1280x720 (HD)",
|
||
"1920x1080 (Full HD)",
|
||
"2560x1440 (2K)",
|
||
"3840x2160 (4K)",
|
||
"自定义..."
|
||
]
|
||
|
||
|
||
@dataclass
|
||
class ImageSpec:
|
||
width: int
|
||
height: int
|
||
fmt: str
|
||
|
||
|
||
# -----------------------------
|
||
# YUV -> RGB 转换工具
|
||
# -----------------------------
|
||
|
||
def yuv_to_rgb_numpy(Y, U, V):
|
||
"""
|
||
Y, U, V: float32 ndarray in [0..255], same shape.
|
||
Returns RGB uint8 ndarray with shape (..., 3).
|
||
BT.601 full-range近似:
|
||
R = Y + 1.402*(V-128)
|
||
G = Y - 0.344136*(U-128) - 0.714136*(V-128)
|
||
B = Y + 1.772*(U-128)
|
||
"""
|
||
Yf = Y.astype(np.float32)
|
||
Uf = U.astype(np.float32) - 128.0
|
||
Vf = V.astype(np.float32) - 128.0
|
||
|
||
R = Yf + 1.402 * Vf
|
||
G = Yf - 0.344136 * Uf - 0.714136 * Vf
|
||
B = Yf + 1.772 * Uf
|
||
|
||
RGB = np.stack([R, G, B], axis=-1)
|
||
np.clip(RGB, 0, 255, out=RGB)
|
||
return RGB.astype(np.uint8)
|
||
|
||
|
||
def unpack_packed_422_to_planes(data: np.ndarray, width: int, height: int, order: str) -> Tuple[np.ndarray, np.ndarray, np.ndarray]:
|
||
"""
|
||
将打包的YUV422(每2像素4字节)解成等大Y平面和下采样的U/V再上采样到全分辨率。
|
||
order: "YUYV", "YVYU", "UYVY", "VYUY"
|
||
返回 Y, U_full, V_full
|
||
"""
|
||
# 每行应有 width*2 字节
|
||
expected = width * height * 2
|
||
if data.size != expected:
|
||
raise ValueError(f"数据长度不匹配,期望 {expected} 字节,实际 {data.size} 字节")
|
||
|
||
# 先 reshape 成 (H, W*2)
|
||
row = data.reshape((height, width * 2))
|
||
# 两像素为一组的4字节块
|
||
blocks = row.reshape((height, width // 2, 4)) # shape (H, W/2, 4)
|
||
|
||
if order == PixelFormat.YUYV:
|
||
Y0 = blocks[:, :, 0]
|
||
U0 = blocks[:, :, 1]
|
||
Y1 = blocks[:, :, 2]
|
||
V0 = blocks[:, :, 3]
|
||
elif order == PixelFormat.YVYU:
|
||
Y0 = blocks[:, :, 0]
|
||
V0 = blocks[:, :, 1]
|
||
Y1 = blocks[:, :, 2]
|
||
U0 = blocks[:, :, 3]
|
||
elif order == PixelFormat.UYVY:
|
||
U0 = blocks[:, :, 0]
|
||
Y0 = blocks[:, :, 1]
|
||
V0 = blocks[:, :, 2]
|
||
Y1 = blocks[:, :, 3]
|
||
elif order == PixelFormat.VYUY:
|
||
V0 = blocks[:, :, 0]
|
||
Y0 = blocks[:, :, 1]
|
||
U0 = blocks[:, :, 2]
|
||
Y1 = blocks[:, :, 3]
|
||
else:
|
||
raise ValueError("不支持的打包422顺序")
|
||
|
||
# 还原Y为 (H, W)
|
||
Y = np.empty((height, width), dtype=np.uint8)
|
||
Y[:, 0::2] = Y0
|
||
Y[:, 1::2] = Y1
|
||
|
||
# U/V 为每2像素共享,横向采样因子2,纵向不变 -> 需要水平上采样到全分辨率
|
||
U = np.repeat(U0, 2, axis=1)
|
||
V = np.repeat(V0, 2, axis=1)
|
||
return Y, U, V
|
||
|
||
|
||
def parse_nvxx(data: np.ndarray, width: int, height: int, chroma_order: str, is_420: bool) -> Tuple[np.ndarray, np.ndarray, np.ndarray]:
|
||
"""
|
||
NV12/NV21 (420) 或 NV16/NV61 (422) 半平面解析。
|
||
chroma_order: "UV" or "VU"
|
||
is_420: True for 4:2:0, False for 4:2:2
|
||
返回 Y, U_full, V_full
|
||
"""
|
||
if is_420:
|
||
y_size = width * height
|
||
c_h = height // 2
|
||
c_size = width * c_h # interleaved 2 channels across width
|
||
expected = y_size + c_size
|
||
if data.size != expected:
|
||
raise ValueError(f"数据长度不匹配(420),期望 {expected} 字节,实际 {data.size} 字节")
|
||
Y = data[:y_size].reshape((height, width))
|
||
UV = data[y_size:].reshape((c_h, width))
|
||
# UV 行内是交错: U,V,U,V,...
|
||
U_sub = UV[:, 0::2]
|
||
V_sub = UV[:, 1::2]
|
||
if chroma_order == "VU":
|
||
U_sub, V_sub = V_sub, U_sub
|
||
# 上采样到全分辨率(双线性可选,这里用最近邻)
|
||
U = np.repeat(np.repeat(U_sub, 2, axis=0), 2, axis=1)
|
||
V = np.repeat(np.repeat(V_sub, 2, axis=0), 2, axis=1)
|
||
else:
|
||
# 422: 色度高方向与亮度同高,水平方向为1/2
|
||
y_size = width * height
|
||
c_h = height
|
||
c_size = width * c_h // 1 // 1 // 2 # 每行交错UV对,宽度一样但每2像素一组 → 列数等于宽
|
||
expected = y_size + (width * c_h) # 因为UV交错,每行宽度与Y一致
|
||
if data.size != expected:
|
||
raise ValueError(f"数据长度不匹配(422),期望 {expected} 字节,实际 {data.size} 字节")
|
||
Y = data[:y_size].reshape((height, width))
|
||
UV = data[y_size:].reshape((c_h, width))
|
||
U_sub = UV[:, 0::2]
|
||
V_sub = UV[:, 1::2]
|
||
if chroma_order == "VU":
|
||
U_sub, V_sub = V_sub, U_sub
|
||
U = np.repeat(U_sub, 2, axis=1)
|
||
V = np.repeat(V_sub, 2, axis=1)
|
||
return Y, U, V
|
||
|
||
|
||
def parse_gray(data: np.ndarray, width: int, height: int) -> Tuple[np.ndarray, np.ndarray, np.ndarray]:
|
||
expected = width * height
|
||
if data.size != expected:
|
||
raise ValueError(f"数据长度不匹配(GRAY),期望 {expected} 字节,实际 {data.size} 字节")
|
||
Y = data.reshape((height, width))
|
||
U = np.full_like(Y, 128, dtype=np.uint8)
|
||
V = np.full_like(Y, 128, dtype=np.uint8)
|
||
return Y, U, V
|
||
|
||
|
||
def parse_yuv444_planar(data: np.ndarray, width: int, height: int) -> Tuple[np.ndarray, np.ndarray, np.ndarray]:
|
||
plane_size = width * height
|
||
expected = plane_size * 3
|
||
if data.size != expected:
|
||
raise ValueError(f"数据长度不匹配(YUV444 planar),期望 {expected} 字节,实际 {data.size} 字节")
|
||
Y = data[0:plane_size].reshape((height, width))
|
||
U = data[plane_size:2*plane_size].reshape((height, width))
|
||
V = data[2*plane_size:3*plane_size].reshape((height, width))
|
||
return Y, U, V
|
||
|
||
|
||
def parse_yuv444_interleaved(data: np.ndarray, width: int, height: int, uv_order: str) -> Tuple[np.ndarray, np.ndarray, np.ndarray]:
|
||
"""
|
||
交错式:按像素 Y U V 或 Y V U 排列(总步长3字节)。
|
||
uv_order: "UV" 或 "VU"
|
||
"""
|
||
expected = width * height * 3
|
||
if data.size != expected:
|
||
raise ValueError(f"数据长度不匹配(YUV444 interleaved),期望 {expected} 字节,实际 {data.size} 字节")
|
||
pix = data.reshape((height, width, 3))
|
||
Y = pix[:, :, 0]
|
||
if uv_order == "UV":
|
||
U = pix[:, :, 1]
|
||
V = pix[:, :, 2]
|
||
else:
|
||
V = pix[:, :, 1]
|
||
U = pix[:, :, 2]
|
||
return Y, U, V
|
||
|
||
|
||
# -----------------------------
|
||
# RGB/BGR 格式解析函数
|
||
# -----------------------------
|
||
|
||
def parse_rgb565(data: np.ndarray, width: int, height: int, is_bgr: bool = False) -> np.ndarray:
|
||
"""
|
||
解析RGB565或BGR565格式
|
||
每个像素2字节:RRRRR GGGGGG BBBBB (16位)
|
||
返回RGB uint8数组 (H, W, 3)
|
||
"""
|
||
expected = width * height * 2
|
||
if data.size != expected:
|
||
raise ValueError(f"数据长度不匹配(RGB565/BGR565),期望 {expected} 字节,实际 {data.size} 字节")
|
||
|
||
# 将数据重新解释为16位整数(小端序)
|
||
pixels = data.view(dtype=np.uint16).reshape((height, width))
|
||
|
||
# 提取RGB分量
|
||
if is_bgr:
|
||
# BGR565: BBBBB GGGGGG RRRRR
|
||
b = ((pixels >> 11) & 0x1F) << 3 # 5-bit to 8-bit
|
||
g = ((pixels >> 5) & 0x3F) << 2 # 6-bit to 8-bit
|
||
r = (pixels & 0x1F) << 3 # 5-bit to 8-bit
|
||
else:
|
||
# RGB565: RRRRR GGGGGG BBBBB
|
||
r = ((pixels >> 11) & 0x1F) << 3
|
||
g = ((pixels >> 5) & 0x3F) << 2
|
||
b = (pixels & 0x1F) << 3
|
||
|
||
# 组合成RGB图像
|
||
rgb = np.stack([r, g, b], axis=-1).astype(np.uint8)
|
||
return rgb
|
||
|
||
|
||
def parse_rgb888(data: np.ndarray, width: int, height: int, is_bgr: bool = False) -> np.ndarray:
|
||
"""
|
||
解析RGB888或BGR888格式
|
||
每个像素3字节:R G B 或 B G R
|
||
返回RGB uint8数组 (H, W, 3)
|
||
"""
|
||
expected = width * height * 3
|
||
if data.size != expected:
|
||
raise ValueError(f"数据长度不匹配(RGB888/BGR888),期望 {expected} 字节,实际 {data.size} 字节")
|
||
|
||
# 重塑为(H, W, 3)
|
||
pixels = data.reshape((height, width, 3))
|
||
|
||
if is_bgr:
|
||
# BGR -> RGB
|
||
rgb = np.zeros_like(pixels)
|
||
rgb[:, :, 0] = pixels[:, :, 2] # R
|
||
rgb[:, :, 1] = pixels[:, :, 1] # G
|
||
rgb[:, :, 2] = pixels[:, :, 0] # B
|
||
return rgb
|
||
else:
|
||
# 已经是RGB格式
|
||
return pixels
|
||
|
||
|
||
def parse_rgb8888(data: np.ndarray, width: int, height: int, is_bgr: bool = False, is_native: bool = False) -> np.ndarray:
|
||
"""
|
||
解析32位RGB/BGR格式
|
||
每个像素4字节:R G B (alpha ignored) 或 B G R (alpha ignored)
|
||
或NRGB8888/NBGR8888(native格式,alpha通道可能位于不同位置)
|
||
返回RGB uint8数组 (H, W, 3)
|
||
"""
|
||
expected = width * height * 4
|
||
if data.size != expected:
|
||
raise ValueError(f"数据长度不匹配(32-bit RGB/BGR),期望 {expected} 字节,实际 {data.size} 字节")
|
||
|
||
# 重塑为(H, W, 4)
|
||
pixels = data.reshape((height, width, 4))
|
||
|
||
if is_native:
|
||
# 对于NRGB8888/NBGR8888,格式可能为 A R G B 或 R G B A,取决于系统字节序
|
||
# 简单起见,我们假设前三个字节是RGB/BGR分量(忽略alpha)
|
||
if is_bgr:
|
||
# NBGR8888: B G R (alpha)
|
||
rgb = np.zeros((height, width, 3), dtype=np.uint8)
|
||
rgb[:, :, 0] = pixels[:, :, 2] # R
|
||
rgb[:, :, 1] = pixels[:, :, 1] # G
|
||
rgb[:, :, 2] = pixels[:, :, 0] # B
|
||
else:
|
||
# NRGB8888: R G B (alpha)
|
||
rgb = pixels[:, :, :3].copy()
|
||
else:
|
||
# RGB8888或BGR8888
|
||
if is_bgr:
|
||
# BGR8888: B G R (alpha)
|
||
rgb = np.zeros((height, width, 3), dtype=np.uint8)
|
||
rgb[:, :, 0] = pixels[:, :, 2] # R
|
||
rgb[:, :, 1] = pixels[:, :, 1] # G
|
||
rgb[:, :, 2] = pixels[:, :, 0] # B
|
||
else:
|
||
# RGB8888: R G B (alpha)
|
||
rgb = pixels[:, :, :3].copy()
|
||
|
||
return rgb
|
||
|
||
|
||
def decode_raw_frame(raw: bytes, spec: ImageSpec, yuv444_uv_order: str = "UV") -> np.ndarray:
|
||
"""
|
||
将原始帧数据按 spec 解码为 RGB np.ndarray (H, W, 3), uint8
|
||
yuv444_uv_order: 针对 YUV444_INTERLEAVED 时 UV顺序选择 "UV" 或 "VU"
|
||
"""
|
||
width, height, fmt = spec.width, spec.height, spec.fmt
|
||
arr = np.frombuffer(raw, dtype=np.uint8)
|
||
|
||
# YUV格式解析
|
||
if fmt in (PixelFormat.YUYV, PixelFormat.YVYU, PixelFormat.UYVY, PixelFormat.VYUY):
|
||
Y, U, V = unpack_packed_422_to_planes(arr, width, height, fmt)
|
||
rgb = yuv_to_rgb_numpy(Y, U, V)
|
||
elif fmt == PixelFormat.NV12:
|
||
Y, U, V = parse_nvxx(arr, width, height, "UV", is_420=True)
|
||
rgb = yuv_to_rgb_numpy(Y, U, V)
|
||
elif fmt == PixelFormat.NV21:
|
||
Y, U, V = parse_nvxx(arr, width, height, "VU", is_420=True)
|
||
rgb = yuv_to_rgb_numpy(Y, U, V)
|
||
elif fmt == PixelFormat.NV16:
|
||
Y, U, V = parse_nvxx(arr, width, height, "UV", is_420=False)
|
||
rgb = yuv_to_rgb_numpy(Y, U, V)
|
||
elif fmt == PixelFormat.NV61:
|
||
Y, U, V = parse_nvxx(arr, width, height, "VU", is_420=False)
|
||
rgb = yuv_to_rgb_numpy(Y, U, V)
|
||
elif fmt == PixelFormat.GRAY:
|
||
Y, U, V = parse_gray(arr, width, height)
|
||
rgb = yuv_to_rgb_numpy(Y, U, V)
|
||
elif fmt == PixelFormat.YUV444_PLANAR:
|
||
Y, U, V = parse_yuv444_planar(arr, width, height)
|
||
rgb = yuv_to_rgb_numpy(Y, U, V)
|
||
elif fmt == PixelFormat.YUV444_INTERLEAVED_UV:
|
||
Y, U, V = parse_yuv444_interleaved(arr, width, height, "UV")
|
||
rgb = yuv_to_rgb_numpy(Y, U, V)
|
||
elif fmt == PixelFormat.YUV444_INTERLEAVED_VU:
|
||
Y, U, V = parse_yuv444_interleaved(arr, width, height, "VU")
|
||
rgb = yuv_to_rgb_numpy(Y, U, V)
|
||
|
||
# RGB/BGR格式解析
|
||
elif fmt == PixelFormat.RGB565:
|
||
rgb = parse_rgb565(arr, width, height, is_bgr=False)
|
||
elif fmt == PixelFormat.BGR565:
|
||
rgb = parse_rgb565(arr, width, height, is_bgr=True)
|
||
elif fmt == PixelFormat.RGB888:
|
||
rgb = parse_rgb888(arr, width, height, is_bgr=False)
|
||
elif fmt == PixelFormat.BGR888:
|
||
rgb = parse_rgb888(arr, width, height, is_bgr=True)
|
||
elif fmt == PixelFormat.RGB8888:
|
||
rgb = parse_rgb8888(arr, width, height, is_bgr=False, is_native=False)
|
||
elif fmt == PixelFormat.BGR8888:
|
||
rgb = parse_rgb8888(arr, width, height, is_bgr=True, is_native=False)
|
||
elif fmt == PixelFormat.NRGB8888:
|
||
rgb = parse_rgb8888(arr, width, height, is_bgr=False, is_native=True)
|
||
elif fmt == PixelFormat.NBGR8888:
|
||
rgb = parse_rgb8888(arr, width, height, is_bgr=True, is_native=True)
|
||
else:
|
||
raise ValueError(f"不支持的像素格式: {fmt}")
|
||
|
||
return rgb
|
||
|
||
|
||
# -----------------------------
|
||
# QImage 显示与栅格绘制
|
||
# -----------------------------
|
||
|
||
class GridPixmap(QtGui.QPixmap):
|
||
def __init__(self, base_image: QtGui.QImage, show_grid: bool, grid_step: int, grid_color: QtGui.QColor):
|
||
super().__init__(QtGui.QPixmap.fromImage(base_image))
|
||
if show_grid and grid_step > 0:
|
||
painter = QtGui.QPainter(self)
|
||
pen = QtGui.QPen(grid_color)
|
||
pen.setWidth(1)
|
||
painter.setPen(pen)
|
||
w = self.width()
|
||
h = self.height()
|
||
# 画垂直线
|
||
x = 0
|
||
while x < w:
|
||
painter.drawLine(x, 0, x, h)
|
||
x += grid_step
|
||
# 画水平线
|
||
y = 0
|
||
while y < h:
|
||
painter.drawLine(0, y, w, y)
|
||
y += grid_step
|
||
painter.end()
|
||
|
||
|
||
def numpy_to_qimage(rgb: np.ndarray) -> QtGui.QImage:
|
||
h, w, _ = rgb.shape
|
||
bytes_per_line = 3 * w
|
||
# 注意:必须保证连续内存
|
||
if not rgb.flags['C_CONTIGUOUS']:
|
||
rgb = np.ascontiguousarray(rgb)
|
||
return QtGui.QImage(rgb.data, w, h, bytes_per_line, QtGui.QImage.Format_RGB888).copy()
|
||
|
||
|
||
# -----------------------------
|
||
# 主窗口与UI
|
||
# -----------------------------
|
||
|
||
class ImageViewer(QtWidgets.QLabel):
|
||
def __init__(self):
|
||
super().__init__()
|
||
self.setAlignment(QtCore.Qt.AlignCenter)
|
||
self.setBackgroundRole(QtGui.QPalette.Base)
|
||
self.setSizePolicy(QtWidgets.QSizePolicy.Ignored, QtWidgets.QSizePolicy.Ignored)
|
||
# 修改:设置为不缩放,保持原始大小
|
||
self.setScaledContents(False)
|
||
|
||
def set_pixmap(self, pixmap: QtGui.QPixmap):
|
||
self.setPixmap(pixmap)
|
||
# 调整窗口以适应图片大小(可选)
|
||
self.resize(pixmap.size())
|
||
|
||
|
||
class MainWindow(QtWidgets.QMainWindow):
|
||
def __init__(self):
|
||
super().__init__()
|
||
self.setWindowTitle("Raw YUV Viewer")
|
||
self.resize(1000, 700)
|
||
|
||
# 创建一个滚动区域来容纳图像查看器(当图像太大时)
|
||
self.scroll_area = QtWidgets.QScrollArea()
|
||
self.scroll_area.setWidgetResizable(True)
|
||
self.scroll_area.setAlignment(QtCore.Qt.AlignCenter)
|
||
|
||
self.viewer = ImageViewer()
|
||
self.scroll_area.setWidget(self.viewer)
|
||
self.setCentralWidget(self.scroll_area)
|
||
|
||
# 当前状态
|
||
self.image_spec = ImageSpec(640, 480, PixelFormat.NV12)
|
||
self.yuv444_uv_order = "UV" # 仅对 YUV444 interleaved 使用
|
||
self.current_file: Optional[str] = None
|
||
self.current_rgb: Optional[np.ndarray] = None
|
||
|
||
# 栅格设置
|
||
self.show_grid = False
|
||
self.grid_step = 8
|
||
self.grid_color = QtGui.QColor(0, 255, 0, 120)
|
||
|
||
self._create_actions()
|
||
self._create_menus()
|
||
|
||
# 启用拖放
|
||
self.setAcceptDrops(True)
|
||
|
||
# 状态栏
|
||
self.statusBar().showMessage("拖入原始帧文件,或 文件 -> 打开")
|
||
|
||
# ---- 菜单和动作 ----
|
||
|
||
def _create_actions(self):
|
||
self.act_open = QtWidgets.QAction("打开...", self)
|
||
self.act_open.setShortcut("Ctrl+O")
|
||
self.act_open.triggered.connect(self.open_file_dialog)
|
||
|
||
self.act_quit = QtWidgets.QAction("退出", self)
|
||
self.act_quit.setShortcut("Ctrl+Q")
|
||
self.act_quit.triggered.connect(self.close)
|
||
|
||
self.act_toggle_grid = QtWidgets.QAction("显示栅格", self, checkable=True)
|
||
self.act_toggle_grid.triggered.connect(self.toggle_grid)
|
||
|
||
self.act_set_grid_step = QtWidgets.QAction("设置栅格步长...", self)
|
||
self.act_set_grid_step.triggered.connect(self.set_grid_step)
|
||
|
||
self.act_rerender = QtWidgets.QAction("重新渲染", self)
|
||
self.act_rerender.setShortcut("F5")
|
||
self.act_rerender.triggered.connect(self.rerender)
|
||
|
||
self.act_fit_to_window = QtWidgets.QAction("适应窗口大小", self, checkable=True)
|
||
self.act_fit_to_window.triggered.connect(self.toggle_fit_to_window)
|
||
|
||
self.act_actual_size = QtWidgets.QAction("实际大小", self)
|
||
self.act_actual_size.triggered.connect(self.show_actual_size)
|
||
|
||
def _create_menus(self):
|
||
m_file = self.menuBar().addMenu("文件")
|
||
m_file.addAction(self.act_open)
|
||
m_file.addSeparator()
|
||
m_file.addAction(self.act_quit)
|
||
|
||
m_settings = self.menuBar().addMenu("设置")
|
||
|
||
# 创建尺寸子菜单
|
||
m_size = m_settings.addMenu("图像尺寸")
|
||
for size_text in PREDEFINED_SIZES:
|
||
if size_text == "自定义...":
|
||
m_size.addSeparator()
|
||
act_custom = QtWidgets.QAction("自定义...", self)
|
||
act_custom.triggered.connect(self.set_custom_size)
|
||
m_size.addAction(act_custom)
|
||
else:
|
||
act = QtWidgets.QAction(size_text, self)
|
||
act.triggered.connect(lambda checked, s=size_text: self.set_predefined_size(s))
|
||
m_size.addAction(act)
|
||
|
||
# 创建格式子菜单
|
||
m_format = m_settings.addMenu("像素格式")
|
||
for fmt in SUPPORTED_FORMATS:
|
||
act = QtWidgets.QAction(fmt, self, checkable=True)
|
||
act.setChecked(fmt == self.image_spec.fmt)
|
||
act.triggered.connect(lambda checked, f=fmt: self.set_pixel_format(f))
|
||
m_format.addAction(act)
|
||
|
||
m_settings.addSeparator()
|
||
m_settings.addAction(self.act_rerender)
|
||
|
||
m_view = self.menuBar().addMenu("视图")
|
||
m_view.addAction(self.act_toggle_grid)
|
||
m_view.addAction(self.act_set_grid_step)
|
||
m_view.addSeparator()
|
||
m_view.addAction(self.act_fit_to_window)
|
||
m_view.addAction(self.act_actual_size)
|
||
|
||
# ---- 事件:拖放 ----
|
||
|
||
def dragEnterEvent(self, event: QtGui.QDragEnterEvent):
|
||
if event.mimeData().hasUrls():
|
||
event.acceptProposedAction()
|
||
|
||
def dropEvent(self, event: QtGui.QDropEvent):
|
||
urls = event.mimeData().urls()
|
||
if not urls:
|
||
return
|
||
path = urls[0].toLocalFile()
|
||
if path:
|
||
self.load_file(path)
|
||
|
||
# ---- 打开/加载/渲染 ----
|
||
|
||
def open_file_dialog(self):
|
||
path, _ = QtWidgets.QFileDialog.getOpenFileName(self, "选择原始帧文件", "", "All Files (*)")
|
||
if path:
|
||
self.load_file(path)
|
||
|
||
def load_file(self, path: str):
|
||
try:
|
||
with open(path, "rb") as f:
|
||
raw = f.read()
|
||
self.current_file = path
|
||
self.statusBar().showMessage(f"已加载: {os.path.basename(path)} ({len(raw)} bytes)")
|
||
self.current_rgb = self.decode_with_current_spec(raw)
|
||
self.update_view()
|
||
except Exception as e:
|
||
QtWidgets.QMessageBox.critical(self, "错误", str(e))
|
||
|
||
def decode_with_current_spec(self, raw: bytes) -> np.ndarray:
|
||
# 对 YUV444_INTERLEAVED_*,uv 顺序已在格式名中决定;也保留一个变量用于未来扩展
|
||
return decode_raw_frame(raw, self.image_spec, self.yuv444_uv_order)
|
||
|
||
def update_view(self):
|
||
if self.current_rgb is None:
|
||
self.viewer.setPixmap(QtGui.QPixmap())
|
||
return
|
||
qimg = numpy_to_qimage(self.current_rgb)
|
||
pm = GridPixmap(qimg, self.show_grid, self.grid_step, self.grid_color)
|
||
self.viewer.set_pixmap(pm)
|
||
|
||
# 更新窗口标题显示当前信息
|
||
self.setWindowTitle(f"Raw YUV Viewer - {self.image_spec.width}x{self.image_spec.height} {self.image_spec.fmt}")
|
||
|
||
def rerender(self):
|
||
if not self.current_file:
|
||
return
|
||
try:
|
||
with open(self.current_file, "rb") as f:
|
||
raw = f.read()
|
||
self.current_rgb = self.decode_with_current_spec(raw)
|
||
self.update_view()
|
||
except Exception as e:
|
||
QtWidgets.QMessageBox.critical(self, "错误", str(e))
|
||
|
||
# ---- 设置项 ----
|
||
|
||
def set_predefined_size(self, size_text: str):
|
||
"""设置预定义的尺寸"""
|
||
size_part = size_text.split()[0]
|
||
w, h = map(int, size_part.split('x'))
|
||
self.image_spec = ImageSpec(w, h, self.image_spec.fmt)
|
||
self.statusBar().showMessage(f"已设置尺寸: {w}x{h}")
|
||
self.rerender()
|
||
|
||
# 更新格式菜单的选中状态
|
||
self.update_format_menu_checked()
|
||
|
||
def set_custom_size(self):
|
||
"""设置自定义尺寸"""
|
||
w, ok1 = QtWidgets.QInputDialog.getInt(self, "图像宽度", "宽 (pixels):", self.image_spec.width, 1, 16384, 1)
|
||
if not ok1:
|
||
return
|
||
h, ok2 = QtWidgets.QInputDialog.getInt(self, "图像高度", "高 (pixels):", self.image_spec.height, 1, 16384, 1)
|
||
if not ok2:
|
||
return
|
||
self.image_spec = ImageSpec(w, h, self.image_spec.fmt)
|
||
self.statusBar().showMessage(f"已设置尺寸: {w}x{h}")
|
||
self.rerender()
|
||
|
||
def set_pixel_format(self, fmt: str):
|
||
"""设置像素格式"""
|
||
self.image_spec = ImageSpec(self.image_spec.width, self.image_spec.height, fmt)
|
||
self.statusBar().showMessage(f"已设置像素格式: {fmt}")
|
||
self.rerender()
|
||
|
||
# 更新格式菜单的选中状态
|
||
self.update_format_menu_checked()
|
||
|
||
def update_format_menu_checked(self):
|
||
"""更新格式菜单的选中状态"""
|
||
format_menu = self.menuBar().actions()[1].menu().actions()[1].menu() # 获取格式子菜单
|
||
for action in format_menu.actions():
|
||
if action.text() == self.image_spec.fmt:
|
||
action.setChecked(True)
|
||
else:
|
||
action.setChecked(False)
|
||
|
||
def toggle_grid(self, checked: bool):
|
||
self.show_grid = checked
|
||
self.update_view()
|
||
|
||
def set_grid_step(self):
|
||
step, ok = QtWidgets.QInputDialog.getInt(self, "栅格步长", "像素步长:", self.grid_step, 1, 512, 1)
|
||
if not ok:
|
||
return
|
||
self.grid_step = step
|
||
self.update_view()
|
||
|
||
def toggle_fit_to_window(self, checked: bool):
|
||
"""切换是否适应窗口大小"""
|
||
if checked:
|
||
self.scroll_area.setWidgetResizable(True)
|
||
self.viewer.setScaledContents(True)
|
||
else:
|
||
self.scroll_area.setWidgetResizable(False)
|
||
self.viewer.setScaledContents(False)
|
||
if self.viewer.pixmap():
|
||
self.viewer.resize(self.viewer.pixmap().size())
|
||
|
||
def show_actual_size(self):
|
||
"""显示实际大小(取消适应窗口)"""
|
||
self.act_fit_to_window.setChecked(False)
|
||
self.scroll_area.setWidgetResizable(False)
|
||
self.viewer.setScaledContents(False)
|
||
if self.viewer.pixmap():
|
||
self.viewer.resize(self.viewer.pixmap().size())
|
||
|
||
|
||
def main():
|
||
app = QtWidgets.QApplication(sys.argv)
|
||
w = MainWindow()
|
||
w.show()
|
||
sys.exit(app.exec_())
|
||
|
||
|
||
if __name__ == "__main__":
|
||
main() |