WaferGraphicsView.cpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553
  1. #include "WaferGraphicsView.h"
  2. #include <QDebug>
  3. #include <QScrollBar>
  4. WaferGraphicsView::WaferGraphicsView(QGraphicsScene* scene, QWidget* parent)
  5. : QGraphicsView(scene, parent), selecting(false), selectionRect(nullptr), m_scene(scene),
  6. scaleFactor(1.0), isDragging(false), thumbnailLabel(nullptr),
  7. thumbnailVisible(false) {
  8. setRenderHint(QPainter::Antialiasing);
  9. // setDragMode(QGraphicsView::ScrollHandDrag); // 支持拖动视图
  10. setTransformationAnchor(QGraphicsView::AnchorUnderMouse); // 缩放时以鼠标为中心
  11. // 初始化缩略图标签
  12. thumbnailLabel = new QLabel(this);
  13. thumbnailLabel->setFixedSize(150, 150);
  14. thumbnailLabel->move(0, 0); // 默认左上角位置
  15. thumbnailLabel->setStyleSheet("background-color: white; border: 1px solid gray;");
  16. thumbnailLabel->installEventFilter(this);
  17. thumbnailLabel->hide();
  18. topLeftIndex = qMakePair(-1, -1);
  19. bottomRightIndex = qMakePair(-1, -1);
  20. setScene(m_scene);
  21. // 禁用并隐藏水平滚动条
  22. setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
  23. setHorizontalScrollBar(nullptr);
  24. // 禁用并隐藏垂直滚动条
  25. setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
  26. setVerticalScrollBar(nullptr);
  27. setDragMode(QGraphicsView::NoDrag);
  28. }
  29. // 事件过滤器用于处理缩略图拖动
  30. bool WaferGraphicsView::eventFilter(QObject* obj, QEvent* event) {
  31. static QPoint dragStartPosition;
  32. if (obj == thumbnailLabel) {
  33. if (event->type() == QEvent::MouseButtonPress) {
  34. QMouseEvent* me = static_cast<QMouseEvent*>(event);
  35. dragStartPosition = me->pos();
  36. return true;
  37. }
  38. else if (event->type() == QEvent::MouseMove) {
  39. QMouseEvent* me = static_cast<QMouseEvent*>(event);
  40. // 计算新位置
  41. QPoint newPos = thumbnailLabel->pos() + (me->pos() - dragStartPosition);
  42. // 限制在视图范围内
  43. int maxX = this->width() - thumbnailLabel->width();
  44. int maxY = this->height() - thumbnailLabel->height();
  45. // 使用qBound限制坐标范围(0 <= x <= maxX,0 <= y <= maxY)
  46. newPos.setX(qBound(0, newPos.x(), maxX));
  47. newPos.setY(qBound(0, newPos.y(), maxY));
  48. thumbnailLabel->move(newPos);
  49. return true;
  50. }
  51. }
  52. return QGraphicsView::eventFilter(obj, event);
  53. }
  54. void WaferGraphicsView::mousePressEvent(QMouseEvent* event) {
  55. if (event->button() == Qt::LeftButton) {
  56. // 清空选中的 DieItem
  57. for (auto& item : selectedItemsMap) {
  58. DieItem* die = dynamic_cast<DieItem*>(item);
  59. if (die) {
  60. die->setSelected(false); // 取消选中状态
  61. }
  62. }
  63. selectedItemsMap.clear();
  64. if (topLeftItem && topLeftItem->scene()) {
  65. topLeftItem->setRightSelected(false);
  66. }
  67. topLeftItem.clear();
  68. if (bottomRightItem && bottomRightItem->scene()) {
  69. bottomRightItem->setRightSelected(false);
  70. }
  71. bottomRightItem.clear();
  72. topLeftIndex = qMakePair(-1, -1);
  73. bottomRightIndex = qMakePair(-1, -1);
  74. // 获取点击位置的 DieItem
  75. if (selectedItem && selectedItem->scene()) {
  76. selectedItem->setLeftSelected(false);
  77. }
  78. selectedItem.clear();
  79. QGraphicsItem* item = itemAt(event->pos());
  80. if (item) {
  81. if (typeid(*item) == typeid(DieItem)) {
  82. selectedItem = static_cast<DieItem*>(item);
  83. selectedItem->setLeftSelected(true);
  84. if (event->button() == Qt::MouseEventCreatedDoubleClick) {
  85. m_pCViewInterface->GetViewMatrix()->MoveWaferToPoint(selectedItem->getDieIndex());
  86. }
  87. }
  88. }
  89. setCursor(Qt::OpenHandCursor); // 按下时设置为小手
  90. selecting = true;
  91. lastPos = event->pos(); // 记录鼠标位置
  92. }
  93. else if (event->button() == Qt::RightButton) {
  94. // 开始框选
  95. selecting = true;
  96. selectionStart = mapToScene(event->pos());
  97. isDragging = false;
  98. if (!selectionRect) {
  99. selectionRect = new QGraphicsRectItem();
  100. selectionRect->setPen(QPen(Qt::NoPen));
  101. selectionRect->setBrush(QBrush(QColor(0, 0, 255, 50))); // 半透明蓝色
  102. scene()->addItem(selectionRect);
  103. }
  104. selectionRect->setRect(QRectF(selectionStart, QSizeF()));
  105. }
  106. QGraphicsView::mousePressEvent(event);
  107. }
  108. void WaferGraphicsView::mouseMoveEvent(QMouseEvent* event) {
  109. if (selecting && selectionRect) {
  110. QPointF currentPos = mapToScene(event->pos());
  111. selectionRect->setRect(QRectF(selectionStart, currentPos).normalized());
  112. isDragging = true;
  113. }
  114. else if (selecting) {
  115. // 计算鼠标当前位置与上次位置的差值
  116. QPointF delta = event->pos() - lastPos;
  117. // 平移视图
  118. horizontalScrollBar()->setValue(horizontalScrollBar()->value() - delta.x());
  119. verticalScrollBar()->setValue(verticalScrollBar()->value() - delta.y());
  120. lastPos = event->pos(); // 更新鼠标位置
  121. }
  122. QGraphicsView::mouseMoveEvent(event);
  123. }
  124. void WaferGraphicsView::mouseReleaseEvent(QMouseEvent* event) {
  125. if (event->button() == Qt::LeftButton) {
  126. setCursor(Qt::ArrowCursor); // 松开时恢复为箭头
  127. selecting = false;
  128. }
  129. else if (event->button() == Qt::RightButton && selecting) {
  130. selecting = false;
  131. if (selectionRect && isDragging) {
  132. if (selectedItem && selectedItem->scene()) {
  133. selectedItem->setLeftSelected(false);
  134. }
  135. selectedItem.clear();
  136. if (topLeftItem && topLeftItem->scene()) {
  137. topLeftItem->setRightSelected(false);
  138. }
  139. topLeftItem.clear();
  140. if (bottomRightItem && bottomRightItem->scene()) {
  141. bottomRightItem->setRightSelected(false);
  142. }
  143. bottomRightItem.clear();
  144. topLeftIndex = qMakePair(-1, -1);
  145. bottomRightIndex = qMakePair(-1, -1);
  146. QRectF selectedArea = selectionRect->rect();
  147. scene()->removeItem(selectionRect);
  148. delete selectionRect;
  149. selectionRect = nullptr;
  150. QList<QGraphicsItem*> items = scene()->items(selectedArea, Qt::IntersectsItemShape);
  151. for (QGraphicsItem* item : items) {
  152. if (typeid(*item) == typeid(DieItem)) {
  153. DieItem* die = dynamic_cast<DieItem*>(item);
  154. if (die) {
  155. // 将 DieItem 添加到 map 中
  156. selectedItemsMap.insert(qMakePair(die->getRow(), die->getCol()), die);
  157. die->setSelected(true); // 设置选中状态
  158. }
  159. }
  160. }
  161. }
  162. if (selectionRect) {
  163. scene()->removeItem(selectionRect);
  164. delete selectionRect;
  165. selectionRect = nullptr;
  166. }
  167. // 如果没有进行拖动,则弹出右键菜单
  168. if (!isDragging) {
  169. QGraphicsItem* item = itemAt(event->pos());
  170. DieItem* die = dynamic_cast<DieItem*>(item);
  171. QMenu menu;
  172. QAction* showThumb = menu.addAction(thumbnailVisible ? tr("Hide thumbnails", "隐藏缩略图") : tr("Show thumbnails", "显示缩略图"));
  173. connect(showThumb, &QAction::triggered, [this] {
  174. thumbnailVisible ? hideThumbnail() : showThumbnail();
  175. });
  176. //menu.addAction(tr("Send Location", "发送位置"), [this] {
  177. // if (selectedItem) {
  178. // qDebug() << "Row:" << selectedItem->getRow() << "Col:" << selectedItem->getCol();
  179. // selectedItem->setLeftSelected(false);
  180. // selectedItem = nullptr;
  181. // }
  182. // });
  183. if (die) {
  184. menu.addAction(tr("move to current location","移动到该位置"), [this, die] {
  185. for (auto& item : selectedItemsMap) {
  186. DieItem* die = dynamic_cast<DieItem*>(item);
  187. if (die) {
  188. die->setSelected(false);
  189. }
  190. }
  191. selectedItemsMap.clear();
  192. if (topLeftItem && topLeftItem->scene()) {
  193. topLeftItem->setRightSelected(false);
  194. }
  195. topLeftItem.clear();
  196. if (bottomRightItem && bottomRightItem->scene()) {
  197. bottomRightItem->setRightSelected(false);
  198. }
  199. bottomRightItem.clear();
  200. topLeftIndex = qMakePair(-1, -1);
  201. bottomRightIndex = qMakePair(-1, -1);
  202. if (selectedItem && selectedItem->scene()) {
  203. selectedItem->setLeftSelected(false);
  204. }
  205. selectedItem.clear();
  206. selectedItem = die;
  207. selectedItem->setLeftSelected(true);
  208. m_pCViewInterface->GetViewMatrix()->MoveWaferToPoint(die->getDieIndex());
  209. });
  210. // 设置区域边界点菜单
  211. menu.addAction(tr("set Top left point","设为左上点"), [this, die] {
  212. if (topLeftItem && topLeftItem->scene()) {
  213. topLeftItem->setRightSelected(false);
  214. }
  215. topLeftItem.clear();
  216. topLeftItem = die;
  217. topLeftItem->setRightSelected(true);
  218. topLeftIndex = qMakePair(die->getRow(), die->getCol());
  219. if (bottomRightIndex.first >= 0) checkAndCreateRegion();
  220. });
  221. menu.addAction(tr("set bottom right point","设为右下点"), [this, die] {
  222. if (bottomRightItem && bottomRightItem->scene()) {
  223. bottomRightItem->setRightSelected(false);
  224. }
  225. bottomRightItem.clear();
  226. bottomRightItem = die;
  227. bottomRightItem->setRightSelected(true);
  228. bottomRightIndex = qMakePair(die->getRow(), die->getCol());
  229. if (topLeftIndex.first >= 0) checkAndCreateRegion();
  230. });
  231. }
  232. menu.addAction(tr("clear the selected area","清除选中区域"), [this] { clearRegion(); });
  233. menu.addAction(tr("set area","设置区域"), [this] { setRegion(); });
  234. menu.exec(event->globalPos());
  235. }
  236. }
  237. QGraphicsView::mouseReleaseEvent(event);
  238. }
  239. void WaferGraphicsView::wheelEvent(QWheelEvent* event) {
  240. if (event->orientation() == Qt::Vertical) {
  241. event->ignore(); // 忽略竖直滚轮事件(即禁用滚动条滑动)
  242. return;
  243. }
  244. event->accept();
  245. }
  246. // 缩略图功能实现
  247. void WaferGraphicsView::showThumbnail() {
  248. ImageInfo image;
  249. m_pCViewInterface->GetViewMatrix()->GetWaferRefImage(image);
  250. QPixmap thumb = convertToPixmap(image);
  251. if (!thumb.isNull()) {
  252. // 如果图片加载成功,设置为缩略图
  253. thumbnailLabel->setPixmap(thumb.scaled(image.width, image.height, Qt::KeepAspectRatio));
  254. thumbnailLabel->show();
  255. thumbnailVisible = true;
  256. }
  257. else {
  258. // 如果加载图片失败,显示"图片加载失败"
  259. thumbnailLabel->setText("图片加载失败");
  260. thumbnailLabel->setAlignment(Qt::AlignCenter); // 居中显示文本
  261. thumbnailLabel->show();
  262. thumbnailVisible = true;
  263. }
  264. }
  265. void WaferGraphicsView::hideThumbnail() {
  266. thumbnailLabel->hide();
  267. thumbnailVisible = false;
  268. thumbnailLabel->move(0, 0);
  269. }
  270. void WaferGraphicsView::checkAndCreateRegion()
  271. {
  272. // 仅当两个点都有效时处理
  273. if (topLeftIndex.first < 0 || bottomRightIndex.first < 0) return;
  274. // 确定行列范围
  275. int startRow = qMin(topLeftIndex.first, bottomRightIndex.first);
  276. int endRow = qMax(topLeftIndex.first, bottomRightIndex.first);
  277. int startCol = qMin(topLeftIndex.second, bottomRightIndex.second);
  278. int endCol = qMax(topLeftIndex.second, bottomRightIndex.second);
  279. // 遍历场景中的所有项
  280. foreach(QGraphicsItem * item, scene()->items()) {
  281. if (DieItem* die = dynamic_cast<DieItem*>(item)) {
  282. int row = die->getRow();
  283. int col = die->getCol();
  284. // 判断是否在区域内
  285. if (row >= startRow && row <= endRow &&
  286. col >= startCol && col <= endCol) {
  287. // 更新选中状态
  288. die->setSelected(true);
  289. selectedItemsMap.insert(qMakePair(row, col), die);
  290. }
  291. }
  292. }
  293. // 重置索引点
  294. topLeftIndex = qMakePair(-1, -1);
  295. bottomRightIndex = qMakePair(-1, -1);
  296. }
  297. void WaferGraphicsView::clearRegion()
  298. {
  299. // 清空选中的 DieItem
  300. for (auto& item : selectedItemsMap) {
  301. DieItem* die = dynamic_cast<DieItem*>(item);
  302. if (die) {
  303. die->setSelected(false); // 取消选中状态
  304. }
  305. }
  306. selectedItemsMap.clear();
  307. if (selectedItem && selectedItem->scene()) {
  308. selectedItem->setLeftSelected(false);
  309. }
  310. selectedItem.clear();
  311. if (topLeftItem && topLeftItem->scene()) {
  312. topLeftItem->setRightSelected(false);
  313. }
  314. topLeftItem.clear();
  315. if (bottomRightItem && bottomRightItem->scene()) {
  316. bottomRightItem->setRightSelected(false);
  317. }
  318. bottomRightItem.clear();
  319. topLeftIndex = qMakePair(-1, -1);
  320. bottomRightIndex = qMakePair(-1, -1);
  321. // 清除缩略图
  322. hideThumbnail();
  323. }
  324. void WaferGraphicsView::setRegion()
  325. {
  326. for (auto it = selectedItemsMap.begin(); it != selectedItemsMap.end(); ++it) {
  327. QPair<int, int> key = it.key(); // 获取当前元素的 key
  328. qDebug() << "Row:" << key.first << ", Col:" << key.second;
  329. }
  330. // 清空选中的 DieItem
  331. for (auto& item : selectedItemsMap) {
  332. DieItem* die = dynamic_cast<DieItem*>(item);
  333. if (die) {
  334. die->setSelected(false); // 取消选中状态
  335. }
  336. }
  337. selectedItemsMap.clear();
  338. if (selectedItem && selectedItem->scene()) {
  339. selectedItem->setLeftSelected(false);
  340. }
  341. selectedItem.clear();
  342. if (topLeftItem && topLeftItem->scene()) {
  343. topLeftItem->setRightSelected(false);
  344. }
  345. topLeftItem.clear();
  346. if (bottomRightItem && bottomRightItem->scene()) {
  347. bottomRightItem->setRightSelected(false);
  348. }
  349. bottomRightItem.clear();
  350. topLeftIndex = qMakePair(-1, -1);
  351. bottomRightIndex = qMakePair(-1, -1);
  352. }
  353. void WaferGraphicsView::initWafer(const QPointF& center, double radius,
  354. int dieWidth, int dieHeight,double angle) {
  355. m_center = center;
  356. m_radius = radius;
  357. m_dieSize = QSize(dieWidth, dieHeight);
  358. // 清空场景
  359. m_scene->clear();
  360. qDebug() << "Center:" << m_center << "Radius:" << m_radius;
  361. // 绘制晶圆外圆
  362. QGraphicsEllipseItem* waferCircle = new QGraphicsEllipseItem(
  363. m_center.x() - radius,
  364. m_center.y() - radius,
  365. (radius) * 2,
  366. (radius) * 2
  367. );
  368. waferCircle->setPen(QPen(Qt::blue, 2));
  369. m_scene->addItem(waferCircle);
  370. double x, y;
  371. if (angle == 0.0) {
  372. x = m_center.x();
  373. y = m_center.y() + radius -5;
  374. }
  375. else if (angle == 90.0) {
  376. x = m_center.x() - radius;
  377. y = m_center.y();
  378. }
  379. else if (angle == 180.0) {
  380. x = m_center.x();
  381. y = m_center.y() - radius;
  382. }
  383. else if (angle == 270.0) {
  384. x = m_center.x() + radius -5;
  385. y = m_center.y();
  386. }
  387. QGraphicsRectItem* rect = m_scene->addRect(x, y, 5, 5,
  388. QPen(Qt::black, 0.5),QBrush(Qt::black));
  389. }
  390. void WaferGraphicsView::addDiePoint(const QPointF& diePos,const WAFER_MATRIX_POINT_INFO_STRUCT& dieInfo) {
  391. // 创建图形项
  392. DieItem* dieItem = new DieItem(dieInfo);
  393. //QPointF diePosUpdate(diePos.x() + dieInfo.nDieRow * m_dieSize.width(), diePos.y() + dieInfo.nDieCol * m_dieSize.height());
  394. dieItem->setRect(
  395. diePos.x() - m_dieSize.width() / 2,
  396. diePos.y() - m_dieSize.height() / 2,
  397. m_dieSize.width(),
  398. m_dieSize.height()
  399. );
  400. // 存储映射关系
  401. m_dieItems.insert(dieInfo.iDieIndex, dieItem);
  402. m_scene->addItem(dieItem);
  403. }
  404. void WaferGraphicsView::drawDieMatrix(const QPointF& center,
  405. int Width, int Height, int DieMatrixId) {
  406. QGraphicsRectItem* rect = m_scene->addRect(center.x(), center.y(), Width, Height,
  407. QPen(Qt::blue, 0.5));
  408. QGraphicsTextItem* text = new QGraphicsTextItem(QString(tr("DieMatrix%1")).arg(DieMatrixId));
  409. text->setPos(center.x(), center.y());
  410. QFont font = text->font(); // 获取当前字体
  411. font.setPointSize(6); // 设置字体大小为 8,可以根据需要调整
  412. text->setFont(font); // 应用新的字体
  413. m_scene->addItem(text);
  414. }
  415. void WaferGraphicsView::setCViewInterface(ns_module::CViewInterface* CViewInterface) {
  416. m_pCViewInterface = CViewInterface;
  417. }
  418. void WaferGraphicsView::yuv422_to_rgb888(const unsigned char* src, unsigned char* dst, int width, int height)
  419. {
  420. for (int i = 0; i < width * height; i += 2) {
  421. unsigned char y0 = src[0];
  422. unsigned char u = src[1];
  423. unsigned char y1 = src[2];
  424. unsigned char v = src[1];
  425. // 简单反色度插值,适用于 U/V 在相邻像素间共享的情况
  426. int r, g, b;
  427. // YUV to RGB 转换公式
  428. #define CLIP(x) qBound(0, int(x), 255)
  429. // Pixel 0
  430. r = CLIP(y0 + 1.402 * (v - 128));
  431. g = CLIP(y0 - 0.344 * (u - 128) - 0.714 * (v - 128));
  432. b = CLIP(y0 + 1.772 * (u - 128));
  433. *dst++ = r;
  434. *dst++ = g;
  435. *dst++ = b;
  436. // Pixel 1
  437. r = CLIP(y1 + 1.402 * (v - 128));
  438. g = CLIP(y1 - 0.344 * (u - 128) - 0.714 * (v - 128));
  439. b = CLIP(y1 + 1.772 * (u - 128));
  440. *dst++ = r;
  441. *dst++ = g;
  442. *dst++ = b;
  443. src += 4;
  444. }
  445. }
  446. QPixmap WaferGraphicsView::convertToPixmap(const ImageInfo& imgData)
  447. {
  448. QImage::Format qFormat = QImage::Format_Invalid;
  449. switch (imgData.format) {
  450. case ImageFormat::GRAY8:
  451. qFormat = QImage::Format_Grayscale8;
  452. break;
  453. case ImageFormat::RGB888:
  454. qFormat = QImage::Format_RGB888;
  455. break;
  456. case ImageFormat::ARGB32:
  457. qFormat = QImage::Format_ARGB32;
  458. break;
  459. case ImageFormat::RGB32:
  460. qFormat = QImage::Format_RGB32;
  461. break;
  462. case ImageFormat::YUV422: {
  463. // 需要先转换为 RGB888
  464. int byteCount = imgData.width * imgData.height * 3;
  465. unsigned char* rgbData = new unsigned char[byteCount];
  466. yuv422_to_rgb888(imgData.data, rgbData, imgData.width, imgData.height);
  467. QImage tmp(rgbData, imgData.width, imgData.height, QImage::Format_RGB888);
  468. QPixmap pixmap = QPixmap::fromImage(tmp);
  469. delete[] rgbData;
  470. return pixmap;
  471. }
  472. default:
  473. qDebug() << "Unsupported image format!";
  474. return QPixmap();
  475. }
  476. QImage qImg(imgData.data, imgData.width, imgData.height,
  477. imgData.width * imgData.channel, qFormat);
  478. return QPixmap::fromImage(qImg);
  479. }