1
0
Fork 0
mirror of https://git.jami.net/savoirfairelinux/jami-client-qt.git synced 2025-08-27 18:23:27 +02:00
jami-client-qt/src/calloverlaymodel.cpp

233 lines
5.8 KiB
C++
Raw Normal View History

/*
* Copyright (C) 2021 by Savoir-faire Linux
* Author: Andreas Traczyk <andreas.traczyk@savoirfairelinux.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
* (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 General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "calloverlaymodel.h"
#include <QEvent>
#include <QMouseEvent>
#include <QQuickWindow>
CallControlListModel::CallControlListModel(QObject* parent)
: QAbstractListModel(parent)
{}
int
CallControlListModel::rowCount(const QModelIndex& parent) const
{
if (parent.isValid())
return 0;
return data_.size();
}
QVariant
CallControlListModel::data(const QModelIndex& index, int role) const
{
if (!index.isValid())
return QVariant();
using namespace CallControl;
auto item = data_.at(index.row());
switch (role) {
case Role::DummyRole:
break;
#define X(t, role) \
case Role::role: \
return QVariant::fromValue(item.role);
CC_ROLES
#undef X
default:
break;
}
return QVariant();
}
QHash<int, QByteArray>
CallControlListModel::roleNames() const
{
using namespace CallControl;
QHash<int, QByteArray> roles;
#define X(t, role) roles[role] = #role;
CC_ROLES
#undef X
return roles;
}
void
CallControlListModel::setBadgeCount(int row, int count)
{
if (row >= rowCount())
return;
data_[row].BadgeCount = count;
auto idx = index(row, 0);
Q_EMIT dataChanged(idx, idx);
}
void
CallControlListModel::addItem(const CallControl::Item& item)
{
beginResetModel();
data_.append(item);
endResetModel();
}
IndexRangeFilterProxyModel::IndexRangeFilterProxyModel(QAbstractListModel* parent)
: QSortFilterProxyModel(parent)
{
setSourceModel(parent);
sourceModel()->data(QModelIndex());
}
bool
IndexRangeFilterProxyModel::filterAcceptsRow(int sourceRow, const QModelIndex& sourceParent) const
{
auto index = sourceModel()->index(sourceRow, 0, sourceParent);
bool predicate = true;
if (filterRole() != Qt::DisplayRole) {
predicate = sourceModel()->data(index, filterRole()).toInt() != 0;
}
return sourceRow <= max_ && sourceRow >= min_ && predicate;
}
void
IndexRangeFilterProxyModel::setRange(int min, int max)
{
min_ = min;
max_ = max;
invalidateFilter();
}
CallOverlayModel::CallOverlayModel(LRCInstance* instance, QObject* parent)
: QObject(parent)
, lrcInstance_(instance)
, primaryModel_(new CallControlListModel(this))
, secondaryModel_(new CallControlListModel(this))
, overflowModel_(new IndexRangeFilterProxyModel(secondaryModel_))
, overflowVisibleModel_(new IndexRangeFilterProxyModel(secondaryModel_))
, overflowHiddenModel_(new IndexRangeFilterProxyModel(secondaryModel_))
{
connect(this,
&CallOverlayModel::overflowIndexChanged,
this,
&CallOverlayModel::setControlRanges);
overflowVisibleModel_->setFilterRole(CallControl::Role::BadgeCount);
}
void
CallOverlayModel::addPrimaryControl(const QVariantMap& props)
{
CallControl::Item item {
#define X(t, role) props[#role].value<t>(),
CC_ROLES
#undef X
};
primaryModel_->addItem(item);
}
void
CallOverlayModel::addSecondaryControl(const QVariantMap& props)
{
CallControl::Item item {
#define X(t, role) props[#role].value<t>(),
CC_ROLES
#undef X
};
secondaryModel_->addItem(item);
setControlRanges();
}
void
CallOverlayModel::setBadgeCount(int row, int count)
{
secondaryModel_->setBadgeCount(row, count);
}
QVariant
CallOverlayModel::primaryModel()
{
return QVariant::fromValue(primaryModel_);
}
QVariant
CallOverlayModel::secondaryModel()
{
return QVariant::fromValue(secondaryModel_);
}
QVariant
CallOverlayModel::overflowModel()
{
return QVariant::fromValue(overflowModel_);
}
QVariant
CallOverlayModel::overflowVisibleModel()
{
return QVariant::fromValue(overflowVisibleModel_);
}
QVariant
CallOverlayModel::overflowHiddenModel()
{
return QVariant::fromValue(overflowHiddenModel_);
}
void
CallOverlayModel::registerFilter(QQuickWindow* object, QQuickItem* item)
{
if (!object || !item || watchedItems_.contains(item))
return;
watchedItems_.push_back(item);
if (watchedItems_.size() == 1)
object->installEventFilter(this);
}
void
CallOverlayModel::unregisterFilter(QQuickWindow* object, QQuickItem* item)
{
if (!object || !item || !watchedItems_.contains(item))
return;
watchedItems_.removeOne(item);
if (watchedItems_.size() == 0)
object->removeEventFilter(this);
}
bool
CallOverlayModel::eventFilter(QObject* object, QEvent* event)
{
if (event->type() == QEvent::MouseMove) {
auto mouseEvent = static_cast<QMouseEvent*>(event);
QPoint eventPos(mouseEvent->x(), mouseEvent->y());
auto windowItem = static_cast<QQuickWindow*>(object)->contentItem();
Q_FOREACH (const auto& item, watchedItems_) {
if (item->contains(windowItem->mapToItem(item, eventPos))) {
Q_EMIT mouseMoved(item);
}
}
}
return QObject::eventFilter(object, event);
}
void
CallOverlayModel::setControlRanges()
{
overflowModel_->setRange(0, overflowIndex_ - 1);
overflowVisibleModel_->setRange(overflowIndex_, secondaryModel_->rowCount());
overflowHiddenModel_->setRange(overflowIndex_, secondaryModel_->rowCount());
}