Skip to content
Snippets Groups Projects
Select Git revision
  • master default protected
  • beta/202505231812
  • stable/20250523.0
  • nightly/20250523.0
  • nightly/20250515.0
  • nightly/20250510.0
  • nightly/20250509.1
  • nightly/20250509.0
  • stable/20250430.1
  • stable/20250430.0
  • beta/202504301614
  • nightly/20250430.0
  • stable/20250424.1
  • beta/202504241506
  • stable/20250424.0
  • nightly/20250424.1
  • nightly/20250424.0
  • nightly/20250422.0
  • beta/202504120241
  • stable/20250411.0
  • nightly/20250411.0
21 results

conversationlistmodel.cpp

Blame
  • Code owners
    Assign users and groups as approvers for specific file changes. Learn more.
    conversationlistmodel.cpp 5.12 KiB
    /*
     * Copyright (C) 2021-2022 Savoir-faire Linux Inc.
     * Author: Andreas Traczyk <andreas.traczyk@savoirfairelinux.com>
     * Author: Mingrui Zhang <mingrui.zhang@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 "conversationlistmodel.h"
    
    #include "uri.h"
    
    ConversationListModel::ConversationListModel(LRCInstance* instance, QObject* parent)
        : ConversationListModelBase(instance, parent)
    {
        if (!model_)
            return;
    
        connect(
            model_,
            &ConversationModel::beginInsertRows,
            this,
            [this](int position, int rows) {
                beginInsertRows(QModelIndex(), position, position + (rows - 1));
            },
            Qt::DirectConnection);
        connect(model_,
                &ConversationModel::endInsertRows,
                this,
                &ConversationListModel::endInsertRows,
                Qt::DirectConnection);
    
        connect(
            model_,
            &ConversationModel::beginRemoveRows,
            this,
            [this](int position, int rows) {
                beginRemoveRows(QModelIndex(), position, position + (rows - 1));
            },
            Qt::DirectConnection);
        connect(model_,
                &ConversationModel::endRemoveRows,
                this,
                &ConversationListModel::endRemoveRows,
                Qt::DirectConnection);
    
        connect(model_, &ConversationModel::dataChanged, this, [this](int position) {
            const auto index = createIndex(position, 0);
            Q_EMIT ConversationListModel::dataChanged(index, index);
        });
    }
    
    int
    ConversationListModel::rowCount(const QModelIndex& parent) const
    {
        // For list models only the root node (an invalid parent) should return the list's size. For all
        // other (valid) parents, rowCount() should return 0 so that it does not become a tree model.
        if (!parent.isValid() && model_) {
            return model_->getConversations().size();
        }
        return 0;
    }
    
    QVariant
    ConversationListModel::data(const QModelIndex& index, int role) const
    {
        const auto& data = model_->getConversations();
        if (!index.isValid() || data.empty())
            return {};
        return dataForItem(data.at(index.row()), role);
    }
    
    ConversationListProxyModel::ConversationListProxyModel(QAbstractListModel* model, QObject* parent)
        : SelectableListProxyModel(model, parent)
    {
        setSortRole(ConversationList::Role::LastInteractionTimeStamp);
        sort(0, Qt::DescendingOrder);
        setFilterCaseSensitivity(Qt::CaseSensitivity::CaseInsensitive);
    }
    
    bool
    ConversationListProxyModel::filterAcceptsRow(int sourceRow, const QModelIndex& sourceParent) const
    {
        QModelIndex index = sourceModel()->index(sourceRow, 0, sourceParent);
        auto rx = filterRegularExpression();
        auto uriStripper = URI(rx.pattern());
        bool stripScheme = (uriStripper.schemeType() < URI::SchemeType::COUNT__);
        FlagPack<URI::Section> flags = URI::Section::USER_INFO | URI::Section::HOSTNAME
                                       | URI::Section::PORT;
        if (!stripScheme) {
            flags |= URI::Section::SCHEME;
        }
        rx.setPattern(uriStripper.format(flags));
    
        using namespace ConversationList;
    
        QStringList toFilter;
        toFilter += index.data(Role::Title).toString();
        toFilter += index.data(Role::Uris).toStringList();
        toFilter += index.data(Role::Monikers).toStringList();
    
        // requests
        auto isRequest = index.data(Role::IsRequest).toBool();
        bool requestFilter = filterRequests_ ? isRequest : !isRequest;
    
        bool match {false};
    
        // banned contacts require exact match
        if (index.data(Role::IsBanned).toBool()) {
            if (!rx.isValid()) {
                Q_FOREACH (const auto& filter, toFilter) {
                    auto matchResult = rx.match(filter);
                    if (matchResult.hasMatch() && matchResult.captured(0) == filter) {
                        match = true;
                        break;
                    }
                }
            }
        } else {
            Q_FOREACH (const auto& filter, toFilter)
                if (rx.match(filter).hasMatch()) {
                    match = true;
                    break;
                }
        }
    
        return requestFilter && match;
    }
    
    bool
    ConversationListProxyModel::lessThan(const QModelIndex& left, const QModelIndex& right) const
    {
        QVariant leftData = sourceModel()->data(left, sortRole());
        QVariant rightData = sourceModel()->data(right, sortRole());
        // we're assuming the sort role data type here is some integral time
        return leftData.toULongLong() < rightData.toULongLong();
    }
    
    void
    ConversationListProxyModel::setFilterRequests(bool filterRequests)
    {
        beginResetModel();
        filterRequests_ = filterRequests;
        endResetModel();
        updateSelection();
    };