c++ - Qt 旋转图像 -> 多边形

标签 c++ qt qimage qpolygon

我有一个 QImage,中心在 (x,y),大小 (w,h)。我应用旋转并最终使用 QTransform 进行缩放。现在我很容易将新图像(旋转/缩放)放在 (x,y) 的中心。

QTransform transform = QTransform().translate(0,0).rotateRadians(r).scale(s,s);
QImage image = new QImage(old_image->transformed(transform);

我想要一个 QPolygon 在旋转/缩放图像的角上有 4 个点。我所有的尝试都失败了。

QPolygon p = QPolygon(QRect(x-w/2, y-h/2, w, h));
QPolygon p2 = transform.map(p);

QPolygon p2 = transform.mapToPolygon(QRect(x-w/2, y-h/2, w, h));

我得到的只是一个位置错误的多边形。哪个必须是正确的代码?

最佳答案

与图像相反,当我们想要围绕其中心旋转多边形(在本例中为矩形)时,我们必须使用像这样的变换

T = translate(center).rotate(r).translate(-center)

(记住转换是从右到左应用的)。暂时放弃缩放。对于图像,这种初始翻译是隐式的。

这种情况下的问题是图像在旋转后改变它的尺寸:新图像的尺寸是包含旋转像素的边界矩形,因此最后的平移不再对你的案例(translate(center))。

相反,你需要这样的东西:

T = translate(new_rect.size() / 2).rotate(r).translate(-original_center)

为此,您必须将转换分成两部分(缩放也带回):

T1 = rotate(r).scale(s, s).translate(-center)
T2 = translate(new_rect.size() / 2)

因为新的矩形大小只能在应用第一次转换后计算。

此外,对于 T2,我不是在谈论中心,而是在谈论一半大小,因为新的中心不会表达与图像相同的含义多边形(例如,图像没有负像素坐标)

使用Qt API表达:

// Image transformation
auto transform = QTransform().rotateRadians(r).scale(s, s);
auto image = m_image.transformed(transform, Qt::SmoothTransformation);

// Assuming something like: QPainter painter(this);
painter.drawImage(0, 0, image);

// Rect to polygon
const auto rect = m_image.rect();
QPolygonF pol;
pol << rect.topLeft();
pol << rect.topRight();
pol << rect.bottomRight();
pol << rect.bottomLeft();

// First transformation
const auto center = rect.center();
transform = transform.translate(-center.x(), -center.y());
const auto pol2 = transform.map(pol);

// Second transformation
const auto rect2 = pol2.boundingRect();
painter.drawPolygon(QTransform().translate(rect2.width() / 2, rect2.height() / 2).map(pol2));

完整代码在 GitHub 中可用.

关于c++ - Qt 旋转图像 -> 多边形,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39194186/

相关文章:

c++ - 将 OpenSSL 静态库添加到 vc++ 项目

c++ - 未启动事件循环时不优雅/杀死 Qt 应用程序

qt - 原始数据到 QImage

qt - 程序可以在较新版本的 Qt 上运行吗?

qt - 需要绘制包含使用 Qt 绘制圆的透明 qimage

c++ - QImage 调整大小事件

c++ - OpenCV:detectMultiScale() 从对象中给出了太多点

c++ - 如何安全地将 uint32_t 中的带符号字段提取为带符号数字(int 或 uint32_t)

c++ - 使用 QTimer 为 5 张图片的幻灯片放映计时

qt - 是否可以从 Qt5 QML 中的信号断开所有插槽?