GrabBag/Tools/VrEyeView/Src/VrEyeViewWidget.cpp

641 lines
22 KiB
C++
Raw Normal View History

2026-02-18 15:11:41 +08:00
#include "VrEyeViewWidget.h"
2026-02-21 00:28:04 +08:00
#include "../SpinBoxPasteHelper.h"
2026-02-18 15:11:41 +08:00
#include <QVBoxLayout>
#include <QHBoxLayout>
#include <QGridLayout>
#include <QGroupBox>
#include <QLabel>
#include <QLineEdit>
#include <QMessageBox>
#include <QFileDialog>
#include <QFileInfo>
#include <QPainter>
#include <QScrollArea>
2026-02-21 00:28:04 +08:00
#include <QResizeEvent>
2026-02-18 15:11:41 +08:00
#include <cmath>
VrEyeViewWidget::VrEyeViewWidget(QWidget* parent)
: QWidget(parent)
, m_eyeDevice(nullptr)
, m_detector(nullptr)
, m_leftImageLabel(nullptr)
, m_rightImageLabel(nullptr)
, m_isConnected(false)
, m_isCapturing(false)
, m_hasNewImage(false)
{
// 创建设备和检测器
IVrEyeDevice::CreateObject(&m_eyeDevice);
m_detector = CreateChessboardDetectorInstance();
setupUI();
2026-02-21 00:28:04 +08:00
// 为所有 SpinBox 安装粘贴过滤器
SpinBoxPasteHelper::install(this);
2026-02-18 15:11:41 +08:00
// 初始化设备
if (m_eyeDevice) {
m_eyeDevice->InitDevice();
}
// 创建显示定时器
m_displayTimer = new QTimer(this);
connect(m_displayTimer, &QTimer::timeout, this, &VrEyeViewWidget::onUpdateDisplay);
m_displayTimer->start(33); // 约30fps
}
VrEyeViewWidget::~VrEyeViewWidget()
{
if (m_isCapturing) {
onStopCapture();
}
if (m_isConnected) {
onDisconnectCamera();
}
if (m_detector) {
DestroyChessboardDetectorInstance(m_detector);
m_detector = nullptr;
}
if (m_eyeDevice) {
delete m_eyeDevice;
m_eyeDevice = nullptr;
}
}
void VrEyeViewWidget::setupUI()
{
QVBoxLayout* mainLayout = new QVBoxLayout(this);
2026-02-21 00:28:04 +08:00
mainLayout->setSpacing(4);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
// ===== 左右目图像并排显示区域 =====
2026-02-18 15:11:41 +08:00
QHBoxLayout* imageLayout = new QHBoxLayout();
QVBoxLayout* leftLayout = new QVBoxLayout();
2026-02-21 00:28:04 +08:00
leftLayout->setSpacing(0);
QLabel* leftTitle = new QLabel("左目", this);
leftTitle->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
leftLayout->addWidget(leftTitle);
2026-02-18 15:11:41 +08:00
m_leftImageLabel = new QLabel(this);
m_leftImageLabel->setMinimumSize(320, 240);
2026-02-21 00:28:04 +08:00
m_leftImageLabel->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
2026-02-18 15:11:41 +08:00
m_leftImageLabel->setAlignment(Qt::AlignCenter);
m_leftImageLabel->setStyleSheet("QLabel { background-color: black; }");
2026-02-21 00:28:04 +08:00
leftLayout->addWidget(m_leftImageLabel, 1);
2026-02-18 15:11:41 +08:00
imageLayout->addLayout(leftLayout);
QVBoxLayout* rightLayout = new QVBoxLayout();
2026-02-21 00:28:04 +08:00
rightLayout->setSpacing(0);
QLabel* rightTitle = new QLabel("右目", this);
rightTitle->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
rightLayout->addWidget(rightTitle);
2026-02-18 15:11:41 +08:00
m_rightImageLabel = new QLabel(this);
m_rightImageLabel->setMinimumSize(320, 240);
2026-02-21 00:28:04 +08:00
m_rightImageLabel->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
2026-02-18 15:11:41 +08:00
m_rightImageLabel->setAlignment(Qt::AlignCenter);
m_rightImageLabel->setStyleSheet("QLabel { background-color: black; }");
2026-02-21 00:28:04 +08:00
rightLayout->addWidget(m_rightImageLabel, 1);
2026-02-18 15:11:41 +08:00
imageLayout->addLayout(rightLayout);
mainLayout->addLayout(imageLayout, 1);
2026-02-21 00:28:04 +08:00
// ===== 第一行:相机连接 + 参数 + 采集按钮 =====
QHBoxLayout* row1 = new QHBoxLayout();
row1->setSpacing(4);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
// 相机IP + 连接/断开
row1->addWidget(new QLabel("IP:", this));
2026-02-18 15:11:41 +08:00
m_editCameraIP = new QLineEdit("192.168.1.100", this);
2026-02-21 00:28:04 +08:00
m_editCameraIP->setFixedWidth(120);
row1->addWidget(m_editCameraIP);
2026-02-18 15:11:41 +08:00
m_btnConnect = new QPushButton("连接", this);
m_btnDisconnect = new QPushButton("断开", this);
m_btnDisconnect->setEnabled(false);
connect(m_btnConnect, &QPushButton::clicked, this, &VrEyeViewWidget::onConnectCamera);
connect(m_btnDisconnect, &QPushButton::clicked, this, &VrEyeViewWidget::onDisconnectCamera);
2026-02-21 00:28:04 +08:00
row1->addWidget(m_btnConnect);
row1->addWidget(m_btnDisconnect);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
// 分隔
row1->addSpacing(8);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
// 曝光/增益
row1->addWidget(new QLabel("曝光:", this));
2026-02-18 15:11:41 +08:00
m_sbExposure = new QSpinBox(this);
m_sbExposure->setRange(100, 10000);
m_sbExposure->setValue(1000);
2026-02-21 00:28:04 +08:00
row1->addWidget(m_sbExposure);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
row1->addWidget(new QLabel("增益:", this));
2026-02-18 15:11:41 +08:00
m_sbGain = new QSpinBox(this);
m_sbGain->setRange(0, 255);
m_sbGain->setValue(100);
2026-02-21 00:28:04 +08:00
row1->addWidget(m_sbGain);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
// 分隔
row1->addSpacing(8);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
// 采集 + 加载按钮
2026-02-18 15:11:41 +08:00
m_btnStartCapture = new QPushButton("开始采集", this);
m_btnStopCapture = new QPushButton("停止采集", this);
2026-02-21 00:28:04 +08:00
QPushButton* btnLoadLeftImage = new QPushButton("加载左图", this);
QPushButton* btnLoadRightImage = new QPushButton("加载右图", this);
2026-02-18 15:11:41 +08:00
m_btnStartCapture->setEnabled(false);
m_btnStopCapture->setEnabled(false);
connect(m_btnStartCapture, &QPushButton::clicked, this, &VrEyeViewWidget::onStartCapture);
connect(m_btnStopCapture, &QPushButton::clicked, this, &VrEyeViewWidget::onStopCapture);
2026-02-21 00:28:04 +08:00
connect(btnLoadLeftImage, &QPushButton::clicked, this, &VrEyeViewWidget::onLoadLeftImage);
connect(btnLoadRightImage, &QPushButton::clicked, this, &VrEyeViewWidget::onLoadRightImage);
row1->addWidget(m_btnStartCapture);
row1->addWidget(m_btnStopCapture);
row1->addWidget(btnLoadLeftImage);
row1->addWidget(btnLoadRightImage);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
row1->addStretch();
mainLayout->addLayout(row1);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
// ===== 第二行:相机内参(左) + 标定板检测(右) =====
QHBoxLayout* row2 = new QHBoxLayout();
row2->setSpacing(4);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
// 相机内参3x3矩阵
QGroupBox* intrinsicsGroup = new QGroupBox("相机内参", this);
QGridLayout* intrinsicsLayout = new QGridLayout(intrinsicsGroup);
intrinsicsLayout->setSpacing(2);
intrinsicsLayout->setContentsMargins(4, 8, 4, 4);
auto createIntrinsicSpinBox = [this](double minVal, double maxVal, double defaultVal) {
QDoubleSpinBox* sb = new QDoubleSpinBox(this);
sb->setRange(minVal, maxVal);
sb->setValue(defaultVal);
sb->setDecimals(2);
return sb;
};
auto createFixedLabel = [this](const QString& text) {
QLabel* lbl = new QLabel(text, this);
lbl->setAlignment(Qt::AlignCenter);
lbl->setStyleSheet("QLabel { color: gray; }");
return lbl;
};
m_sbFx = createIntrinsicSpinBox(100, 5000, 2384.8520129909352);
m_sbFx->setPrefix("fx: ");
m_sbFx->setToolTip("fx: X方向焦距(像素)");
m_sbFy = createIntrinsicSpinBox(100, 5000, 2384.8520129909352);
m_sbFy->setPrefix("fy: ");
m_sbFy->setToolTip("fy: Y方向焦距(像素)");
m_sbCx = createIntrinsicSpinBox(0, 2000, 232.37469863891602);
m_sbCx->setPrefix("cx: ");
m_sbCx->setToolTip("cx: 主点X坐标(像素)");
m_sbCy = createIntrinsicSpinBox(0, 2000, 1054.649814605713);
m_sbCy->setPrefix("cy: ");
m_sbCy->setToolTip("cy: 主点Y坐标(像素)");
intrinsicsLayout->addWidget(m_sbFx, 0, 0);
intrinsicsLayout->addWidget(createFixedLabel("0"), 0, 1);
intrinsicsLayout->addWidget(m_sbCx, 0, 2);
intrinsicsLayout->addWidget(createFixedLabel("0"), 1, 0);
intrinsicsLayout->addWidget(m_sbFy, 1, 1);
intrinsicsLayout->addWidget(m_sbCy, 1, 2);
intrinsicsLayout->addWidget(createFixedLabel("0"), 2, 0);
intrinsicsLayout->addWidget(createFixedLabel("0"), 2, 1);
intrinsicsLayout->addWidget(createFixedLabel("1"), 2, 2);
row2->addWidget(intrinsicsGroup);
// 标定板检测
2026-02-18 15:11:41 +08:00
QGroupBox* detectionGroup = new QGroupBox("标定板检测", this);
QGridLayout* detectionLayout = new QGridLayout(detectionGroup);
2026-02-21 00:28:04 +08:00
detectionLayout->setSpacing(2);
detectionLayout->setContentsMargins(4, 8, 4, 4);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
detectionLayout->addWidget(new QLabel("角点宽:", this), 0, 0);
2026-02-18 15:11:41 +08:00
m_sbPatternWidth = new QSpinBox(this);
m_sbPatternWidth->setRange(2, 20);
m_sbPatternWidth->setValue(8);
detectionLayout->addWidget(m_sbPatternWidth, 0, 1);
2026-02-21 00:28:04 +08:00
detectionLayout->addWidget(new QLabel("角点高:", this), 0, 2);
2026-02-18 15:11:41 +08:00
m_sbPatternHeight = new QSpinBox(this);
m_sbPatternHeight->setRange(2, 20);
m_sbPatternHeight->setValue(11);
detectionLayout->addWidget(m_sbPatternHeight, 0, 3);
2026-02-21 00:28:04 +08:00
detectionLayout->addWidget(new QLabel("格子(mm):", this), 1, 0);
2026-02-18 15:11:41 +08:00
m_sbSquareSize = new QDoubleSpinBox(this);
m_sbSquareSize->setRange(1.0, 100.0);
m_sbSquareSize->setValue(30.0);
m_sbSquareSize->setDecimals(2);
detectionLayout->addWidget(m_sbSquareSize, 1, 1);
2026-02-21 00:28:04 +08:00
m_cbAdaptiveThresh = new QCheckBox("自适应阈值", this);
m_cbAdaptiveThresh->setChecked(true);
m_cbNormalizeImage = new QCheckBox("归一化图像", this);
m_cbNormalizeImage->setChecked(true);
detectionLayout->addWidget(m_cbAdaptiveThresh, 1, 2);
detectionLayout->addWidget(m_cbNormalizeImage, 1, 3);
2026-02-18 15:11:41 +08:00
m_cbAutoDetect = new QCheckBox("自动检测", this);
2026-02-21 00:28:04 +08:00
detectionLayout->addWidget(m_cbAutoDetect, 2, 0, 1, 2);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
m_btnDetect = new QPushButton("计算标定板信息", this);
2026-02-18 15:11:41 +08:00
m_btnDetect->setEnabled(false);
connect(m_btnDetect, &QPushButton::clicked, this, &VrEyeViewWidget::onDetectChessboard);
2026-02-21 00:28:04 +08:00
detectionLayout->addWidget(m_btnDetect, 2, 2, 1, 2);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
row2->addWidget(detectionGroup);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
mainLayout->addLayout(row2);
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
// ===== 状态栏 =====
2026-02-18 15:11:41 +08:00
m_lblStatus = new QLabel("状态: 未连接", this);
m_lblDetectionResult = new QLabel("检测结果: 无", this);
mainLayout->addWidget(m_lblStatus);
mainLayout->addWidget(m_lblDetectionResult);
}
void VrEyeViewWidget::SetDetectionCallback(DetectionCallback callback)
{
m_detectionCallback = callback;
}
void VrEyeViewWidget::onConnectCamera()
{
if (!m_eyeDevice) return;
QString ip = m_editCameraIP->text();
int ret = m_eyeDevice->OpenDevice(ip.toStdString().c_str(), false, false, false);
if (ret == 0) {
m_isConnected = true;
m_btnConnect->setEnabled(false);
m_btnDisconnect->setEnabled(true);
m_btnStartCapture->setEnabled(true);
m_lblStatus->setText("状态: 已连接");
// 设置相机参数
unsigned int exposure = m_sbExposure->value();
unsigned int gain = m_sbGain->value();
m_eyeDevice->SetEyeExpose(exposure);
m_eyeDevice->SetEyeGain(gain);
} else {
QMessageBox::warning(this, "错误", QString("连接相机失败,错误码: %1").arg(ret));
}
}
void VrEyeViewWidget::onDisconnectCamera()
{
if (!m_eyeDevice) return;
if (m_isCapturing) {
onStopCapture();
}
m_eyeDevice->CloseDevice();
m_isConnected = false;
m_btnConnect->setEnabled(true);
m_btnDisconnect->setEnabled(false);
m_btnStartCapture->setEnabled(false);
m_lblStatus->setText("状态: 未连接");
}
void VrEyeViewWidget::onStartCapture()
{
if (!m_eyeDevice || !m_isConnected) return;
2026-02-21 00:28:04 +08:00
int ret = m_eyeDevice->StartCapture(OnImageCallback, this);
2026-02-18 15:11:41 +08:00
if (ret == 0) {
m_isCapturing = true;
m_btnStartCapture->setEnabled(false);
m_btnStopCapture->setEnabled(true);
m_btnDetect->setEnabled(true);
m_lblStatus->setText("状态: 采集中");
} else {
QMessageBox::warning(this, "错误", QString("开始采集失败,错误码: %1").arg(ret));
}
}
void VrEyeViewWidget::onStopCapture()
{
if (!m_eyeDevice) return;
2026-02-21 00:28:04 +08:00
m_eyeDevice->StopCapture();
2026-02-18 15:11:41 +08:00
m_isCapturing = false;
m_btnStartCapture->setEnabled(true);
m_btnStopCapture->setEnabled(false);
m_lblStatus->setText("状态: 已连接");
}
2026-02-21 00:28:04 +08:00
void VrEyeViewWidget::OnImageCallback(SVzNLImageData* pLeftImage,
SVzNLImageData* pRightImage,
SVzNLImageData* /*pCenterImage*/,
const SVzOutputFrameProps* /*pFrameProps*/,
void* pUserData)
2026-02-18 15:11:41 +08:00
{
VrEyeViewWidget* pThis = static_cast<VrEyeViewWidget*>(pUserData);
if (pThis) {
2026-02-21 00:28:04 +08:00
pThis->ProcessImageData(pLeftImage, pRightImage);
2026-02-18 15:11:41 +08:00
}
}
2026-02-21 00:28:04 +08:00
/**
* @brief SDK QImage
*/
static QImage SvzImageToQImage(SVzNLImageData* pImage)
2026-02-18 15:11:41 +08:00
{
2026-02-21 00:28:04 +08:00
if (!pImage || !pImage->pBuffer || pImage->nWidth == 0 || pImage->nHeight == 0) {
return QImage();
}
int w = static_cast<int>(pImage->nWidth);
int h = static_cast<int>(pImage->nHeight);
if (pImage->eImageType == keVzNLImageType_RGB888) {
// RGB888: 直接拷贝
return QImage(pImage->pBuffer, w, h, w * 3, QImage::Format_RGB888).copy();
} else if (pImage->eImageType == keVzNLImageType_BGR888) {
// BGR888: 转为 RGB
QImage img(pImage->pBuffer, w, h, w * 3, QImage::Format_RGB888);
return img.rgbSwapped();
} else if (pImage->eImageType == keVzNLImageType_GRAY) {
// 灰度图
return QImage(pImage->pBuffer, w, h, w, QImage::Format_Grayscale8).copy();
} else if (pImage->eImageType == keVzNLImageType_BGRA8888) {
// BGRA -> ARGB32
QImage img(pImage->pBuffer, w, h, w * 4, QImage::Format_ARGB32);
return img.copy();
}
// 其他格式:按灰度处理
if (pImage->nChannels == 1) {
return QImage(pImage->pBuffer, w, h, w, QImage::Format_Grayscale8).copy();
} else if (pImage->nChannels == 3) {
return QImage(pImage->pBuffer, w, h, w * 3, QImage::Format_RGB888).copy();
}
return QImage();
}
void VrEyeViewWidget::ProcessImageData(SVzNLImageData* pLeftImage, SVzNLImageData* pRightImage)
{
QImage left = SvzImageToQImage(pLeftImage);
QImage right = SvzImageToQImage(pRightImage);
{
std::lock_guard<std::mutex> lock(m_captureMutex);
if (!left.isNull()) m_captureLeft = left;
if (!right.isNull()) m_captureRight = right;
m_hasNewImage = true;
}
2026-02-18 15:11:41 +08:00
}
void VrEyeViewWidget::onUpdateDisplay()
{
2026-02-21 00:28:04 +08:00
if (!m_hasNewImage) return;
{
std::lock_guard<std::mutex> lock(m_captureMutex);
if (!m_captureLeft.isNull()) m_leftImage = m_captureLeft;
if (!m_captureRight.isNull()) m_rightImage = m_captureRight;
m_hasNewImage = false;
}
// 清除旧的检测角点
m_lastLeftCorners.clear();
m_lastRightCorners.clear();
UpdateImageDisplay();
2026-02-18 15:11:41 +08:00
}
void VrEyeViewWidget::onDetectChessboard()
{
if (!m_detector) return;
if (m_leftImage.isNull() || m_rightImage.isNull()) {
m_lblDetectionResult->setText("检测结果: 请先加载左右目图片");
return;
}
// 设置检测参数
m_detector->SetDetectionFlags(
m_cbAdaptiveThresh->isChecked(),
m_cbNormalizeImage->isChecked(),
false);
// 准备相机内参
CameraIntrinsics intrinsics;
intrinsics.fx = m_sbFx->value();
intrinsics.fy = m_sbFy->value();
intrinsics.cx = m_sbCx->value();
intrinsics.cy = m_sbCy->value();
// 检测左目标定板
QImage leftRgb = m_leftImage.convertToFormat(QImage::Format_RGB888);
ChessboardDetectResult leftResult;
int ret = m_detector->DetectChessboardWithPose(
leftRgb.bits(),
leftRgb.width(),
leftRgb.height(),
3,
m_sbPatternWidth->value(),
m_sbPatternHeight->value(),
m_sbSquareSize->value(),
intrinsics,
leftResult);
// 检测右目标定板
QImage rightRgb = m_rightImage.convertToFormat(QImage::Format_RGB888);
ChessboardDetectResult rightResult;
int retRight = m_detector->DetectChessboardWithPose(
rightRgb.bits(),
rightRgb.width(),
rightRgb.height(),
3,
m_sbPatternWidth->value(),
m_sbPatternHeight->value(),
m_sbSquareSize->value(),
intrinsics,
rightResult);
// 在两张图上绘制角点并显示
if (ret == 0 && leftResult.detected) {
m_lastLeftCorners = leftResult.corners;
} else {
m_lastLeftCorners.clear();
}
if (retRight == 0 && rightResult.detected) {
m_lastRightCorners = rightResult.corners;
} else {
m_lastRightCorners.clear();
}
// 在副本上绘制角点并显示(不修改原图)
QImage leftDisplay = m_leftImage.copy();
QImage rightDisplay = m_rightImage.copy();
DrawCorners(leftDisplay, m_lastLeftCorners);
DrawCorners(rightDisplay, m_lastRightCorners);
QPixmap leftPm = QPixmap::fromImage(leftDisplay);
m_leftImageLabel->setPixmap(
leftPm.scaled(m_leftImageLabel->size(), Qt::KeepAspectRatio, Qt::SmoothTransformation));
QPixmap rightPm = QPixmap::fromImage(rightDisplay);
m_rightImageLabel->setPixmap(
rightPm.scaled(m_rightImageLabel->size(), Qt::KeepAspectRatio, Qt::SmoothTransformation));
if (ret == 0 && leftResult.detected && retRight == 0 && rightResult.detected) {
m_lastDetection.detected = true;
if (leftResult.hasPose) {
m_lastDetection.x = leftResult.center.x;
m_lastDetection.y = leftResult.center.y;
m_lastDetection.z = leftResult.center.z;
m_lastDetection.nx = leftResult.normal.x;
m_lastDetection.ny = leftResult.normal.y;
m_lastDetection.nz = leftResult.normal.z;
m_lastDetection.rx = leftResult.eulerAngles[0];
m_lastDetection.ry = leftResult.eulerAngles[1];
m_lastDetection.rz = leftResult.eulerAngles[2];
QString resultText = QString("检测结果: "
"左目 %1 个角点, 右目 %2 个角点 | "
2026-02-21 00:28:04 +08:00
"位置: (%3, %4, %5)mm | "
"法向量: (%6, %7, %8) | "
"姿态: (%9, %10, %11)")
2026-02-18 15:11:41 +08:00
.arg(m_lastLeftCorners.size())
.arg(m_lastRightCorners.size())
2026-02-21 00:28:04 +08:00
.arg(m_lastDetection.x, 0, 'f', 3)
.arg(m_lastDetection.y, 0, 'f', 3)
.arg(m_lastDetection.z, 0, 'f', 3)
.arg(m_lastDetection.nx, 0, 'f', 3)
.arg(m_lastDetection.ny, 0, 'f', 3)
.arg(m_lastDetection.nz, 0, 'f', 3)
.arg(m_lastDetection.rx, 0, 'f', 3)
.arg(m_lastDetection.ry, 0, 'f', 3)
.arg(m_lastDetection.rz, 0, 'f', 3);
2026-02-18 15:11:41 +08:00
m_lblDetectionResult->setText(resultText);
if (m_detectionCallback) {
m_detectionCallback(m_lastDetection);
}
emit chessboardDetected(m_lastDetection);
}
} else {
m_lastDetection.detected = false;
QString errorMsg = "检测结果: ";
if (ret != 0 || !leftResult.detected) {
errorMsg += "左目未检测到标定板 ";
}
if (retRight != 0 || !rightResult.detected) {
errorMsg += "右目未检测到标定板";
}
m_lblDetectionResult->setText(errorMsg);
}
}
2026-02-21 00:28:04 +08:00
void VrEyeViewWidget::onLoadLeftImage()
2026-02-18 15:11:41 +08:00
{
QString fileName = QFileDialog::getOpenFileName(
2026-02-21 00:28:04 +08:00
this, "选择左目图片", "",
2026-02-18 15:11:41 +08:00
"图片文件 (*.png *.jpg *.jpeg *.bmp);;所有文件 (*.*)");
2026-02-21 00:28:04 +08:00
if (fileName.isEmpty()) return;
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
QImage image(fileName);
if (image.isNull()) {
2026-02-18 15:11:41 +08:00
QMessageBox::warning(this, "错误",
2026-02-21 00:28:04 +08:00
QString("无法加载左目图片:\n%1").arg(fileName));
2026-02-18 15:11:41 +08:00
return;
}
2026-02-21 00:28:04 +08:00
m_leftImage = image;
UpdateImageDisplay();
QFileInfo fi(fileName);
m_lblStatus->setText(QString("状态: 已加载左目 %1 (%2x%3)")
.arg(fi.fileName())
.arg(image.width()).arg(image.height()));
if (!m_leftImage.isNull() && !m_rightImage.isNull()) {
m_btnDetect->setEnabled(true);
2026-02-18 15:11:41 +08:00
}
2026-02-21 00:28:04 +08:00
}
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
void VrEyeViewWidget::onLoadRightImage()
{
QString fileName = QFileDialog::getOpenFileName(
this, "选择右目图片", "",
"图片文件 (*.png *.jpg *.jpeg *.bmp);;所有文件 (*.*)");
if (fileName.isEmpty()) return;
QImage image(fileName);
if (image.isNull()) {
2026-02-18 15:11:41 +08:00
QMessageBox::warning(this, "错误",
2026-02-21 00:28:04 +08:00
QString("无法加载右目图片:\n%1").arg(fileName));
2026-02-18 15:11:41 +08:00
return;
}
2026-02-21 00:28:04 +08:00
m_rightImage = image;
2026-02-18 15:11:41 +08:00
UpdateImageDisplay();
2026-02-21 00:28:04 +08:00
QFileInfo fi(fileName);
m_lblStatus->setText(QString("状态: 已加载右目 %1 (%2x%3)")
.arg(fi.fileName())
.arg(image.width()).arg(image.height()));
2026-02-18 15:11:41 +08:00
2026-02-21 00:28:04 +08:00
if (!m_leftImage.isNull() && !m_rightImage.isNull()) {
m_btnDetect->setEnabled(true);
2026-02-18 15:11:41 +08:00
}
}
void VrEyeViewWidget::UpdateImageDisplay()
{
// 左目:保持比例缩放到 label 大小,不拉伸
if (!m_leftImage.isNull()) {
2026-02-21 00:28:04 +08:00
QImage leftDisplay = m_leftImage.copy();
if (!m_lastLeftCorners.empty()) {
DrawCorners(leftDisplay, m_lastLeftCorners);
}
QPixmap pm = QPixmap::fromImage(leftDisplay);
2026-02-18 15:11:41 +08:00
m_leftImageLabel->setPixmap(
pm.scaled(m_leftImageLabel->size(), Qt::KeepAspectRatio, Qt::SmoothTransformation));
}
// 右目
if (!m_rightImage.isNull()) {
2026-02-21 00:28:04 +08:00
QImage rightDisplay = m_rightImage.copy();
if (!m_lastRightCorners.empty()) {
DrawCorners(rightDisplay, m_lastRightCorners);
}
QPixmap pm = QPixmap::fromImage(rightDisplay);
2026-02-18 15:11:41 +08:00
m_rightImageLabel->setPixmap(
pm.scaled(m_rightImageLabel->size(), Qt::KeepAspectRatio, Qt::SmoothTransformation));
}
}
2026-02-21 00:28:04 +08:00
void VrEyeViewWidget::resizeEvent(QResizeEvent* event)
{
QWidget::resizeEvent(event);
UpdateImageDisplay();
}
2026-02-18 15:11:41 +08:00
void VrEyeViewWidget::DrawCorners(QImage& image, const std::vector<Point2D>& corners)
{
if (corners.empty()) return;
QPainter painter(&image);
painter.setPen(QPen(Qt::green, 3));
for (const auto& pt : corners) {
painter.drawEllipse(QPointF(pt.x, pt.y), 5, 5);
}
}