我可以在不使用光栅绘画的情况下将一个QImage添加到另一个QImage吗?

时间:2016-03-14 13:58:01

标签: qt qpainter qimage

我的项目要求是我可以创建宽度高达36000像素(高度小得多)的图像 (图像来自QGraphicsScene)。

我遇到了限制:QPainter限制了光栅绘制的设备大小:

void QRasterPaintEnginePrivate::systemStateChanged()
{
    deviceRectUnclipped = QRect(0, 0,
            qMin(QT_RASTER_COORD_LIMIT, device->width()),
            qMin(QT_RASTER_COORD_LIMIT, device->height()));
    ....
}

// This limitations comes from qgrayraster.c. Any higher and
// rasterization of shapes will produce incorrect results.
const int QT_RASTER_COORD_LIMIT = 32767;

(我尝试排查问题...... Rendering a large QGraphicsScene on a QImage clips it off

所以...我想,我可以创建2张图片然后添加它们吗? (一个在彼此的末尾)

if(wOutput > 32767)
{
    QImage image1 = QImage(32767, hOutput, QImage::Format_Mono);
    image1.fill(QColor(Qt::white).rgb());
    QRectF source(0, 0, 32767, hOutput);
    QRectF target(0, 0, 32767, hOutput);
    QPainter painter;
    painter.begin(&image1);
    outputScene->render(&painter, target, source);
    painter.end();

    QImage image2 = QImage(wOutput - 32767, hOutput, QImage::Format_Mono);
    image2.fill(QColor(Qt::white).rgb());
    source = QRectF(32767, 0, wOutput - 32767, hOutput);
    target = QRectF(0, 0, wOutput - 32767, hOutput);    
    painter.begin(&image2);
    outputScene->render(&painter, target, source);
    painter.end();

    // now create a combination, add image2 at the end of image1
    QImage image = QImage(wOutput, hOutput, QImage::Format_Mono);
    painter.begin(&image);
    painter.drawImage(0, 0, image1);
    painter.drawImage(32767, hOutput, image2);
    painter.end();
}
else
{
    // just create the image
}

看起来很合乎逻辑......但输出没有显示image2。显然......我正在使用相同的光栅画......具有相同的限制!

我还可以在另一端添加图像? (注意 - 我的"大"大小是"宽度"所以我甚至不认为我可以使用scanline来更快地复制像素。

1 个答案:

答案 0 :(得分:1)

您可以使用QImage::scanLine获取像素数据并进行复制。 然而,QImage::Format_Mono使它变得更复杂,因为您必须考虑像素数据的对齐(QImage::Format_Mono每像素有1位,因此每个像素有8个像素。)

我建议使用可被8整除的宽度生成第一个图像(例如32760),这样就可以复制第二个图像的行而不会移动这些位。

两张源图像中的颜色表也应该相同。

您可以这样做:

int w1 = 32760;
QImage image1 = QImage(w1, hOutput, QImage::Format_Mono);
//grab the first  image....
//....
int w2 = wOutput - w1;
QImage image2 = QImage(w2, hOutput, QImage::Format_Mono);
//grab the second image....
//....

int bytesPerLine1 = w1 / 8; //is divisible by 8
int bytesPerLine2 = ceil(float(w2) / 8.0f); //should be right :)
QImage image = QImage(wOutput, hOutput, QImage::Format_Mono);
image.setColorTable(image1.colorTable());
for(int i = 0; i < hOutput; ++i)
{
    uchar* dstSL = image.scanLine(i);
    uchar* src1SL = image1.scanLine(i);
    memcpy(dstSL, src1SL, bytesPerLine1);
    uchar* src2SL = image2.scanLine(i);
    memcpy(&dstSL[bytesPerLine1], src2SL, bytesPerLine2);
}

我建议您阅读QImage文档:Pixel ManipulationQImage::Format