/* SPDX-FileCopyrightText: 2014 Kai Uwe Broulik SPDX-FileCopyrightText: 2014 Martin Klapetek SPDX-License-Identifier: GPL-2.0-or-later */ #include "timezonemodel.h" #include "timezonesi18n.h" #include #include #include #include TimeZoneFilterProxy::TimeZoneFilterProxy(QObject *parent) : QSortFilterProxyModel(parent) { m_stringMatcher.setCaseSensitivity(Qt::CaseInsensitive); } bool TimeZoneFilterProxy::filterAcceptsRow(int source_row, const QModelIndex &source_parent) const { if (!sourceModel() || (m_filterString.isEmpty() && !m_onlyShowChecked)) { return true; } const bool checked = sourceModel()->index(source_row, 0, source_parent).data(TimeZoneModel::CheckedRole).toBool(); if (m_onlyShowChecked && !checked) { return false; } const QString city = sourceModel()->index(source_row, 0, source_parent).data(TimeZoneModel::CityRole).toString(); const QString region = sourceModel()->index(source_row, 0, source_parent).data(TimeZoneModel::RegionRole).toString(); const QString comment = sourceModel()->index(source_row, 0, source_parent).data(TimeZoneModel::CommentRole).toString(); if (m_stringMatcher.indexIn(city) != -1 || m_stringMatcher.indexIn(region) != -1 || m_stringMatcher.indexIn(comment) != -1) { return true; } return false; } void TimeZoneFilterProxy::setFilterString(const QString &filterString) { m_filterString = filterString; m_stringMatcher.setPattern(filterString); Q_EMIT filterStringChanged(); invalidateFilter(); } void TimeZoneFilterProxy::setOnlyShowChecked(const bool show) { if (m_onlyShowChecked == show) { return; } m_onlyShowChecked = show; Q_EMIT onlyShowCheckedChanged(); invalidateFilter(); } //============================================================================= TimeZoneModel::TimeZoneModel(QObject *parent) : QAbstractListModel(parent) , m_timezonesI18n(new TimeZonesI18n(this)) { update(); QDBusConnection::sessionBus().connect(QString(), QStringLiteral("/org/kde/kcmshell_clock"), // QStringLiteral("org.kde.kcmshell_clock"), QStringLiteral("clockUpdated"), this, SLOT(slotUpdate())); } TimeZoneModel::~TimeZoneModel() { } int TimeZoneModel::rowCount(const QModelIndex &parent) const { Q_UNUSED(parent); return m_data.count(); } QVariant TimeZoneModel::data(const QModelIndex &index, int role) const { if (index.isValid()) { TimeZoneData currentData = m_data.at(index.row()); switch (role) { case TimeZoneIdRole: return currentData.id; case RegionRole: return currentData.region; case CityRole: return currentData.city; case CommentRole: return currentData.comment; case CheckedRole: return currentData.checked; case IsLocalTimeZoneRole: return currentData.isLocalTimeZone; } } return QVariant(); } bool TimeZoneModel::setData(const QModelIndex &index, const QVariant &value, int role) { if (!index.isValid() || value.isNull()) { return false; } if (role == CheckedRole) { m_data[index.row()].checked = value.toBool(); Q_EMIT dataChanged(index, index); if (m_data[index.row()].checked) { m_selectedTimeZones.append(m_data[index.row()].id); m_offsetData.insert(m_data[index.row()].id, m_data[index.row()].offsetFromUtc); } else { m_selectedTimeZones.removeAll(m_data[index.row()].id); m_offsetData.remove(m_data[index.row()].id); } sortTimeZones(); Q_EMIT selectedTimeZonesChanged(); return true; } return false; } void TimeZoneModel::update() { beginResetModel(); m_data.clear(); TimeZoneData local; local.isLocalTimeZone = true; local.id = QStringLiteral("Local"); local.region = i18nc("This means \"Local Timezone\"", "Local"); local.city = m_timezonesI18n->i18nCity(QString::fromLocal8Bit(QTimeZone::systemTimeZoneId())); local.comment = i18n("System's local time zone"); local.checked = false; m_data.append(local); const QList systemTimeZones = QTimeZone::availableTimeZoneIds(); for (const QByteArray &id : systemTimeZones) { const QTimeZone zone(id); const QString continent = m_timezonesI18n->i18nContinents(QString::fromUtf8(QByteArrayView(id).mid(0, id.indexOf('/')))); TimeZoneData newData; newData.isLocalTimeZone = false; newData.id = QString::fromLocal8Bit(id); newData.city = m_timezonesI18n->i18nCity(newData.id); newData.region = zone.country() == QLocale::AnyCountry ? QString() : continent + QLatin1Char('/') + m_timezonesI18n->i18nCountry(zone.country()); newData.comment = zone.comment(); newData.checked = false; newData.offsetFromUtc = zone.offsetFromUtc(QDateTime::currentDateTimeUtc()); m_data.append(newData); } std::sort(m_data.begin() + 1, m_data.end(), [](const TimeZoneData &left, const TimeZoneData &right) { return left.city.compare(right.city) < 0; }); endResetModel(); } void TimeZoneModel::setSelectedTimeZones(const QStringList &selectedTimeZones) { m_selectedTimeZones = selectedTimeZones; for (int i = 0; i < m_data.size(); i++) { if (m_selectedTimeZones.contains(m_data.at(i).id)) { m_data[i].checked = true; m_offsetData.insert(m_data[i].id, m_data[i].offsetFromUtc); QModelIndex index = createIndex(i, 0); Q_EMIT dataChanged(index, index); } } sortTimeZones(); } void TimeZoneModel::selectLocalTimeZone() { m_data[0].checked = true; Q_EMIT dataChanged(index(0, 0), index(0, 0), QList{CheckedRole}); } QString TimeZoneModel::localTimeZoneCity() { return m_data[0].city; } void TimeZoneModel::slotUpdate() { update(); setProperty("selectedTimeZones", m_selectedTimeZones); } QHash TimeZoneModel::roleNames() const { return QHash({ {TimeZoneIdRole, "timeZoneId"}, {RegionRole, "region"}, {CityRole, "city"}, {CommentRole, "comment"}, {CheckedRole, "checked"}, {IsLocalTimeZoneRole, "isLocalTimeZone"}, }); } void TimeZoneModel::sortTimeZones() { std::sort(m_selectedTimeZones.begin(), m_selectedTimeZones.end(), [this](const QString &a, const QString &b) { return m_offsetData.value(a) < m_offsetData.value(b); }); }