2021-02-04 09:07:20 +00:00
|
|
|
/*
|
|
|
|
SPDX-FileCopyrightText: 2021 Vlad Zahorodnii <vlad.zahorodnii@kde.org>
|
|
|
|
|
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
|
|
|
|
2022-12-15 20:35:22 +00:00
|
|
|
#include "scene/shadowitem.h"
|
2023-08-31 12:23:42 +00:00
|
|
|
#include "compositor.h"
|
2023-01-17 18:22:28 +00:00
|
|
|
#include "scene/workspacescene.h"
|
2021-02-04 09:07:20 +00:00
|
|
|
#include "shadow.h"
|
2023-11-17 08:50:33 +00:00
|
|
|
#include "window.h"
|
2021-02-04 09:07:20 +00:00
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
2023-01-17 18:22:28 +00:00
|
|
|
ShadowTextureProvider::ShadowTextureProvider(Shadow *shadow)
|
|
|
|
: m_shadow(shadow)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
ShadowTextureProvider::~ShadowTextureProvider()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-12-18 22:20:28 +00:00
|
|
|
ShadowItem::ShadowItem(Shadow *shadow, Window *window, Scene *scene, Item *parent)
|
2022-12-18 12:30:40 +00:00
|
|
|
: Item(scene, parent)
|
2021-08-12 09:07:38 +00:00
|
|
|
, m_window(window)
|
2021-02-04 09:07:20 +00:00
|
|
|
, m_shadow(shadow)
|
2023-01-17 18:22:28 +00:00
|
|
|
, m_textureProvider(Compositor::self()->scene()->createShadowTextureProvider(shadow))
|
2021-02-04 09:07:20 +00:00
|
|
|
{
|
2021-06-10 10:02:10 +00:00
|
|
|
connect(shadow, &Shadow::offsetChanged, this, &ShadowItem::updateGeometry);
|
|
|
|
connect(shadow, &Shadow::rectChanged, this, &ShadowItem::updateGeometry);
|
2021-02-04 09:07:20 +00:00
|
|
|
connect(shadow, &Shadow::textureChanged, this, &ShadowItem::handleTextureChanged);
|
|
|
|
|
2021-06-10 10:02:10 +00:00
|
|
|
updateGeometry();
|
2021-02-04 09:07:20 +00:00
|
|
|
handleTextureChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
ShadowItem::~ShadowItem()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Shadow *ShadowItem::shadow() const
|
|
|
|
{
|
2021-09-03 10:24:17 +00:00
|
|
|
return m_shadow;
|
2021-02-04 09:07:20 +00:00
|
|
|
}
|
|
|
|
|
2023-01-17 18:22:28 +00:00
|
|
|
ShadowTextureProvider *ShadowItem::textureProvider() const
|
|
|
|
{
|
|
|
|
return m_textureProvider.get();
|
|
|
|
}
|
|
|
|
|
2021-06-10 10:02:10 +00:00
|
|
|
void ShadowItem::updateGeometry()
|
2021-02-04 09:07:20 +00:00
|
|
|
{
|
2022-05-16 20:13:39 +00:00
|
|
|
const QRectF rect = m_shadow->rect() + m_shadow->offset();
|
2021-02-04 09:07:20 +00:00
|
|
|
|
|
|
|
setPosition(rect.topLeft());
|
|
|
|
setSize(rect.size());
|
2021-05-21 10:51:23 +00:00
|
|
|
discardQuads();
|
2021-02-04 09:07:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ShadowItem::handleTextureChanged()
|
|
|
|
{
|
|
|
|
scheduleRepaint(rect());
|
2021-05-21 10:51:23 +00:00
|
|
|
discardQuads();
|
2023-01-17 18:22:28 +00:00
|
|
|
m_textureDirty = true;
|
2021-05-21 10:51:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void distributeHorizontally(QRectF &leftRect, QRectF &rightRect)
|
|
|
|
{
|
|
|
|
if (leftRect.right() > rightRect.left()) {
|
|
|
|
const qreal boundedRight = std::min(leftRect.right(), rightRect.right());
|
|
|
|
const qreal boundedLeft = std::max(leftRect.left(), rightRect.left());
|
|
|
|
const qreal halfOverlap = (boundedRight - boundedLeft) / 2.0;
|
|
|
|
leftRect.setRight(boundedRight - halfOverlap);
|
|
|
|
rightRect.setLeft(boundedLeft + halfOverlap);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void distributeVertically(QRectF &topRect, QRectF &bottomRect)
|
|
|
|
{
|
|
|
|
if (topRect.bottom() > bottomRect.top()) {
|
|
|
|
const qreal boundedBottom = std::min(topRect.bottom(), bottomRect.bottom());
|
|
|
|
const qreal boundedTop = std::max(topRect.top(), bottomRect.top());
|
|
|
|
const qreal halfOverlap = (boundedBottom - boundedTop) / 2.0;
|
|
|
|
topRect.setBottom(boundedBottom - halfOverlap);
|
|
|
|
bottomRect.setTop(boundedTop + halfOverlap);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WindowQuadList ShadowItem::buildQuads() const
|
|
|
|
{
|
|
|
|
// Do not draw shadows if window width or window height is less than 5 px. 5 is an arbitrary choice.
|
2021-08-12 09:07:38 +00:00
|
|
|
if (!m_window->wantsShadowToBeRendered() || m_window->width() < 5 || m_window->height() < 5) {
|
2021-05-21 10:51:23 +00:00
|
|
|
return WindowQuadList();
|
|
|
|
}
|
|
|
|
|
|
|
|
const QSizeF top(m_shadow->elementSize(Shadow::ShadowElementTop));
|
|
|
|
const QSizeF topRight(m_shadow->elementSize(Shadow::ShadowElementTopRight));
|
|
|
|
const QSizeF right(m_shadow->elementSize(Shadow::ShadowElementRight));
|
|
|
|
const QSizeF bottomRight(m_shadow->elementSize(Shadow::ShadowElementBottomRight));
|
|
|
|
const QSizeF bottom(m_shadow->elementSize(Shadow::ShadowElementBottom));
|
|
|
|
const QSizeF bottomLeft(m_shadow->elementSize(Shadow::ShadowElementBottomLeft));
|
|
|
|
const QSizeF left(m_shadow->elementSize(Shadow::ShadowElementLeft));
|
|
|
|
const QSizeF topLeft(m_shadow->elementSize(Shadow::ShadowElementTopLeft));
|
|
|
|
|
|
|
|
const QMarginsF shadowMargins(
|
2022-03-23 10:13:38 +00:00
|
|
|
std::max({topLeft.width(), left.width(), bottomLeft.width()}),
|
|
|
|
std::max({topLeft.height(), top.height(), topRight.height()}),
|
|
|
|
std::max({topRight.width(), right.width(), bottomRight.width()}),
|
|
|
|
std::max({bottomRight.height(), bottom.height(), bottomLeft.height()}));
|
2021-05-21 10:51:23 +00:00
|
|
|
|
2021-06-10 10:02:10 +00:00
|
|
|
const QRectF outerRect = rect();
|
2021-05-21 10:51:23 +00:00
|
|
|
|
|
|
|
const int width = shadowMargins.left() + std::max(top.width(), bottom.width()) + shadowMargins.right();
|
|
|
|
const int height = shadowMargins.top() + std::max(left.height(), right.height()) + shadowMargins.bottom();
|
|
|
|
|
|
|
|
QRectF topLeftRect;
|
|
|
|
if (!topLeft.isEmpty()) {
|
|
|
|
topLeftRect = QRectF(outerRect.topLeft(), topLeft);
|
|
|
|
} else {
|
|
|
|
topLeftRect = QRectF(outerRect.left() + shadowMargins.left(),
|
|
|
|
outerRect.top() + shadowMargins.top(), 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
QRectF topRightRect;
|
|
|
|
if (!topRight.isEmpty()) {
|
|
|
|
topRightRect = QRectF(outerRect.right() - topRight.width(), outerRect.top(),
|
|
|
|
topRight.width(), topRight.height());
|
|
|
|
} else {
|
|
|
|
topRightRect = QRectF(outerRect.right() - shadowMargins.right(),
|
|
|
|
outerRect.top() + shadowMargins.top(), 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
QRectF bottomRightRect;
|
|
|
|
if (!bottomRight.isEmpty()) {
|
|
|
|
bottomRightRect = QRectF(outerRect.right() - bottomRight.width(),
|
|
|
|
outerRect.bottom() - bottomRight.height(),
|
|
|
|
bottomRight.width(), bottomRight.height());
|
|
|
|
} else {
|
|
|
|
bottomRightRect = QRectF(outerRect.right() - shadowMargins.right(),
|
|
|
|
outerRect.bottom() - shadowMargins.bottom(), 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
QRectF bottomLeftRect;
|
|
|
|
if (!bottomLeft.isEmpty()) {
|
|
|
|
bottomLeftRect = QRectF(outerRect.left(), outerRect.bottom() - bottomLeft.height(),
|
|
|
|
bottomLeft.width(), bottomLeft.height());
|
|
|
|
} else {
|
|
|
|
bottomLeftRect = QRectF(outerRect.left() + shadowMargins.left(),
|
|
|
|
outerRect.bottom() - shadowMargins.bottom(), 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Re-distribute the corner tiles so no one of them is overlapping with others.
|
|
|
|
// By doing this, we assume that shadow's corner tiles are symmetric
|
|
|
|
// and it is OK to not draw top/right/bottom/left tile between corners.
|
|
|
|
// For example, let's say top-left and top-right tiles are overlapping.
|
|
|
|
// In that case, the right side of the top-left tile will be shifted to left,
|
|
|
|
// the left side of the top-right tile will shifted to right, and the top
|
|
|
|
// tile won't be rendered.
|
|
|
|
distributeHorizontally(topLeftRect, topRightRect);
|
|
|
|
distributeHorizontally(bottomLeftRect, bottomRightRect);
|
|
|
|
distributeVertically(topLeftRect, bottomLeftRect);
|
|
|
|
distributeVertically(topRightRect, bottomRightRect);
|
|
|
|
|
|
|
|
qreal tx1 = 0.0,
|
|
|
|
tx2 = 0.0,
|
|
|
|
ty1 = 0.0,
|
|
|
|
ty2 = 0.0;
|
|
|
|
|
|
|
|
WindowQuadList quads;
|
|
|
|
quads.reserve(8);
|
|
|
|
|
|
|
|
if (topLeftRect.isValid()) {
|
|
|
|
tx1 = 0.0;
|
|
|
|
ty1 = 0.0;
|
2021-07-06 13:38:35 +00:00
|
|
|
tx2 = topLeftRect.width();
|
|
|
|
ty2 = topLeftRect.height();
|
2021-06-10 10:32:37 +00:00
|
|
|
WindowQuad topLeftQuad;
|
2022-03-23 10:13:38 +00:00
|
|
|
topLeftQuad[0] = WindowVertex(topLeftRect.left(), topLeftRect.top(), tx1, ty1);
|
|
|
|
topLeftQuad[1] = WindowVertex(topLeftRect.right(), topLeftRect.top(), tx2, ty1);
|
2021-05-21 10:51:23 +00:00
|
|
|
topLeftQuad[2] = WindowVertex(topLeftRect.right(), topLeftRect.bottom(), tx2, ty2);
|
2022-03-23 10:13:38 +00:00
|
|
|
topLeftQuad[3] = WindowVertex(topLeftRect.left(), topLeftRect.bottom(), tx1, ty2);
|
2021-05-21 10:51:23 +00:00
|
|
|
quads.append(topLeftQuad);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (topRightRect.isValid()) {
|
2021-07-06 13:38:35 +00:00
|
|
|
tx1 = width - topRightRect.width();
|
2021-05-21 10:51:23 +00:00
|
|
|
ty1 = 0.0;
|
2021-07-06 13:38:35 +00:00
|
|
|
tx2 = width;
|
|
|
|
ty2 = topRightRect.height();
|
2021-06-10 10:32:37 +00:00
|
|
|
WindowQuad topRightQuad;
|
2022-03-23 10:13:38 +00:00
|
|
|
topRightQuad[0] = WindowVertex(topRightRect.left(), topRightRect.top(), tx1, ty1);
|
|
|
|
topRightQuad[1] = WindowVertex(topRightRect.right(), topRightRect.top(), tx2, ty1);
|
2021-05-21 10:51:23 +00:00
|
|
|
topRightQuad[2] = WindowVertex(topRightRect.right(), topRightRect.bottom(), tx2, ty2);
|
2022-03-23 10:13:38 +00:00
|
|
|
topRightQuad[3] = WindowVertex(topRightRect.left(), topRightRect.bottom(), tx1, ty2);
|
2021-05-21 10:51:23 +00:00
|
|
|
quads.append(topRightQuad);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bottomRightRect.isValid()) {
|
2021-07-06 13:38:35 +00:00
|
|
|
tx1 = width - bottomRightRect.width();
|
|
|
|
tx2 = width;
|
|
|
|
ty1 = height - bottomRightRect.height();
|
|
|
|
ty2 = height;
|
2021-06-10 10:32:37 +00:00
|
|
|
WindowQuad bottomRightQuad;
|
2022-03-23 10:13:38 +00:00
|
|
|
bottomRightQuad[0] = WindowVertex(bottomRightRect.left(), bottomRightRect.top(), tx1, ty1);
|
|
|
|
bottomRightQuad[1] = WindowVertex(bottomRightRect.right(), bottomRightRect.top(), tx2, ty1);
|
2021-05-21 10:51:23 +00:00
|
|
|
bottomRightQuad[2] = WindowVertex(bottomRightRect.right(), bottomRightRect.bottom(), tx2, ty2);
|
2022-03-23 10:13:38 +00:00
|
|
|
bottomRightQuad[3] = WindowVertex(bottomRightRect.left(), bottomRightRect.bottom(), tx1, ty2);
|
2021-05-21 10:51:23 +00:00
|
|
|
quads.append(bottomRightQuad);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bottomLeftRect.isValid()) {
|
|
|
|
tx1 = 0.0;
|
2021-07-06 13:38:35 +00:00
|
|
|
tx2 = bottomLeftRect.width();
|
|
|
|
ty1 = height - bottomLeftRect.height();
|
|
|
|
ty2 = height;
|
2021-06-10 10:32:37 +00:00
|
|
|
WindowQuad bottomLeftQuad;
|
2022-03-23 10:13:38 +00:00
|
|
|
bottomLeftQuad[0] = WindowVertex(bottomLeftRect.left(), bottomLeftRect.top(), tx1, ty1);
|
|
|
|
bottomLeftQuad[1] = WindowVertex(bottomLeftRect.right(), bottomLeftRect.top(), tx2, ty1);
|
2021-05-21 10:51:23 +00:00
|
|
|
bottomLeftQuad[2] = WindowVertex(bottomLeftRect.right(), bottomLeftRect.bottom(), tx2, ty2);
|
2022-03-23 10:13:38 +00:00
|
|
|
bottomLeftQuad[3] = WindowVertex(bottomLeftRect.left(), bottomLeftRect.bottom(), tx1, ty2);
|
2021-05-21 10:51:23 +00:00
|
|
|
quads.append(bottomLeftQuad);
|
|
|
|
}
|
|
|
|
|
|
|
|
QRectF topRect(QPointF(topLeftRect.right(), outerRect.top()),
|
|
|
|
QPointF(topRightRect.left(), outerRect.top() + top.height()));
|
|
|
|
|
|
|
|
QRectF rightRect(QPointF(outerRect.right() - right.width(), topRightRect.bottom()),
|
|
|
|
QPointF(outerRect.right(), bottomRightRect.top()));
|
|
|
|
|
|
|
|
QRectF bottomRect(QPointF(bottomLeftRect.right(), outerRect.bottom() - bottom.height()),
|
|
|
|
QPointF(bottomRightRect.left(), outerRect.bottom()));
|
|
|
|
|
|
|
|
QRectF leftRect(QPointF(outerRect.left(), topLeftRect.bottom()),
|
|
|
|
QPointF(outerRect.left() + left.width(), bottomLeftRect.top()));
|
|
|
|
|
|
|
|
// Re-distribute left/right and top/bottom shadow tiles so they don't
|
|
|
|
// overlap when the window is too small. Please notice that we don't
|
|
|
|
// fix overlaps between left/top(left/bottom, right/top, and so on)
|
|
|
|
// corner tiles because corresponding counter parts won't be valid when
|
|
|
|
// the window is too small, which means they won't be rendered.
|
|
|
|
distributeHorizontally(leftRect, rightRect);
|
|
|
|
distributeVertically(topRect, bottomRect);
|
|
|
|
|
|
|
|
if (topRect.isValid()) {
|
2021-07-06 13:38:35 +00:00
|
|
|
tx1 = shadowMargins.left();
|
2021-05-21 10:51:23 +00:00
|
|
|
ty1 = 0.0;
|
2021-07-06 13:38:35 +00:00
|
|
|
tx2 = tx1 + top.width();
|
|
|
|
ty2 = topRect.height();
|
2021-06-10 10:32:37 +00:00
|
|
|
WindowQuad topQuad;
|
2022-03-23 10:13:38 +00:00
|
|
|
topQuad[0] = WindowVertex(topRect.left(), topRect.top(), tx1, ty1);
|
|
|
|
topQuad[1] = WindowVertex(topRect.right(), topRect.top(), tx2, ty1);
|
2021-05-21 10:51:23 +00:00
|
|
|
topQuad[2] = WindowVertex(topRect.right(), topRect.bottom(), tx2, ty2);
|
2022-03-23 10:13:38 +00:00
|
|
|
topQuad[3] = WindowVertex(topRect.left(), topRect.bottom(), tx1, ty2);
|
2021-05-21 10:51:23 +00:00
|
|
|
quads.append(topQuad);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rightRect.isValid()) {
|
2021-07-06 13:38:35 +00:00
|
|
|
tx1 = width - rightRect.width();
|
|
|
|
ty1 = shadowMargins.top();
|
|
|
|
tx2 = width;
|
|
|
|
ty2 = ty1 + right.height();
|
2021-06-10 10:32:37 +00:00
|
|
|
WindowQuad rightQuad;
|
2022-03-23 10:13:38 +00:00
|
|
|
rightQuad[0] = WindowVertex(rightRect.left(), rightRect.top(), tx1, ty1);
|
|
|
|
rightQuad[1] = WindowVertex(rightRect.right(), rightRect.top(), tx2, ty1);
|
2021-05-21 10:51:23 +00:00
|
|
|
rightQuad[2] = WindowVertex(rightRect.right(), rightRect.bottom(), tx2, ty2);
|
2022-03-23 10:13:38 +00:00
|
|
|
rightQuad[3] = WindowVertex(rightRect.left(), rightRect.bottom(), tx1, ty2);
|
2021-05-21 10:51:23 +00:00
|
|
|
quads.append(rightQuad);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bottomRect.isValid()) {
|
2021-07-06 13:38:35 +00:00
|
|
|
tx1 = shadowMargins.left();
|
|
|
|
ty1 = height - bottomRect.height();
|
|
|
|
tx2 = tx1 + bottom.width();
|
|
|
|
ty2 = height;
|
2021-06-10 10:32:37 +00:00
|
|
|
WindowQuad bottomQuad;
|
2022-03-23 10:13:38 +00:00
|
|
|
bottomQuad[0] = WindowVertex(bottomRect.left(), bottomRect.top(), tx1, ty1);
|
|
|
|
bottomQuad[1] = WindowVertex(bottomRect.right(), bottomRect.top(), tx2, ty1);
|
2021-05-21 10:51:23 +00:00
|
|
|
bottomQuad[2] = WindowVertex(bottomRect.right(), bottomRect.bottom(), tx2, ty2);
|
2022-03-23 10:13:38 +00:00
|
|
|
bottomQuad[3] = WindowVertex(bottomRect.left(), bottomRect.bottom(), tx1, ty2);
|
2021-05-21 10:51:23 +00:00
|
|
|
quads.append(bottomQuad);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (leftRect.isValid()) {
|
|
|
|
tx1 = 0.0;
|
2021-07-06 13:38:35 +00:00
|
|
|
ty1 = shadowMargins.top();
|
|
|
|
tx2 = leftRect.width();
|
|
|
|
ty2 = ty1 + left.height();
|
2021-06-10 10:32:37 +00:00
|
|
|
WindowQuad leftQuad;
|
2022-03-23 10:13:38 +00:00
|
|
|
leftQuad[0] = WindowVertex(leftRect.left(), leftRect.top(), tx1, ty1);
|
|
|
|
leftQuad[1] = WindowVertex(leftRect.right(), leftRect.top(), tx2, ty1);
|
2021-05-21 10:51:23 +00:00
|
|
|
leftQuad[2] = WindowVertex(leftRect.right(), leftRect.bottom(), tx2, ty2);
|
2022-03-23 10:13:38 +00:00
|
|
|
leftQuad[3] = WindowVertex(leftRect.left(), leftRect.bottom(), tx1, ty2);
|
2021-05-21 10:51:23 +00:00
|
|
|
quads.append(leftQuad);
|
|
|
|
}
|
|
|
|
|
|
|
|
return quads;
|
2021-02-04 09:07:20 +00:00
|
|
|
}
|
|
|
|
|
2023-01-17 18:22:28 +00:00
|
|
|
void ShadowItem::preprocess()
|
|
|
|
{
|
|
|
|
if (m_textureDirty) {
|
|
|
|
m_textureDirty = false;
|
|
|
|
m_textureProvider->update();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-04 09:07:20 +00:00
|
|
|
} // namespace KWin
|
2023-07-05 06:30:14 +00:00
|
|
|
|
|
|
|
#include "moc_shadowitem.cpp"
|