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.
449 lines
14 KiB
449 lines
14 KiB
/* |
|
SPDX-FileCopyrightText: 2013 Mark Gaiser <markg85@gmail.com> |
|
SPDX-FileCopyrightText: 2016 Martin Klapetek <mklapetek@kde.org> |
|
SPDX-FileCopyrightText: 2021 Carl Schwan <carlschwan@kde.org> |
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later |
|
*/ |
|
|
|
#include "daysmodel.h" |
|
#include "eventdatadecorator.h" |
|
|
|
#include <QByteArray> |
|
#include <QDebug> |
|
#include <QDir> |
|
#include <QMetaObject> |
|
|
|
constexpr int maxEventDisplayed = 5; |
|
|
|
class DaysModelPrivate |
|
{ |
|
public: |
|
explicit DaysModelPrivate(); |
|
|
|
QList<DayData> *m_data = nullptr; |
|
QList<QObject *> m_qmlData; |
|
QMultiHash<QDate, CalendarEvents::EventData> m_eventsData; |
|
QHash<QDate /* Gregorian */, QDate> m_alternateDatesData; |
|
QHash<QDate, CalendarEvents::CalendarEventsPlugin::SubLabel> m_subLabelsData; |
|
|
|
QDate m_lastRequestedAgendaDate; |
|
bool m_agendaNeedsUpdate = false; |
|
|
|
std::unique_ptr<EventPluginsManager> m_pluginsManager; |
|
}; |
|
|
|
DaysModelPrivate::DaysModelPrivate() |
|
{ |
|
} |
|
|
|
DaysModel::DaysModel(QObject *parent) |
|
: QAbstractItemModel(parent) |
|
, d(new DaysModelPrivate) |
|
{ |
|
} |
|
|
|
DaysModel::~DaysModel() |
|
{ |
|
delete d; |
|
} |
|
|
|
void DaysModel::setSourceData(QList<DayData> *data) |
|
{ |
|
if (d->m_data != data) { |
|
beginResetModel(); |
|
d->m_data = data; |
|
endResetModel(); |
|
} |
|
} |
|
|
|
int DaysModel::rowCount(const QModelIndex &parent) const |
|
{ |
|
if (!parent.isValid()) { |
|
// day count |
|
if (d->m_data->size() <= 0) { |
|
return 0; |
|
} else { |
|
return d->m_data->size(); |
|
} |
|
} else { |
|
// event count |
|
const auto &eventDatas = data(parent, Roles::Events).value<QList<CalendarEvents::EventData>>(); |
|
Q_ASSERT(eventDatas.count() <= maxEventDisplayed); |
|
return eventDatas.count(); |
|
} |
|
} |
|
|
|
int DaysModel::columnCount(const QModelIndex &parent) const |
|
{ |
|
Q_UNUSED(parent) |
|
return 1; |
|
} |
|
|
|
QVariant DaysModel::data(const QModelIndex &index, int role) const |
|
{ |
|
if (!index.isValid()) { |
|
return {}; |
|
} |
|
|
|
const int row = index.row(); |
|
|
|
if (!index.parent().isValid()) { |
|
// Fetch days in month |
|
const DayData ¤tData = d->m_data->at(row); |
|
const QDate currentDate(currentData.yearNumber, currentData.monthNumber, currentData.dayNumber); |
|
|
|
switch (role) { |
|
case isCurrent: |
|
return currentData.isCurrent; |
|
case containsEventItems: |
|
return d->m_eventsData.contains(currentDate); |
|
case Events: |
|
return QVariant::fromValue(d->m_eventsData.values(currentDate)); |
|
case EventCount: |
|
return d->m_eventsData.values(currentDate).count(); |
|
case containsMajorEventItems: |
|
return hasMajorEventAtDate(currentDate); |
|
case containsMinorEventItems: |
|
return hasMinorEventAtDate(currentDate); |
|
case dayNumber: |
|
return currentData.dayNumber; |
|
case monthNumber: |
|
return currentData.monthNumber; |
|
case yearNumber: |
|
return currentData.yearNumber; |
|
default: |
|
break; |
|
} |
|
|
|
if (d->m_alternateDatesData.count(currentDate)) { |
|
switch (role) { |
|
case AlternateYearNumber: |
|
return d->m_alternateDatesData.value(currentDate).year(); |
|
case AlternateMonthNumber: |
|
return d->m_alternateDatesData.value(currentDate).month(); |
|
case AlternateDayNumber: |
|
return d->m_alternateDatesData.value(currentDate).day(); |
|
default: |
|
break; |
|
} |
|
} |
|
|
|
if (d->m_subLabelsData.count(currentDate)) { |
|
switch (role) { |
|
case SubLabel: |
|
return d->m_subLabelsData.value(currentDate).label; |
|
case SubYearLabel: |
|
return d->m_subLabelsData.value(currentDate).yearLabel; |
|
case SubMonthLabel: |
|
return d->m_subLabelsData.value(currentDate).monthLabel; |
|
case SubDayLabel: |
|
return d->m_subLabelsData.value(currentDate).dayLabel; |
|
default: |
|
break; |
|
} |
|
} |
|
} else { |
|
// Fetch event in day |
|
const auto &eventDatas = data(index.parent(), Roles::Events).value<QList<CalendarEvents::EventData>>(); |
|
if (eventDatas.count() < row) { |
|
return {}; |
|
} |
|
|
|
const auto &eventData = eventDatas[row]; |
|
switch (role) { |
|
case EventColor: |
|
return eventData.eventColor(); |
|
} |
|
} |
|
return {}; |
|
} |
|
|
|
void DaysModel::update() |
|
{ |
|
if (d->m_data->size() <= 0) { |
|
return; |
|
} |
|
|
|
// We need to reset the model since m_data has already been changed here |
|
// and we can't remove the events manually with beginRemoveRows() since |
|
// we don't know where the old events were located. |
|
beginResetModel(); |
|
d->m_eventsData.clear(); |
|
d->m_alternateDatesData.clear(); |
|
d->m_subLabelsData.clear(); |
|
endResetModel(); |
|
|
|
const QDate modelFirstDay(d->m_data->at(0).yearNumber, d->m_data->at(0).monthNumber, d->m_data->at(0).dayNumber); |
|
|
|
if (d->m_pluginsManager) { |
|
const auto plugins = d->m_pluginsManager->plugins(); |
|
for (CalendarEvents::CalendarEventsPlugin *eventsPlugin : plugins) { |
|
eventsPlugin->loadEventsForDateRange(modelFirstDay, modelFirstDay.addDays(42)); |
|
} |
|
} |
|
|
|
// We always have 42 items (or weeks * num of days in week) so we only have to tell the view that the data changed. |
|
Q_EMIT dataChanged(index(0, 0), index(d->m_data->count() - 1, 0)); |
|
} |
|
|
|
void DaysModel::onDataReady(const QMultiHash<QDate, CalendarEvents::EventData> &data) |
|
{ |
|
d->m_eventsData.reserve(d->m_eventsData.size() + data.size()); |
|
for (int i = 0; i < d->m_data->count(); i++) { |
|
const DayData ¤tData = d->m_data->at(i); |
|
const QDate currentDate(currentData.yearNumber, currentData.monthNumber, currentData.dayNumber); |
|
if (!data.values(currentDate).isEmpty()) { |
|
// Make sure we don't display more than maxEventDisplayed events. |
|
const int currentCount = d->m_eventsData.values(currentDate).count(); |
|
if (currentCount >= maxEventDisplayed) { |
|
break; |
|
} |
|
|
|
const int addedEventCount = std::min<int>(currentCount + data.values(currentDate).count(), maxEventDisplayed) - currentCount; |
|
|
|
// Add event |
|
beginInsertRows(index(i, 0), 0, addedEventCount - 1); |
|
int stopCounter = currentCount; |
|
for (const auto &dataDay : data.values(currentDate)) { |
|
if (stopCounter >= maxEventDisplayed) { |
|
break; |
|
} |
|
stopCounter++; |
|
d->m_eventsData.insert(currentDate, dataDay); |
|
} |
|
endInsertRows(); |
|
} |
|
} |
|
|
|
if (data.contains(QDate::currentDate())) { |
|
d->m_agendaNeedsUpdate = true; |
|
} |
|
|
|
// only the containsEventItems roles may have changed |
|
Q_EMIT dataChanged(index(0, 0), |
|
index(d->m_data->count() - 1, 0), |
|
{containsEventItems, containsMajorEventItems, containsMinorEventItems, Events, EventCount}); |
|
|
|
Q_EMIT agendaUpdated(QDate::currentDate()); |
|
} |
|
|
|
void DaysModel::onEventModified(const CalendarEvents::EventData &data) |
|
{ |
|
QList<QDate> updatesList; |
|
auto i = d->m_eventsData.begin(); |
|
while (i != d->m_eventsData.end()) { |
|
if (i->uid() == data.uid()) { |
|
*i = data; |
|
updatesList << i.key(); |
|
} |
|
|
|
++i; |
|
} |
|
|
|
if (!updatesList.isEmpty()) { |
|
d->m_agendaNeedsUpdate = true; |
|
} |
|
|
|
for (const QDate date : std::as_const(updatesList)) { |
|
const QModelIndex changedIndex = indexForDate(date); |
|
if (changedIndex.isValid()) { |
|
Q_EMIT dataChanged(changedIndex, changedIndex, {containsEventItems, containsMajorEventItems, containsMinorEventItems, EventColor}); |
|
} |
|
Q_EMIT agendaUpdated(date); |
|
} |
|
} |
|
|
|
void DaysModel::onEventRemoved(const QString &uid) |
|
{ |
|
// HACK We should update the model with beginRemoveRows instead of |
|
// using beginResetModel() since this creates a small visual glitches |
|
// if an event is removed in Korganizer and the calendar is open. |
|
// Using beginRemoveRows instead we make the code a lot more complex |
|
// and if not done correctly will introduce bugs. |
|
beginResetModel(); |
|
QList<QDate> updatesList; |
|
auto i = d->m_eventsData.begin(); |
|
while (i != d->m_eventsData.end()) { |
|
if (i->uid() == uid) { |
|
updatesList << i.key(); |
|
i = d->m_eventsData.erase(i); |
|
} else { |
|
++i; |
|
} |
|
} |
|
|
|
if (!updatesList.isEmpty()) { |
|
d->m_agendaNeedsUpdate = true; |
|
} |
|
|
|
for (const QDate date : std::as_const(updatesList)) { |
|
const QModelIndex changedIndex = indexForDate(date); |
|
if (changedIndex.isValid()) { |
|
Q_EMIT dataChanged(changedIndex, changedIndex, {containsEventItems, containsMajorEventItems, containsMinorEventItems}); |
|
} |
|
|
|
Q_EMIT agendaUpdated(date); |
|
} |
|
endResetModel(); |
|
} |
|
|
|
void DaysModel::onAlternateDateReady(const QHash<QDate, QDate> &data) |
|
{ |
|
d->m_alternateDatesData.reserve(d->m_alternateDatesData.size() + data.size()); |
|
for (int i = 0; i < d->m_data->count(); i++) { |
|
const DayData ¤tData = d->m_data->at(i); |
|
const QDate currentDate(currentData.yearNumber, currentData.monthNumber, currentData.dayNumber); |
|
if (data.count(currentDate) == 0) { |
|
continue; |
|
} |
|
// Add an alternate date |
|
d->m_alternateDatesData.insert(currentDate, data.value(currentDate)); |
|
} |
|
|
|
Q_EMIT dataChanged(index(0, 0), index(d->m_data->count() - 1, 0), {AlternateYearNumber, AlternateMonthNumber, AlternateDayNumber}); |
|
} |
|
|
|
void DaysModel::onSubLabelReady(const QHash<QDate, CalendarEvents::CalendarEventsPlugin::SubLabel> &data) |
|
{ |
|
d->m_subLabelsData.reserve(d->m_subLabelsData.size() + data.size()); |
|
for (int i = 0; i < d->m_data->count(); i++) { |
|
const DayData ¤tData = d->m_data->at(i); |
|
const QDate currentDate(currentData.yearNumber, currentData.monthNumber, currentData.dayNumber); |
|
if (data.count(currentDate) == 0) { |
|
continue; |
|
} |
|
// Add/Overwrite a sub-label based on priority |
|
if (const auto &value = data.value(currentDate); !d->m_subLabelsData.count(currentDate) |
|
|| (d->m_subLabelsData.count(currentDate) && value.priority > d->m_subLabelsData.value(currentDate).priority)) { |
|
d->m_subLabelsData.insert(currentDate, value); |
|
} |
|
} |
|
|
|
Q_EMIT dataChanged(index(0, 0), index(d->m_data->count() - 1, 0), {SubLabel, SubYearLabel, SubMonthLabel, SubDayLabel}); |
|
} |
|
|
|
QList<QObject *> DaysModel::eventsForDate(const QDate &date) |
|
{ |
|
if (d->m_lastRequestedAgendaDate == date && !d->m_agendaNeedsUpdate) { |
|
return d->m_qmlData; |
|
} |
|
|
|
d->m_lastRequestedAgendaDate = date; |
|
qDeleteAll(d->m_qmlData); |
|
d->m_qmlData.clear(); |
|
|
|
QList<CalendarEvents::EventData> events = d->m_eventsData.values(date); |
|
d->m_qmlData.reserve(events.size()); |
|
|
|
// sort events by their time and type |
|
std::sort(events.begin(), events.end(), [](const CalendarEvents::EventData &a, const CalendarEvents::EventData &b) { |
|
return b.type() > a.type() || b.startDateTime() > a.startDateTime(); |
|
}); |
|
|
|
for (const CalendarEvents::EventData &event : std::as_const(events)) { |
|
d->m_qmlData << new EventDataDecorator(event, this); |
|
} |
|
|
|
d->m_agendaNeedsUpdate = false; |
|
return d->m_qmlData; |
|
} |
|
|
|
QModelIndex DaysModel::indexForDate(const QDate &date) |
|
{ |
|
if (!d->m_data) { |
|
return QModelIndex(); |
|
} |
|
|
|
const DayData &firstDay = d->m_data->at(0); |
|
const QDate firstDate(firstDay.yearNumber, firstDay.monthNumber, firstDay.dayNumber); |
|
|
|
qint64 daysTo = firstDate.daysTo(date); |
|
|
|
return createIndex(daysTo, 0); |
|
} |
|
|
|
bool DaysModel::hasMajorEventAtDate(const QDate &date) const |
|
{ |
|
auto it = d->m_eventsData.find(date); |
|
while (it != d->m_eventsData.end() && it.key() == date) { |
|
if (!it.value().isMinor()) { |
|
return true; |
|
} |
|
++it; |
|
} |
|
return false; |
|
} |
|
|
|
bool DaysModel::hasMinorEventAtDate(const QDate &date) const |
|
{ |
|
auto it = d->m_eventsData.find(date); |
|
while (it != d->m_eventsData.end() && it.key() == date) { |
|
if (it.value().isMinor()) { |
|
return true; |
|
} |
|
++it; |
|
} |
|
return false; |
|
} |
|
|
|
void DaysModel::setPluginsManager(QObject *manager) |
|
{ |
|
EventPluginsManager *m = qobject_cast<EventPluginsManager *>(manager); |
|
|
|
if (!m) { |
|
return; |
|
} |
|
|
|
d->m_pluginsManager.reset(m); |
|
|
|
connect(d->m_pluginsManager.get(), &EventPluginsManager::dataReady, this, &DaysModel::onDataReady); |
|
connect(d->m_pluginsManager.get(), &EventPluginsManager::eventModified, this, &DaysModel::onEventModified); |
|
connect(d->m_pluginsManager.get(), &EventPluginsManager::eventRemoved, this, &DaysModel::onEventRemoved); |
|
connect(d->m_pluginsManager.get(), &EventPluginsManager::alternateDateReady, this, &DaysModel::onAlternateDateReady); |
|
connect(d->m_pluginsManager.get(), &EventPluginsManager::subLabelReady, this, &DaysModel::onSubLabelReady); |
|
connect(d->m_pluginsManager.get(), &EventPluginsManager::pluginsChanged, this, &DaysModel::update); |
|
|
|
QMetaObject::invokeMethod(this, "update", Qt::QueuedConnection); |
|
} |
|
|
|
QHash<int, QByteArray> DaysModel::roleNames() const |
|
{ |
|
return { |
|
{isCurrent, "isCurrent"}, |
|
{containsEventItems, "containsEventItems"}, |
|
{containsMajorEventItems, "containsMajorEventItems"}, |
|
{containsMinorEventItems, "containsMinorEventItems"}, |
|
{dayNumber, "dayNumber"}, |
|
{monthNumber, "monthNumber"}, |
|
{yearNumber, "yearNumber"}, |
|
{EventColor, "eventColor"}, |
|
{EventCount, "eventCount"}, |
|
{Events, "events"}, |
|
{AlternateYearNumber, "alternateYearNumber"}, |
|
{AlternateMonthNumber, "alternateMonthNumber"}, |
|
{AlternateDayNumber, "alternateDayNumber"}, |
|
{SubLabel, "subLabel"}, |
|
{SubYearLabel, "subYearLabel"}, |
|
{SubMonthLabel, "subMonthLabel"}, |
|
{SubDayLabel, "subDayLabel"}, |
|
}; |
|
} |
|
|
|
QModelIndex DaysModel::index(int row, int column, const QModelIndex &parent) const |
|
{ |
|
if (parent.isValid()) { |
|
return createIndex(row, column, (intptr_t)parent.row()); |
|
} |
|
return createIndex(row, column, nullptr); |
|
} |
|
|
|
QModelIndex DaysModel::parent(const QModelIndex &child) const |
|
{ |
|
if (child.internalId()) { |
|
return createIndex(child.internalId(), 0, nullptr); |
|
} |
|
return QModelIndex(); |
|
} |
|
|
|
Q_DECLARE_METATYPE(CalendarEvents::EventData)
|
|
|