mirror of
https://github.com/linuxdeepin/dde-dock.git
synced 2025-06-04 09:23:03 +00:00

wayland下时尚模式的xcb_connection_t的值为nullptr,构造函数中传入一个可用的值 Log: 修复wayland环境时尚模式下打开企业微信崩溃的问题 Influence: 进入wayland桌面,进入时尚模式,打开企业微信,双击,观察企业微信是否正常打开 Task: https://pms.uniontech.com/task-view-196629.html Change-Id: Ib0a5ebdadbce672474db9dab57f27da263b8096b
223 lines
7.5 KiB
C++
223 lines
7.5 KiB
C++
/*
|
||
* Copyright (C) 2018 ~ 2025 Deepin Technology Co., Ltd.
|
||
*
|
||
* Author: fanpengcheng <fanpengcheng_cm@deepin.com>
|
||
*
|
||
* Maintainer: fanpengcheng <fanpengcheng_cm@deepin.com>
|
||
*
|
||
* This program is free software: you can redistribute it and/or modify
|
||
* it under the terms of the GNU General Public License as published by
|
||
* the Free Software Foundation, either version 3 of the License, or
|
||
* 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 General Public License
|
||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
*/
|
||
#include "tray_delegate.h"
|
||
#include "tray_gridview.h"
|
||
#include "tray_model.h"
|
||
#include "widgets/xembedtrayitemwidget.h"
|
||
#include "widgets/indicatortrayitem.h"
|
||
#include "widgets/indicatorplugin.h"
|
||
#include "widgets/snitrayitemwidget.h"
|
||
#include "widgets/expandiconwidget.h"
|
||
#include "utils.h"
|
||
|
||
#include <DGuiApplicationHelper>
|
||
|
||
#include <QPointer>
|
||
#include <QDebug>
|
||
#include <QEvent>
|
||
#include <QKeyEvent>
|
||
#include <QApplication>
|
||
#include <QPainterPath>
|
||
|
||
#include <xcb/xcb_icccm.h>
|
||
#include <X11/Xlib.h>
|
||
|
||
TrayDelegate::TrayDelegate(QListView *view, QObject *parent)
|
||
: QStyledItemDelegate(parent)
|
||
, m_position(Dock::Position::Bottom)
|
||
, m_listView(view)
|
||
{
|
||
}
|
||
|
||
void TrayDelegate::setPositon(Dock::Position position)
|
||
{
|
||
m_position = position;
|
||
}
|
||
|
||
QWidget *TrayDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index) const
|
||
{
|
||
Q_UNUSED(option);
|
||
|
||
TrayIconType type = index.data(TrayModel::TypeRole).value<TrayIconType>();
|
||
QString key = index.data(TrayModel::KeyRole).value<QString>();
|
||
QString servicePath = index.data(TrayModel::ServiceRole).value<QString>();
|
||
quint32 winId = index.data(TrayModel::WinIdRole).value<quint32>();
|
||
|
||
BaseTrayWidget *trayWidget = nullptr;
|
||
if(type == TrayIconType::XEMBED) {
|
||
if (Utils::IS_WAYLAND_DISPLAY) {
|
||
static Display *display = XOpenDisplay(nullptr);
|
||
static int screenp = 0;
|
||
static xcb_connection_t *xcb_connection = xcb_connect(qgetenv("DISPLAY"), &screenp);
|
||
trayWidget = new XEmbedTrayItemWidget(winId, xcb_connection, display, parent);
|
||
} else {
|
||
trayWidget = new XEmbedTrayItemWidget(winId, nullptr, nullptr, parent);
|
||
}
|
||
const TrayModel *model = qobject_cast<const TrayModel *>(index.model());
|
||
if (model)
|
||
connect(model, &TrayModel::requestUpdateIcon, trayWidget, &BaseTrayWidget::updateIcon);
|
||
} else if (type == TrayIconType::SNI) {
|
||
trayWidget = new SNITrayItemWidget(servicePath, parent);
|
||
} else if (type == TrayIconType::EXPANDICON) {
|
||
ExpandIconWidget *expandWidget = new ExpandIconWidget(parent);
|
||
expandWidget->setPositonValue(m_position);
|
||
connect(expandWidget, &ExpandIconWidget::trayVisbleChanged, this, [ = ](bool visible) {
|
||
Q_EMIT visibleChanged(index, visible);
|
||
});
|
||
connect(this, &TrayDelegate::requestDrag, this, &TrayDelegate::onRequestDrag);
|
||
trayWidget = expandWidget;
|
||
} else if (type == TrayIconType::INDICATOR) {
|
||
QString indicateName = key;
|
||
int flagIndex = indicateName.indexOf("indicator:");
|
||
if (flagIndex >= 0)
|
||
indicateName = indicateName.right(indicateName.length() - QString("indicator:").length());
|
||
IndicatorTrayItem *indicatorWidget = new IndicatorTrayItem(indicateName, parent);
|
||
TrayModel *dataModel = qobject_cast<TrayModel *>(m_listView->model());
|
||
if (IndicatorTrayItem *sourceIndicatorWidget = dataModel->indicatorWidget(key)) {
|
||
const QByteArray pixmapData = sourceIndicatorWidget->pixmapData();
|
||
if (!pixmapData.isEmpty())
|
||
indicatorWidget->setPixmapData(pixmapData);
|
||
const QString text = sourceIndicatorWidget->text();
|
||
if (!text.isEmpty())
|
||
indicatorWidget->setText(text);
|
||
}
|
||
trayWidget = indicatorWidget;
|
||
}
|
||
|
||
if (trayWidget)
|
||
trayWidget->setFixedSize(16, 16);
|
||
|
||
return trayWidget;
|
||
}
|
||
|
||
void TrayDelegate::onRequestDrag(bool on)
|
||
{
|
||
ExpandIconWidget *expandwidget = expandWidget();
|
||
if (!expandwidget)
|
||
return;
|
||
|
||
if (on) {
|
||
expandwidget->setTrayPanelVisible(true);
|
||
} else {
|
||
// 如果释放鼠标,则判断当前鼠标的位置是否在托盘内部,如果在,则无需隐藏
|
||
QPoint currentPoint = QCursor::pos();
|
||
QWidget *view = expandwidget->popupTrayView();
|
||
expandwidget->setTrayPanelVisible(view->geometry().contains(currentPoint));
|
||
}
|
||
}
|
||
|
||
void TrayDelegate::setEditorData(QWidget *editor, const QModelIndex &index) const
|
||
{
|
||
BaseTrayWidget *widget = static_cast<BaseTrayWidget *>(editor);
|
||
if (widget) {
|
||
widget->setNeedShow(!index.data(TrayModel::Blank).toBool());
|
||
}
|
||
}
|
||
|
||
QSize TrayDelegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const
|
||
{
|
||
Q_UNUSED(option);
|
||
Q_UNUSED(index);
|
||
|
||
// 如果是弹出托盘,则显示正常大小
|
||
if (isPopupTray())
|
||
return QSize(ITEM_SIZE, ITEM_SIZE);
|
||
|
||
// 如果是任务栏的托盘,则高度显示为listView的高度或宽度
|
||
if (m_position == Dock::Position::Top || m_position == Dock::Position::Bottom)
|
||
return QSize(ITEM_SIZE, m_listView->height());
|
||
|
||
return QSize(m_listView->width(), ITEM_SIZE);
|
||
}
|
||
|
||
void TrayDelegate::updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex &index) const
|
||
{
|
||
Q_UNUSED(index);
|
||
QRect rect = option.rect;
|
||
// 让控件居中显示
|
||
editor->setGeometry(rect.x() + (rect.width() - ICON_SIZE) / 2,
|
||
rect.y() + (rect.height() - ICON_SIZE) / 2,
|
||
ICON_SIZE, ICON_SIZE);
|
||
}
|
||
|
||
void TrayDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
|
||
{
|
||
Q_UNUSED(index);
|
||
|
||
if (!isPopupTray())
|
||
return;
|
||
|
||
QColor borderColor;
|
||
QColor backColor;
|
||
if (DGuiApplicationHelper::instance()->themeType() == DGuiApplicationHelper::LightType) {
|
||
// 白色主题的情况下
|
||
borderColor = Qt::black;
|
||
borderColor.setAlpha(static_cast<int>(255 * 0.05));
|
||
backColor = Qt::white;
|
||
backColor.setAlpha(static_cast<int>(255 * 0.4));
|
||
} else {
|
||
borderColor = Qt::black;
|
||
borderColor.setAlpha(static_cast<int>(255 * 0.2));
|
||
backColor = Qt::black;
|
||
backColor.setAlpha(static_cast<int>(255 * 0.4));
|
||
}
|
||
|
||
painter->save();
|
||
QPainterPath path;
|
||
path.addRoundedRect(option.rect, 8, 8);
|
||
painter->setRenderHint(QPainter::Antialiasing);
|
||
painter->fillPath(path, backColor);
|
||
painter->setPen(borderColor);
|
||
painter->drawPath(path);
|
||
painter->restore();
|
||
}
|
||
|
||
ExpandIconWidget *TrayDelegate::expandWidget()
|
||
{
|
||
if (!m_listView)
|
||
return nullptr;
|
||
|
||
QAbstractItemModel *dataModel = m_listView->model();
|
||
if (!dataModel)
|
||
return nullptr;
|
||
|
||
for (int i = 0; i < dataModel->rowCount() - 1; i++) {
|
||
QModelIndex index = dataModel->index(i, 0);
|
||
ExpandIconWidget *widget = qobject_cast<ExpandIconWidget *>(m_listView->indexWidget(index));
|
||
if (widget)
|
||
return widget;
|
||
}
|
||
|
||
return nullptr;
|
||
}
|
||
|
||
bool TrayDelegate::isPopupTray() const
|
||
{
|
||
if (!m_listView)
|
||
return false;
|
||
|
||
TrayModel *dataModel = qobject_cast<TrayModel *>(m_listView->model());
|
||
if (!dataModel)
|
||
return false;
|
||
|
||
return dataModel->isIconTray();
|
||
}
|