You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
350 lines
11 KiB
350 lines
11 KiB
/* |
|
Copyright (C) 2014 Martin Klapetek <mklapetek@kde.org> |
|
|
|
This library is free software; you can redistribute it and/or |
|
modify it under the terms of the GNU Lesser General Public |
|
License as published by the Free Software Foundation; either |
|
version 2.1 of the License, or (at your option) any later version. |
|
|
|
This library 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 |
|
Lesser General Public License for more details. |
|
|
|
You should have received a copy of the GNU Lesser General Public |
|
License along with this library; if not, write to the Free Software |
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
|
*/ |
|
|
|
#include "notificationshelper.h" |
|
|
|
#include <QGuiApplication> |
|
#include <qfontmetrics.h> |
|
#include <QTimer> |
|
#include <QQuickWindow> |
|
#include <QQuickItem> |
|
#include <QQmlEngine> |
|
#include <QReadWriteLock> |
|
#include <QDebug> |
|
|
|
NotificationsHelper::NotificationsHelper(QObject *parent) |
|
: QObject(parent), |
|
m_popupLocation(NotificationsHelper::BottomRight), |
|
m_busy(false) |
|
{ |
|
m_mutex = new QReadWriteLock(QReadWriteLock::Recursive); |
|
m_offset = QFontMetrics(QGuiApplication::font()).boundingRect(QStringLiteral("M")).height(); |
|
|
|
m_dispatchTimer = new QTimer(this); |
|
m_dispatchTimer->setInterval(500); |
|
m_dispatchTimer->setSingleShot(true); |
|
connect(m_dispatchTimer, &QTimer::timeout, [this](){m_busy = false; processQueues();}); |
|
} |
|
|
|
NotificationsHelper::~NotificationsHelper() |
|
{ |
|
qDeleteAll(m_availablePopups); |
|
qDeleteAll(m_popupsOnScreen); |
|
delete m_mutex; |
|
} |
|
|
|
void NotificationsHelper::setPopupLocation(PositionOnScreen popupLocation) |
|
{ |
|
if (m_popupLocation != popupLocation) { |
|
m_popupLocation = popupLocation; |
|
emit popupLocationChanged(); |
|
|
|
repositionPopups(); |
|
} |
|
} |
|
|
|
void NotificationsHelper::setPlasmoidScreenGeometry(const QRect &plasmoidScreenGeometry) |
|
{ |
|
m_plasmoidScreen = plasmoidScreenGeometry; |
|
repositionPopups(); |
|
} |
|
|
|
void NotificationsHelper::addNotificationPopup(QObject *win) |
|
{ |
|
QQuickWindow *popup = qobject_cast<QQuickWindow*>(win); |
|
m_availablePopups.append(popup); |
|
|
|
// Don't let QML ever delete this component |
|
QQmlEngine::setObjectOwnership(win, QQmlEngine::CppOwnership); |
|
|
|
connect(win, SIGNAL(notificationTimeout()), |
|
this, SLOT(onPopupClosed())); |
|
|
|
connect(popup, &QWindow::heightChanged, this, &NotificationsHelper::repositionPopups, Qt::UniqueConnection); |
|
connect(popup, &QWindow::visibleChanged, this, &NotificationsHelper::onPopupShown, Qt::UniqueConnection); |
|
|
|
popup->setProperty("initialPositionSet", false); |
|
} |
|
|
|
void NotificationsHelper::onPopupShown() |
|
{ |
|
QWindow *popup = qobject_cast<QWindow*>(sender()); |
|
if (!popup || !popup->isVisible()) { |
|
return; |
|
} |
|
|
|
// Make sure Dialog lays everything out and gets proper geometry |
|
QMetaObject::invokeMethod(popup, "updateVisibility", Qt::DirectConnection, Q_ARG(bool, true)); |
|
|
|
// Now we can position the popups properly as the geometry is now known |
|
repositionPopups(); |
|
} |
|
|
|
void NotificationsHelper::processQueues() |
|
{ |
|
if (m_busy) { |
|
return; |
|
} |
|
|
|
m_mutex->lockForRead(); |
|
bool shouldProcessShow = !m_showQueue.isEmpty() && !m_availablePopups.isEmpty(); |
|
m_mutex->unlock(); |
|
|
|
if (shouldProcessShow) { |
|
m_busy = true; |
|
processShow(); |
|
// Return here, makes the movement more clear and easier to follow |
|
return; |
|
} |
|
|
|
m_mutex->lockForRead(); |
|
bool shouldProcessHide = !m_hideQueue.isEmpty(); |
|
m_mutex->unlock(); |
|
|
|
if (shouldProcessHide) { |
|
m_busy = true; |
|
processHide(); |
|
} |
|
} |
|
|
|
void NotificationsHelper::processShow() |
|
{ |
|
m_mutex->lockForWrite(); |
|
const QVariantMap notificationData = m_showQueue.takeFirst(); |
|
m_mutex->unlock(); |
|
|
|
QString sourceName = notificationData.value(QStringLiteral("source")).toString(); |
|
|
|
// Try getting existing popup for the given source |
|
// (case of notification being just updated) |
|
QQuickWindow *popup = m_sourceMap.value(sourceName); |
|
|
|
if (!popup) { |
|
// No existing notification for the given source, |
|
// take one from the available popups |
|
m_mutex->lockForWrite(); |
|
popup = m_availablePopups.takeFirst(); |
|
m_popupsOnScreen << popup; |
|
m_sourceMap.insert(sourceName, popup); |
|
m_mutex->unlock(); |
|
// Set the source name directly on the popup object too |
|
// to avoid looking up the notificationProperties map as above |
|
popup->setProperty("sourceName", sourceName); |
|
} |
|
|
|
// Populate the popup with data, this is the component's own QML method |
|
QMetaObject::invokeMethod(popup, "populatePopup", Qt::DirectConnection, Q_ARG(QVariant, notificationData)); |
|
|
|
QTimer::singleShot(300, popup, &QWindow::show); |
|
|
|
if (!m_dispatchTimer->isActive()) { |
|
m_dispatchTimer->start(); |
|
} |
|
} |
|
|
|
void NotificationsHelper::processHide() |
|
{ |
|
m_mutex->lockForWrite(); |
|
QQuickWindow *popup = m_hideQueue.takeFirst(); |
|
m_mutex->unlock(); |
|
|
|
if (popup) { |
|
m_mutex->lockForWrite(); |
|
// Remove the popup from the active list and return it into the available list |
|
m_popupsOnScreen.removeAll(popup); |
|
m_sourceMap.remove(popup->property("sourceName").toString()); |
|
if (!m_availablePopups.contains(popup)) { |
|
// make extra sure that pointers in here aren't doubled |
|
m_availablePopups.append(popup); |
|
} |
|
m_mutex->unlock(); |
|
|
|
popup->hide(); |
|
|
|
// Make sure the popup gets placed correctly |
|
// next time it's put on screen |
|
popup->setProperty("initialPositionSet", false); |
|
} |
|
|
|
m_mutex->lockForRead(); |
|
bool shouldReposition = !m_popupsOnScreen.isEmpty();// && m_showQueue.isEmpty(); |
|
m_mutex->unlock(); |
|
|
|
if (shouldReposition) { |
|
repositionPopups(); |
|
} |
|
|
|
if (!m_dispatchTimer->isActive()) { |
|
m_dispatchTimer->start(); |
|
} |
|
} |
|
|
|
void NotificationsHelper::displayNotification(const QVariantMap ¬ificationData) |
|
{ |
|
if (notificationData.isEmpty()) { |
|
return; |
|
} |
|
|
|
QVariant sourceName = notificationData.value(QStringLiteral("source")); |
|
|
|
// first check if we don't already have data for the same source |
|
// which would mean that the notification was just updated |
|
// so remove the old one and append the newest data only |
|
QMutableListIterator<QVariantMap> i(m_showQueue); |
|
while (i.hasNext()) { |
|
if (i.next().value(QStringLiteral("source")) == sourceName) { |
|
m_mutex->lockForWrite(); |
|
i.remove(); |
|
m_mutex->unlock(); |
|
} |
|
} |
|
|
|
// ...also look into the hide queue, if it's already queued |
|
// for hiding, we need to remove it from there otherwise |
|
// it will get closed too soon |
|
QMutableListIterator<QQuickWindow*> j(m_hideQueue); |
|
while (j.hasNext()) { |
|
if (j.next()->property("sourceName") == sourceName) { |
|
m_mutex->lockForWrite(); |
|
j.remove(); |
|
m_mutex->unlock(); |
|
} |
|
} |
|
|
|
m_mutex->lockForWrite(); |
|
m_showQueue.append(notificationData); |
|
m_mutex->unlock(); |
|
|
|
if (!m_dispatchTimer->isActive()) { |
|
// If the dispatch timer is not already running, process |
|
// the queues directly, that should cut the time between |
|
// notification emitting the event and popup displaying |
|
processQueues(); |
|
} |
|
} |
|
|
|
void NotificationsHelper::closePopup(const QString &sourceName) |
|
{ |
|
QQuickWindow *popup = m_sourceMap.value(sourceName); |
|
|
|
m_mutex->lockForRead(); |
|
bool shouldQueue = popup && !m_hideQueue.contains(popup); |
|
m_mutex->unlock(); |
|
|
|
// Make sure the notification that was closed (programatically) |
|
// is not in the show queue. This is important otherwise that |
|
// notification will be shown and then never closed (because |
|
// the close event arrives here, before it's even shown) |
|
QMutableListIterator<QVariantMap> i(m_showQueue); |
|
while (i.hasNext()) { |
|
if (i.next().value(QStringLiteral("source")) == sourceName) { |
|
m_mutex->lockForWrite(); |
|
i.remove(); |
|
m_mutex->unlock(); |
|
} |
|
} |
|
|
|
if (shouldQueue) { |
|
m_mutex->lockForWrite(); |
|
m_hideQueue.append(popup); |
|
m_mutex->unlock(); |
|
|
|
if (!m_dispatchTimer->isActive()) { |
|
processQueues(); |
|
} |
|
} |
|
} |
|
|
|
void NotificationsHelper::onPopupClosed() |
|
{ |
|
QQuickWindow *popup = qobject_cast<QQuickWindow*>(sender()); |
|
|
|
m_mutex->lockForRead(); |
|
bool shouldQueue = popup && !m_hideQueue.contains(popup); |
|
m_mutex->unlock(); |
|
|
|
if (shouldQueue) { |
|
m_mutex->lockForWrite(); |
|
m_hideQueue << popup; |
|
m_mutex->unlock(); |
|
|
|
if (!m_dispatchTimer->isActive()) { |
|
processQueues(); |
|
} |
|
} |
|
} |
|
|
|
void NotificationsHelper::repositionPopups() |
|
{ |
|
int cumulativeHeight = m_offset; |
|
|
|
m_mutex->lockForWrite(); |
|
|
|
for (int i = 0; i < m_popupsOnScreen.size(); ++i) { |
|
if (m_popupLocation == NotificationsHelper::TopLeft |
|
|| m_popupLocation == NotificationsHelper::TopCenter |
|
|| m_popupLocation == NotificationsHelper::TopRight) { |
|
|
|
int posY = m_plasmoidScreen.top() + cumulativeHeight; |
|
|
|
if (m_popupsOnScreen[i]->isVisible() && m_popupsOnScreen[i]->property("initialPositionSet").toBool() == true && m_popupsOnScreen[i]->y() != 0) { |
|
//if it's visible, go through setProperty which animates it |
|
m_popupsOnScreen[i]->setProperty("y", posY); |
|
} else { |
|
// ...otherwise just set it directly |
|
m_popupsOnScreen[i]->setY(posY); |
|
m_popupsOnScreen[i]->setProperty("initialPositionSet", true); |
|
} |
|
} else { |
|
int posY = m_plasmoidScreen.bottom() - cumulativeHeight - m_popupsOnScreen[i]->contentItem()->height(); |
|
|
|
if (m_popupsOnScreen[i]->isVisible() && m_popupsOnScreen[i]->property("initialPositionSet").toBool() == true && m_popupsOnScreen[i]->y() != 0) { |
|
m_popupsOnScreen[i]->setProperty("y", posY); |
|
} else { |
|
m_popupsOnScreen[i]->setY(posY); |
|
m_popupsOnScreen[i]->setProperty("initialPositionSet", true); |
|
} |
|
} |
|
|
|
switch (m_popupLocation) { |
|
case TopRight: |
|
case BottomRight: |
|
m_popupsOnScreen[i]->setX(m_plasmoidScreen.right() - m_popupsOnScreen[i]->contentItem()->width() - m_offset); |
|
break; |
|
case TopCenter: |
|
case BottomCenter: |
|
m_popupsOnScreen[i]->setX(m_plasmoidScreen.x() + (m_plasmoidScreen.width() / 2) - (m_popupsOnScreen[i]->contentItem()->width() / 2)); |
|
break; |
|
case TopLeft: |
|
case BottomLeft: |
|
m_popupsOnScreen[i]->setX(m_plasmoidScreen.left() + m_offset); |
|
break; |
|
case Left: |
|
case Center: |
|
case Right: |
|
// Fall-through to make the compiler happy |
|
break; |
|
} |
|
|
|
cumulativeHeight += (m_popupsOnScreen[i]->contentItem()->height() + m_offset); |
|
} |
|
|
|
m_mutex->unlock(); |
|
} |
|
|
|
|
|
|