2017-01-31 21:25:00 +02:00
/*
* Copyright 2011 by Aaron Seigo < aseigo @ kde . org >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU Library General Public License version 2 ,
* or ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details
*
* You should have received a copy of the GNU Library General Public
* License along with this program ; if not , write to the
* Free Software Foundation , Inc . ,
* 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 , USA .
*/
# include "panelshadows_p.h"
# include <QWindow>
# include <QPainter>
2019-12-29 11:55:14 +02:00
# include <config-latte.h>
# include <KWindowSystem>
# if HAVE_X11
# include <QX11Info>
# include <X11/Xatom.h>
# include <X11/Xlib.h>
# include <X11/Xlib-xcb.h>
# include <fixx11h.h>
# endif
2018-12-02 02:05:52 +02:00
# include <KWayland/Client/connection_thread.h>
# include <KWayland/Client/registry.h>
# include <KWayland/Client/shadow.h>
# include <KWayland/Client/shm_pool.h>
# include <KWayland/Client/surface.h>
2019-12-29 11:55:14 +02:00
# include <qdebug.h>
2017-01-31 21:25:00 +02:00
2018-07-03 22:15:45 +03:00
class PanelShadows : : Private
{
2017-01-31 21:25:00 +02:00
public :
Private ( PanelShadows * shadows )
: q ( shadows )
# if HAVE_X11
2019-12-29 11:55:14 +02:00
, _connection ( nullptr ) ,
_gc ( 0x0 )
2017-01-31 21:25:00 +02:00
, m_isX11 ( KWindowSystem : : isPlatformX11 ( ) )
# endif
{
setupWaylandIntegration ( ) ;
}
2019-12-29 11:55:14 +02:00
~ Private ( )
{
2017-01-31 21:25:00 +02:00
// Do not call clearPixmaps() from here: it creates new QPixmap(),
// which causes a crash when application is stopping.
freeX11Pixmaps ( ) ;
}
void freeX11Pixmaps ( ) ;
void freeWaylandBuffers ( ) ;
void clearPixmaps ( ) ;
void setupPixmaps ( ) ;
2019-12-29 11:55:14 +02:00
Qt : : HANDLE createPixmap ( const QPixmap & source ) ;
2017-01-31 21:25:00 +02:00
void initPixmap ( const QString & element ) ;
QPixmap initEmptyPixmap ( const QSize & size ) ;
void updateShadow ( const QWindow * window , Plasma : : FrameSvg : : EnabledBorders ) ;
void updateShadowX11 ( const QWindow * window , Plasma : : FrameSvg : : EnabledBorders ) ;
void updateShadowWayland ( const QWindow * window , Plasma : : FrameSvg : : EnabledBorders ) ;
void clearShadow ( const QWindow * window ) ;
void clearShadowX11 ( const QWindow * window ) ;
void clearShadowWayland ( const QWindow * window ) ;
void updateShadows ( ) ;
void setupData ( Plasma : : FrameSvg : : EnabledBorders enabledBorders ) ;
2019-12-29 11:55:14 +02:00
bool hasShadows ( ) const ;
2017-01-31 21:25:00 +02:00
void setupWaylandIntegration ( ) ;
PanelShadows * q ;
QList < QPixmap > m_shadowPixmaps ;
QPixmap m_emptyCornerPix ;
QPixmap m_emptyCornerLeftPix ;
QPixmap m_emptyCornerTopPix ;
QPixmap m_emptyCornerRightPix ;
QPixmap m_emptyCornerBottomPix ;
QPixmap m_emptyVerticalPix ;
QPixmap m_emptyHorizontalPix ;
# if HAVE_X11
//! xcb connection
2019-12-29 11:55:14 +02:00
xcb_connection_t * _connection ;
2017-01-31 21:25:00 +02:00
//! graphical context
xcb_gcontext_t _gc ;
bool m_isX11 ;
# endif
2019-12-29 11:55:14 +02:00
struct Wayland {
2017-01-31 21:25:00 +02:00
KWayland : : Client : : ShadowManager * manager = nullptr ;
KWayland : : Client : : ShmPool * shmPool = nullptr ;
QList < KWayland : : Client : : Buffer : : Ptr > shadowBuffers ;
} ;
Wayland m_wayland ;
2019-12-29 11:55:14 +02:00
QHash < Plasma : : FrameSvg : : EnabledBorders , QVector < unsigned long > > data ;
2017-01-31 21:25:00 +02:00
QHash < const QWindow * , Plasma : : FrameSvg : : EnabledBorders > m_windows ;
} ;
2018-07-03 22:15:45 +03:00
class PanelShadowsSingleton
{
2017-01-31 21:25:00 +02:00
public :
2019-12-29 11:55:14 +02:00
PanelShadowsSingleton ( )
{
2017-01-31 21:25:00 +02:00
}
2019-12-29 11:55:14 +02:00
PanelShadows self ;
2017-01-31 21:25:00 +02:00
} ;
Q_GLOBAL_STATIC ( PanelShadowsSingleton , privatePanelShadowsSelf )
PanelShadows : : PanelShadows ( QObject * parent , const QString & prefix )
: Plasma : : Svg ( parent ) ,
d ( new Private ( this ) )
{
setImagePath ( prefix ) ;
2019-12-29 11:55:14 +02:00
connect ( this , & Plasma : : Svg : : repaintNeeded , this , [ this ] ( ) {
d - > updateShadows ( ) ;
} ) ;
2017-01-31 21:25:00 +02:00
}
PanelShadows : : ~ PanelShadows ( )
{
delete d ;
}
PanelShadows * PanelShadows : : self ( )
{
return & privatePanelShadowsSelf - > self ;
}
void PanelShadows : : addWindow ( const QWindow * window , Plasma : : FrameSvg : : EnabledBorders enabledBorders )
{
if ( ! window ) {
return ;
}
d - > m_windows [ window ] = enabledBorders ;
d - > updateShadow ( window , enabledBorders ) ;
2019-12-29 11:55:14 +02:00
connect ( window , & QObject : : destroyed , this , [ this , window ] ( ) {
d - > m_windows . remove ( window ) ;
if ( d - > m_windows . isEmpty ( ) ) {
d - > clearPixmaps ( ) ;
}
} ) ;
2017-01-31 21:25:00 +02:00
}
void PanelShadows : : removeWindow ( const QWindow * window )
{
if ( ! d - > m_windows . contains ( window ) ) {
return ;
}
d - > m_windows . remove ( window ) ;
2018-08-30 13:40:34 +03:00
disconnect ( window , nullptr , this , nullptr ) ;
2017-01-31 21:25:00 +02:00
d - > clearShadow ( window ) ;
if ( d - > m_windows . isEmpty ( ) ) {
d - > clearPixmaps ( ) ;
}
}
2019-12-29 11:55:14 +02:00
bool PanelShadows : : hasShadows ( ) const
{
return hasElement ( QStringLiteral ( " shadow-left " ) ) ;
}
2017-01-31 21:25:00 +02:00
void PanelShadows : : setEnabledBorders ( const QWindow * window , Plasma : : FrameSvg : : EnabledBorders enabledBorders )
{
if ( ! window | | ! d - > m_windows . contains ( window ) ) {
return ;
}
2017-11-24 22:34:02 +02:00
d - > m_windows [ window ] = enabledBorders ;
2017-01-31 21:25:00 +02:00
d - > updateShadow ( window , enabledBorders ) ;
}
void PanelShadows : : Private : : updateShadows ( )
{
2019-12-29 11:55:14 +02:00
const bool hadShadowsBefore = ! m_shadowPixmaps . isEmpty ( ) ;
2017-01-31 21:25:00 +02:00
2019-12-29 11:55:14 +02:00
// has shadows now?
if ( hasShadows ( ) ) {
if ( hadShadowsBefore ) {
clearPixmaps ( ) ;
}
for ( auto i = m_windows . constBegin ( ) ; i ! = m_windows . constEnd ( ) ; + + i ) {
updateShadow ( i . key ( ) , i . value ( ) ) ;
}
} else {
if ( hadShadowsBefore ) {
for ( auto i = m_windows . constBegin ( ) ; i ! = m_windows . constEnd ( ) ; + + i ) {
clearShadow ( i . key ( ) ) ;
}
clearPixmaps ( ) ;
}
2017-01-31 21:25:00 +02:00
}
}
2019-12-29 11:55:14 +02:00
Qt : : HANDLE PanelShadows : : Private : : createPixmap ( const QPixmap & source )
2017-01-31 21:25:00 +02:00
{
// do nothing for invalid pixmaps
2019-12-29 11:55:14 +02:00
if ( source . isNull ( ) ) return nullptr ;
2017-01-31 21:25:00 +02:00
/*
in some cases , pixmap handle is invalid . This is the case notably
when Qt uses to RasterEngine . In this case , we create an X11 Pixmap
explicitly and draw the source pixmap on it .
*/
2019-12-29 11:55:14 +02:00
# if HAVE_X11
2017-01-31 21:25:00 +02:00
if ( ! m_isX11 ) {
2018-08-30 13:40:34 +03:00
return nullptr ;
2017-01-31 21:25:00 +02:00
}
// check connection
2019-12-29 11:55:14 +02:00
if ( ! _connection ) _connection = QX11Info : : connection ( ) ;
2017-01-31 21:25:00 +02:00
2019-12-29 11:55:14 +02:00
const int width ( source . width ( ) ) ;
const int height ( source . height ( ) ) ;
2017-01-31 21:25:00 +02:00
// create X11 pixmap
2019-12-29 11:55:14 +02:00
Pixmap pixmap = XCreatePixmap ( QX11Info : : display ( ) , QX11Info : : appRootWindow ( ) , width , height , 32 ) ;
2017-01-31 21:25:00 +02:00
// check gc
2019-12-29 11:55:14 +02:00
if ( ! _gc )
{
_gc = xcb_generate_id ( _connection ) ;
xcb_create_gc ( _connection , _gc , pixmap , 0 , nullptr ) ;
2017-01-31 21:25:00 +02:00
}
// // create explicitly shared QPixmap from it
// QPixmap dest( QPixmap::fromX11Pixmap( pixmap, QPixmap::ExplicitlyShared ) );
//
// // create surface for pixmap
// {
// QPainter painter( &dest );
// painter.setCompositionMode( QPainter::CompositionMode_Source );
// painter.drawPixmap( 0, 0, source );
// }
//
//
// return pixmap;
2019-12-29 11:55:14 +02:00
QImage image ( source . toImage ( ) ) ;
2017-01-31 21:25:00 +02:00
xcb_put_image (
_connection , XCB_IMAGE_FORMAT_Z_PIXMAP , pixmap , _gc ,
image . width ( ) , image . height ( ) , 0 , 0 ,
0 , 32 ,
2019-12-30 23:52:58 +02:00
image . byteCount ( ) , image . constBits ( ) ) ;
2017-01-31 21:25:00 +02:00
return ( Qt : : HANDLE ) pixmap ;
2019-12-29 11:55:14 +02:00
# else
2017-01-31 21:25:00 +02:00
return 0 ;
2019-12-29 11:55:14 +02:00
# endif
2017-01-31 21:25:00 +02:00
}
void PanelShadows : : Private : : initPixmap ( const QString & element )
{
m_shadowPixmaps < < q - > pixmap ( element ) ;
}
QPixmap PanelShadows : : Private : : initEmptyPixmap ( const QSize & size )
{
# if HAVE_X11
if ( ! m_isX11 ) {
return QPixmap ( ) ;
}
QPixmap tempEmptyPix ( size ) ;
if ( ! size . isEmpty ( ) ) {
tempEmptyPix . fill ( Qt : : transparent ) ;
}
return tempEmptyPix ;
# else
Q_UNUSED ( size )
return QPixmap ( ) ;
# endif
}
void PanelShadows : : Private : : setupPixmaps ( )
{
clearPixmaps ( ) ;
initPixmap ( QStringLiteral ( " shadow-top " ) ) ;
initPixmap ( QStringLiteral ( " shadow-topright " ) ) ;
initPixmap ( QStringLiteral ( " shadow-right " ) ) ;
initPixmap ( QStringLiteral ( " shadow-bottomright " ) ) ;
initPixmap ( QStringLiteral ( " shadow-bottom " ) ) ;
initPixmap ( QStringLiteral ( " shadow-bottomleft " ) ) ;
initPixmap ( QStringLiteral ( " shadow-left " ) ) ;
initPixmap ( QStringLiteral ( " shadow-topleft " ) ) ;
2019-12-29 11:55:14 +02:00
m_emptyCornerPix = initEmptyPixmap ( QSize ( 1 , 1 ) ) ;
2017-01-31 21:25:00 +02:00
m_emptyCornerLeftPix = initEmptyPixmap ( QSize ( q - > elementSize ( QStringLiteral ( " shadow-topleft " ) ) . width ( ) , 1 ) ) ;
m_emptyCornerTopPix = initEmptyPixmap ( QSize ( 1 , q - > elementSize ( QStringLiteral ( " shadow-topleft " ) ) . height ( ) ) ) ;
m_emptyCornerRightPix = initEmptyPixmap ( QSize ( q - > elementSize ( QStringLiteral ( " shadow-bottomright " ) ) . width ( ) , 1 ) ) ;
m_emptyCornerBottomPix = initEmptyPixmap ( QSize ( 1 , q - > elementSize ( QStringLiteral ( " shadow-bottomright " ) ) . height ( ) ) ) ;
m_emptyVerticalPix = initEmptyPixmap ( QSize ( 1 , q - > elementSize ( QStringLiteral ( " shadow-left " ) ) . height ( ) ) ) ;
m_emptyHorizontalPix = initEmptyPixmap ( QSize ( q - > elementSize ( QStringLiteral ( " shadow-top " ) ) . width ( ) , 1 ) ) ;
if ( m_wayland . shmPool ) {
for ( auto it = m_shadowPixmaps . constBegin ( ) ; it ! = m_shadowPixmaps . constEnd ( ) ; + + it ) {
m_wayland . shadowBuffers < < m_wayland . shmPool - > createBuffer ( it - > toImage ( ) ) ;
}
}
}
void PanelShadows : : Private : : setupData ( Plasma : : FrameSvg : : EnabledBorders enabledBorders )
{
# if HAVE_X11
if ( ! m_isX11 ) {
return ;
}
//shadow-top
if ( enabledBorders & Plasma : : FrameSvg : : TopBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_shadowPixmaps [ 0 ] ) ) ;
} else {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyHorizontalPix ) ) ;
}
//shadow-topright
if ( enabledBorders & Plasma : : FrameSvg : : TopBorder & &
enabledBorders & Plasma : : FrameSvg : : RightBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_shadowPixmaps [ 1 ] ) ) ;
} else if ( enabledBorders & Plasma : : FrameSvg : : TopBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerTopPix ) ) ;
} else if ( enabledBorders & Plasma : : FrameSvg : : RightBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerRightPix ) ) ;
} else {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerPix ) ) ;
}
//shadow-right
if ( enabledBorders & Plasma : : FrameSvg : : RightBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_shadowPixmaps [ 2 ] ) ) ;
} else {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyVerticalPix ) ) ;
}
//shadow-bottomright
if ( enabledBorders & Plasma : : FrameSvg : : BottomBorder & &
enabledBorders & Plasma : : FrameSvg : : RightBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_shadowPixmaps [ 3 ] ) ) ;
} else if ( enabledBorders & Plasma : : FrameSvg : : BottomBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerBottomPix ) ) ;
} else if ( enabledBorders & Plasma : : FrameSvg : : RightBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerRightPix ) ) ;
} else {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerPix ) ) ;
}
//shadow-bottom
if ( enabledBorders & Plasma : : FrameSvg : : BottomBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_shadowPixmaps [ 4 ] ) ) ;
} else {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyHorizontalPix ) ) ;
}
//shadow-bottomleft
if ( enabledBorders & Plasma : : FrameSvg : : BottomBorder & &
enabledBorders & Plasma : : FrameSvg : : LeftBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_shadowPixmaps [ 5 ] ) ) ;
} else if ( enabledBorders & Plasma : : FrameSvg : : BottomBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerBottomPix ) ) ;
} else if ( enabledBorders & Plasma : : FrameSvg : : LeftBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerLeftPix ) ) ;
} else {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerPix ) ) ;
}
//shadow-left
if ( enabledBorders & Plasma : : FrameSvg : : LeftBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_shadowPixmaps [ 6 ] ) ) ;
} else {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyVerticalPix ) ) ;
}
//shadow-topleft
if ( enabledBorders & Plasma : : FrameSvg : : TopBorder & &
enabledBorders & Plasma : : FrameSvg : : LeftBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_shadowPixmaps [ 7 ] ) ) ;
} else if ( enabledBorders & Plasma : : FrameSvg : : TopBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerTopPix ) ) ;
} else if ( enabledBorders & Plasma : : FrameSvg : : LeftBorder ) {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerLeftPix ) ) ;
} else {
data [ enabledBorders ] < < reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerPix ) ) ;
}
# endif
int left , top , right , bottom = 0 ;
QSize marginHint ;
if ( enabledBorders & Plasma : : FrameSvg : : TopBorder ) {
marginHint = q - > elementSize ( QStringLiteral ( " shadow-hint-top-margin " ) ) ;
if ( marginHint . isValid ( ) ) {
top = marginHint . height ( ) ;
} else {
top = m_shadowPixmaps [ 0 ] . height ( ) ; // top
}
} else {
top = 1 ;
}
if ( enabledBorders & Plasma : : FrameSvg : : RightBorder ) {
marginHint = q - > elementSize ( QStringLiteral ( " shadow-hint-right-margin " ) ) ;
if ( marginHint . isValid ( ) ) {
right = marginHint . width ( ) ;
} else {
right = m_shadowPixmaps [ 2 ] . width ( ) ; // right
}
} else {
right = 1 ;
}
if ( enabledBorders & Plasma : : FrameSvg : : BottomBorder ) {
marginHint = q - > elementSize ( QStringLiteral ( " shadow-hint-bottom-margin " ) ) ;
if ( marginHint . isValid ( ) ) {
bottom = marginHint . height ( ) ;
} else {
bottom = m_shadowPixmaps [ 4 ] . height ( ) ; // bottom
}
} else {
bottom = 1 ;
}
if ( enabledBorders & Plasma : : FrameSvg : : LeftBorder ) {
marginHint = q - > elementSize ( QStringLiteral ( " shadow-hint-left-margin " ) ) ;
if ( marginHint . isValid ( ) ) {
left = marginHint . width ( ) ;
} else {
left = m_shadowPixmaps [ 6 ] . width ( ) ; // left
}
} else {
left = 1 ;
}
data [ enabledBorders ] < < top < < right < < bottom < < left ;
}
void PanelShadows : : Private : : freeX11Pixmaps ( )
{
# if HAVE_X11
if ( ! m_isX11 ) {
return ;
}
auto * display = QX11Info : : display ( ) ;
if ( ! display ) {
return ;
}
2019-12-29 11:55:14 +02:00
foreach ( const QPixmap & pixmap , m_shadowPixmaps ) {
2017-01-31 21:25:00 +02:00
if ( ! pixmap . isNull ( ) ) {
XFreePixmap ( display , reinterpret_cast < unsigned long > ( createPixmap ( pixmap ) ) ) ;
}
}
if ( ! m_emptyCornerPix . isNull ( ) ) {
XFreePixmap ( display , reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerPix ) ) ) ;
}
if ( ! m_emptyCornerBottomPix . isNull ( ) ) {
XFreePixmap ( display , reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerBottomPix ) ) ) ;
}
if ( ! m_emptyCornerLeftPix . isNull ( ) ) {
XFreePixmap ( display , reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerLeftPix ) ) ) ;
}
if ( ! m_emptyCornerRightPix . isNull ( ) ) {
XFreePixmap ( display , reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerRightPix ) ) ) ;
}
if ( ! m_emptyCornerTopPix . isNull ( ) ) {
XFreePixmap ( display , reinterpret_cast < unsigned long > ( createPixmap ( m_emptyCornerTopPix ) ) ) ;
}
if ( ! m_emptyVerticalPix . isNull ( ) ) {
XFreePixmap ( display , reinterpret_cast < unsigned long > ( createPixmap ( m_emptyVerticalPix ) ) ) ;
}
if ( ! m_emptyHorizontalPix . isNull ( ) ) {
XFreePixmap ( display , reinterpret_cast < unsigned long > ( createPixmap ( m_emptyHorizontalPix ) ) ) ;
}
# endif
}
void PanelShadows : : Private : : clearPixmaps ( )
{
# if HAVE_X11
freeX11Pixmaps ( ) ;
m_emptyCornerPix = QPixmap ( ) ;
m_emptyCornerBottomPix = QPixmap ( ) ;
m_emptyCornerLeftPix = QPixmap ( ) ;
m_emptyCornerRightPix = QPixmap ( ) ;
m_emptyCornerTopPix = QPixmap ( ) ;
m_emptyVerticalPix = QPixmap ( ) ;
m_emptyHorizontalPix = QPixmap ( ) ;
# endif
freeWaylandBuffers ( ) ;
m_shadowPixmaps . clear ( ) ;
data . clear ( ) ;
}
void PanelShadows : : Private : : freeWaylandBuffers ( )
{
m_wayland . shadowBuffers . clear ( ) ;
}
void PanelShadows : : Private : : updateShadow ( const QWindow * window , Plasma : : FrameSvg : : EnabledBorders enabledBorders )
{
2019-12-29 11:55:14 +02:00
if ( ! hasShadows ( ) ) {
return ;
}
2017-01-31 21:25:00 +02:00
2019-12-29 11:55:14 +02:00
# if HAVE_X11
2017-01-31 21:25:00 +02:00
if ( m_isX11 ) {
updateShadowX11 ( window , enabledBorders ) ;
}
# endif
if ( m_wayland . manager ) {
updateShadowWayland ( window , enabledBorders ) ;
}
}
void PanelShadows : : Private : : updateShadowX11 ( const QWindow * window , Plasma : : FrameSvg : : EnabledBorders enabledBorders )
{
# if HAVE_X11
if ( m_shadowPixmaps . isEmpty ( ) ) {
setupPixmaps ( ) ;
}
if ( ! data . contains ( enabledBorders ) ) {
setupData ( enabledBorders ) ;
}
Display * dpy = QX11Info : : display ( ) ;
Atom atom = XInternAtom ( dpy , " _KDE_NET_WM_SHADOW " , False ) ;
2017-11-24 22:34:02 +02:00
// qDebug() << "going to set the shadow of" << window->winId() << "to" << data;
2017-01-31 21:25:00 +02:00
XChangeProperty ( dpy , window - > winId ( ) , atom , XA_CARDINAL , 32 , PropModeReplace ,
reinterpret_cast < const unsigned char * > ( data [ enabledBorders ] . constData ( ) ) , data [ enabledBorders ] . size ( ) ) ;
# endif
}
void PanelShadows : : Private : : updateShadowWayland ( const QWindow * window , Plasma : : FrameSvg : : EnabledBorders enabledBorders )
{
if ( ! m_wayland . shmPool ) {
return ;
}
if ( m_wayland . shadowBuffers . isEmpty ( ) ) {
setupPixmaps ( ) ;
}
// TODO: check whether the surface already has a shadow
2019-12-29 11:55:14 +02:00
KWayland : : Client : : Surface * surface = KWayland : : Client : : Surface : : fromWindow ( const_cast < QWindow * > ( window ) ) ;
2017-01-31 21:25:00 +02:00
if ( ! surface ) {
return ;
}
auto shadow = m_wayland . manager - > createShadow ( surface , surface ) ;
//shadow-top
if ( enabledBorders & Plasma : : FrameSvg : : TopBorder ) {
shadow - > attachTop ( m_wayland . shadowBuffers . at ( 0 ) ) ;
}
//shadow-topright
if ( enabledBorders & Plasma : : FrameSvg : : TopBorder & &
enabledBorders & Plasma : : FrameSvg : : RightBorder ) {
shadow - > attachTopRight ( m_wayland . shadowBuffers . at ( 1 ) ) ;
}
//shadow-right
if ( enabledBorders & Plasma : : FrameSvg : : RightBorder ) {
shadow - > attachRight ( m_wayland . shadowBuffers . at ( 2 ) ) ;
}
//shadow-bottomright
if ( enabledBorders & Plasma : : FrameSvg : : BottomBorder & &
enabledBorders & Plasma : : FrameSvg : : RightBorder ) {
shadow - > attachBottomRight ( m_wayland . shadowBuffers . at ( 3 ) ) ;
}
//shadow-bottom
if ( enabledBorders & Plasma : : FrameSvg : : BottomBorder ) {
shadow - > attachBottom ( m_wayland . shadowBuffers . at ( 4 ) ) ;
}
//shadow-bottomleft
if ( enabledBorders & Plasma : : FrameSvg : : BottomBorder & &
enabledBorders & Plasma : : FrameSvg : : LeftBorder ) {
shadow - > attachBottomLeft ( m_wayland . shadowBuffers . at ( 5 ) ) ;
}
//shadow-left
if ( enabledBorders & Plasma : : FrameSvg : : LeftBorder ) {
shadow - > attachLeft ( m_wayland . shadowBuffers . at ( 6 ) ) ;
}
//shadow-topleft
if ( enabledBorders & Plasma : : FrameSvg : : TopBorder & &
enabledBorders & Plasma : : FrameSvg : : LeftBorder ) {
shadow - > attachTopLeft ( m_wayland . shadowBuffers . at ( 7 ) ) ;
}
QSize marginHint ;
2019-12-29 11:55:14 +02:00
QMarginsF margins ;
2017-01-31 21:25:00 +02:00
if ( enabledBorders & Plasma : : FrameSvg : : TopBorder ) {
marginHint = q - > elementSize ( QStringLiteral ( " shadow-hint-top-margin " ) ) ;
if ( marginHint . isValid ( ) ) {
margins . setTop ( marginHint . height ( ) ) ;
} else {
margins . setTop ( m_shadowPixmaps [ 0 ] . height ( ) ) ;
}
}
if ( enabledBorders & Plasma : : FrameSvg : : RightBorder ) {
marginHint = q - > elementSize ( QStringLiteral ( " shadow-hint-right-margin " ) ) ;
if ( marginHint . isValid ( ) ) {
margins . setRight ( marginHint . width ( ) ) ;
} else {
margins . setRight ( m_shadowPixmaps [ 2 ] . width ( ) ) ;
}
}
if ( enabledBorders & Plasma : : FrameSvg : : BottomBorder ) {
marginHint = q - > elementSize ( QStringLiteral ( " shadow-hint-bottom-margin " ) ) ;
if ( marginHint . isValid ( ) ) {
margins . setBottom ( marginHint . height ( ) ) ;
} else {
margins . setBottom ( m_shadowPixmaps [ 4 ] . height ( ) ) ;
}
}
if ( enabledBorders & Plasma : : FrameSvg : : LeftBorder ) {
marginHint = q - > elementSize ( QStringLiteral ( " shadow-hint-left-margin " ) ) ;
if ( marginHint . isValid ( ) ) {
margins . setLeft ( marginHint . width ( ) ) ;
} else {
margins . setLeft ( m_shadowPixmaps [ 6 ] . width ( ) ) ;
}
}
shadow - > setOffsets ( margins ) ;
shadow - > commit ( ) ;
surface - > commit ( KWayland : : Client : : Surface : : CommitFlag : : None ) ;
}
void PanelShadows : : Private : : clearShadow ( const QWindow * window )
{
2019-12-29 11:55:14 +02:00
if ( ! static_cast < const QSurface * > ( window ) - > surfaceHandle ( ) ) {
2017-01-31 21:25:00 +02:00
qWarning ( ) < < " Cannot clear shadow from window without native surface! " ;
return ;
}
# if HAVE_X11
if ( m_isX11 ) {
clearShadowX11 ( window ) ;
}
# endif
if ( m_wayland . manager ) {
clearShadowWayland ( window ) ;
}
}
2019-12-29 11:55:14 +02:00
void PanelShadows : : Private : : clearShadowX11 ( const QWindow * window )
2017-01-31 21:25:00 +02:00
{
# if HAVE_X11
Display * dpy = QX11Info : : display ( ) ;
Atom atom = XInternAtom ( dpy , " _KDE_NET_WM_SHADOW " , False ) ;
XDeleteProperty ( dpy , window - > winId ( ) , atom ) ;
# endif
}
void PanelShadows : : Private : : clearShadowWayland ( const QWindow * window )
{
2019-12-29 11:55:14 +02:00
KWayland : : Client : : Surface * surface = KWayland : : Client : : Surface : : fromWindow ( const_cast < QWindow * > ( window ) ) ;
2017-01-31 21:25:00 +02:00
if ( ! surface ) {
return ;
}
m_wayland . manager - > removeShadow ( surface ) ;
surface - > commit ( KWayland : : Client : : Surface : : CommitFlag : : None ) ;
}
2019-12-29 11:55:14 +02:00
bool PanelShadows : : Private : : hasShadows ( ) const
2017-01-31 21:25:00 +02:00
{
2019-12-29 11:55:14 +02:00
return q - > hasShadows ( ) ;
2017-01-31 21:25:00 +02:00
}
void PanelShadows : : Private : : setupWaylandIntegration ( )
{
if ( ! KWindowSystem : : isPlatformWayland ( ) ) {
return ;
}
using namespace KWayland : : Client ;
ConnectionThread * connection = ConnectionThread : : fromApplication ( q ) ;
if ( ! connection ) {
return ;
}
Registry * registry = new Registry ( q ) ;
registry - > create ( connection ) ;
connect ( registry , & Registry : : shadowAnnounced , q ,
2019-12-29 11:55:14 +02:00
[ this , registry ] ( quint32 name , quint32 version ) {
m_wayland . manager = registry - > createShadowManager ( name , version , q ) ;
updateShadows ( ) ;
} , Qt : : QueuedConnection
) ;
2017-01-31 21:25:00 +02:00
connect ( registry , & Registry : : shmAnnounced , q ,
2019-12-29 11:55:14 +02:00
[ this , registry ] ( quint32 name , quint32 version ) {
m_wayland . shmPool = registry - > createShmPool ( name , version , q ) ;
updateShadows ( ) ;
} , Qt : : QueuedConnection
) ;
2017-01-31 21:25:00 +02:00
registry - > setup ( ) ;
connection - > roundtrip ( ) ;
}
# include "moc_panelshadows_p.cpp"
2018-08-30 13:40:34 +03:00