48 const QRect br = region.boundingRect();
49 if (QWindowsContext::verbose > 1)
50 qCDebug(lcQpaBackingStore) <<
__FUNCTION__ <<
this << window << offset << br;
51 QWindowsWindow *rw = QWindowsWindow::windowsWindowOf(window);
54 const bool hasAlpha = rw->format().hasAlpha();
55 const Qt::WindowFlags flags = window->flags();
57 if (rw->isLayered() && hasAlpha && QWindowsWindow::hasNoNativeFrame(rw->handle(), flags)) {
59 QRect r = QHighDpi::toNativePixels(window->frameGeometry(), window);
60 QMargins frameMargins = rw->frameMargins();
61 QRect dirtyRect = br.translated(offset + QPoint(frameMargins.left(), frameMargins.top()));
63 SIZE size = {r.width(), r.height()};
64 POINT ptDst = {r.x(), r.y()};
66 BLENDFUNCTION blend = {AC_SRC_OVER, 0, BYTE(qRound(255.0 * rw->opacity())), AC_SRC_ALPHA};
67 RECT dirty = {dirtyRect.x(), dirtyRect.y(),
68 dirtyRect.x() + dirtyRect.width(), dirtyRect.y() + dirtyRect.height()};
69 UPDATELAYEREDWINDOWINFO info = {
sizeof(info),
nullptr, &ptDst, &size,
70 m_image->hdc(), &ptSrc, 0, &blend, ULW_ALPHA, &dirty};
71 const BOOL result = UpdateLayeredWindowIndirect(rw->handle(), &info);
73 qErrnoWarning(
"UpdateLayeredWindowIndirect failed for ptDst=(%d, %d),"
74 " size=(%dx%d), dirty=(%dx%d %d, %d)", r.x(), r.y(),
75 r.width(), r.height(), dirtyRect.width(), dirtyRect.height(),
76 dirtyRect.x(), dirtyRect.y());
78 const HDC dc = rw->getDC();
80 qErrnoWarning(
"%s: GetDC failed",
__FUNCTION__);
84 if (!BitBlt(dc, br.x(), br.y(), br.width(), br.height(),
85 m_image->hdc(), br.x() + offset.x(), br.y() + offset.y(), SRCCOPY)) {
86 const DWORD lastError = GetLastError();
87 if (lastError != ERROR_SUCCESS && lastError != ERROR_INVALID_HANDLE)
88 qErrnoWarning(
int(lastError),
"%s: BitBlt failed",
__FUNCTION__);
94 if (QWindowsContext::verbose > 2 && lcQpaBackingStore().isDebugEnabled()) {
96 const QString fileName = QString::fromLatin1(
"win%1_%2.png").
97 arg(rw->winId()).arg(n++);
98 m_image->image().save(fileName);
99 qCDebug(lcQpaBackingStore) <<
"Wrote " << m_image->image().size() << fileName;
105 if (m_image.isNull() || m_image->image().size() != size) {
106#ifndef QT_NO_DEBUG_OUTPUT
108 qCDebug(lcQpaBackingStore)
109 <<
__FUNCTION__ <<
' ' << window() <<
' ' << size <<
' ' << region
110 <<
" from: " << (m_image.isNull() ? QSize() : m_image->image().size());
113 QImage::Format format = window()->format().hasAlpha() ?
114 QImage::Format_ARGB32_Premultiplied : QWindowsNativeImage::systemFormat();
119 if (QImage::toPixelFormat(format).alphaUsage() == QPixelFormat::UsesAlpha)
120 m_alphaNeedsFill =
true;
122 format = qt_maybeDataCompatibleAlphaVersion(format);
124 QWindowsNativeImage *oldwni = m_image.data();
125 auto *newwni =
new QWindowsNativeImage(size.width(), size.height(), format);
127 if (oldwni && !region.isEmpty()) {
128 const QImage &oldimg(oldwni->image());
129 QImage &newimg(newwni->image());
130 QRegion staticRegion(region);
131 staticRegion &= QRect(0, 0, oldimg.width(), oldimg.height());
132 staticRegion &= QRect(0, 0, newimg.width(), newimg.height());
133 QPainter painter(&newimg);
134 painter.setCompositionMode(QPainter::CompositionMode_Source);
135 for (
const QRect &rect : staticRegion)
136 painter.drawImage(rect, oldimg, rect);
139 m_image.reset(newwni);
Singleton container for all relevant information.