Eventviews

agenda.cpp
1/*
2 SPDX-FileCopyrightText: 2001 Cornelius Schumacher <schumacher@kde.org>
3 SPDX-FileCopyrightText: 2003-2004 Reinhold Kainhofer <reinhold@kainhofer.com>
4 SPDX-FileCopyrightText: 2010 Klarälvdalens Datakonsult AB, a KDAB Group company <info@kdab.net>
5 SPDX-FileContributor: Kevin Krammer <krake@kdab.com>
6 SPDX-FileContributor: Sergio Martins <sergio@kdab.com>
7
8 Marcus Bains line.
9 SPDX-FileCopyrightText: 2001 Ali Rahimi <ali@mit.edu>
10
11 SPDX-License-Identifier: GPL-2.0-or-later WITH Qt-Commercial-exception-1.0
12*/
13#include "agenda.h"
14#include "agendaview.h"
15#include "prefs.h"
16
17#include <Akonadi/CalendarUtils>
18#include <Akonadi/IncidenceChanger>
19#include <CalendarSupport/Utils>
20
21#include <KCalendarCore/Incidence>
22
23#include <KCalUtils/RecurrenceActions>
24
25#include "calendarview_debug.h"
26#include <KMessageBox>
27
28#include <KLocalizedString>
29#include <QApplication>
30#include <QHash>
31#include <QLabel>
32#include <QMouseEvent>
33#include <QMultiHash>
34#include <QPainter>
35#include <QPointer>
36#include <QResizeEvent>
37#include <QScrollBar>
38#include <QTimer>
39#include <QWheelEvent>
40
41#include <chrono>
42#include <cmath>
43
44using namespace std::chrono_literals; // for fabs()
45
46using namespace EventViews;
47
48///////////////////////////////////////////////////////////////////////////////
49class EventViews::MarcusBainsPrivate
50{
51public:
52 MarcusBainsPrivate(EventView *eventView, Agenda *agenda)
53 : mEventView(eventView)
54 , mAgenda(agenda)
55 {
56 }
57
58 [[nodiscard]] int todayColumn() const;
59
60public:
61 EventView *const mEventView;
62 Agenda *const mAgenda;
63 QTimer *mTimer = nullptr;
64 QLabel *mTimeBox = nullptr; // Label showing the current time
65 QDateTime mOldDateTime;
66 int mOldTodayCol = -1;
67};
68
69int MarcusBainsPrivate::todayColumn() const
70{
71 const QDate currentDate = QDate::currentDate();
72
73 int col = 0;
74 const KCalendarCore::DateList dateList = mAgenda->dateList();
75 for (const QDate &date : dateList) {
76 if (date == currentDate) {
77 return QApplication::isRightToLeft() ? mAgenda->columns() - 1 - col : col;
78 }
79 ++col;
80 }
81
82 return -1;
83}
84
85MarcusBains::MarcusBains(EventView *eventView, Agenda *agenda)
86 : QFrame(agenda)
87 , d(new MarcusBainsPrivate(eventView, agenda))
88{
89 d->mTimeBox = new QLabel(d->mAgenda);
90 d->mTimeBox->setAlignment(Qt::AlignRight | Qt::AlignBottom);
91
92 d->mTimer = new QTimer(this);
93 d->mTimer->setSingleShot(true);
94 connect(d->mTimer, &QTimer::timeout, this, &MarcusBains::updateLocation);
95 d->mTimer->start(0);
96}
97
98MarcusBains::~MarcusBains() = default;
99
100void MarcusBains::updateLocation()
101{
102 updateLocationRecalc();
103}
104
105void MarcusBains::updateLocationRecalc(bool recalculate)
106{
107 const bool showSeconds = d->mEventView->preferences()->marcusBainsShowSeconds();
108 const QColor color = d->mEventView->preferences()->agendaMarcusBainsLineLineColor();
109
110 const QDateTime now = QDateTime::currentDateTime();
111 const QTime time = now.time();
112
113 if (now.date() != d->mOldDateTime.date()) {
114 recalculate = true; // New day
115 }
116 const int todayCol = recalculate ? d->todayColumn() : d->mOldTodayCol;
117
118 // Number of minutes since beginning of the day
119 const int minutes = time.hour() * 60 + time.minute();
120 const int minutesPerCell = 24 * 60 / d->mAgenda->rows();
121
122 d->mOldDateTime = now;
123 d->mOldTodayCol = todayCol;
124
125 int y = int(minutes * d->mAgenda->gridSpacingY() / minutesPerCell);
126 int x = int(d->mAgenda->gridSpacingX() * todayCol);
127
128 bool hideIt = !(d->mEventView->preferences()->marcusBainsEnabled());
129 if (!isHidden() && (hideIt || (todayCol < 0))) {
130 hide();
131 d->mTimeBox->hide();
132 return;
133 }
134
135 if (isHidden() && !hideIt) {
136 show();
137 d->mTimeBox->show();
138 }
139
140 /* Line */
141 // It seems logical to adjust the line width with the label's font weight
142 const int fw = d->mEventView->preferences()->agendaMarcusBainsLineFont().weight();
143 setLineWidth(1 + abs(fw - QFont::Normal) / QFont::Light);
145 QPalette pal = palette();
146 pal.setColor(QPalette::Window, color); // for Oxygen
147 pal.setColor(QPalette::WindowText, color); // for Plastique
148 setPalette(pal);
149 if (recalculate) {
150 setFixedSize(int(d->mAgenda->gridSpacingX()), 1);
151 }
152 move(x, y);
153 raise();
154
155 /* Label */
156 d->mTimeBox->setFont(d->mEventView->preferences()->agendaMarcusBainsLineFont());
157 QPalette pal1 = d->mTimeBox->palette();
158 pal1.setColor(QPalette::WindowText, color);
159 if (!d->mEventView->preferences()->useSystemColor()) {
160 pal1.setColor(QPalette::Window, d->mEventView->preferences()->agendaGridBackgroundColor());
161 } else {
163 }
164 d->mTimeBox->setPalette(pal1);
165 d->mTimeBox->setAutoFillBackground(true);
166 d->mTimeBox->setText(QLocale::system().toString(time, showSeconds ? QLocale::LongFormat : QLocale::ShortFormat));
167 d->mTimeBox->adjustSize();
168 if (y - d->mTimeBox->height() >= 0) {
169 y -= d->mTimeBox->height();
170 } else {
171 y++;
172 }
173 if (x - d->mTimeBox->width() + d->mAgenda->gridSpacingX() > 0) {
174 x += int(d->mAgenda->gridSpacingX() - d->mTimeBox->width() - 1);
175 } else {
176 x++;
177 }
178 d->mTimeBox->move(x, y);
179 d->mTimeBox->raise();
180
181 if (showSeconds || recalculate) {
182 d->mTimer->start(1s);
183 } else {
184 d->mTimer->start(1000 * (60 - time.second()));
185 }
186}
187
188////////////////////////////////////////////////////////////////////////////
189
190class EventViews::AgendaPrivate
191{
192public:
193 AgendaPrivate(AgendaView *agendaView, QScrollArea *scrollArea, int columns, int rows, int rowSize, bool isInteractive)
194 : mAgendaView(agendaView)
195 , mScrollArea(scrollArea)
196 , mAllDayMode(false)
197 , mColumns(columns)
198 , mRows(rows)
199 , mGridSpacingX(0.0)
200 , mGridSpacingY(rowSize)
201 , mDesiredGridSpacingY(rowSize)
202 , mChanger(nullptr)
203 , mResizeBorderWidth(0)
204 , mScrollBorderWidth(0)
205 , mScrollDelay(0)
206 , mScrollOffset(0)
207 , mWorkingHoursEnable(false)
208 , mHolidayMask(nullptr)
209 , mWorkingHoursYTop(0)
210 , mWorkingHoursYBottom(0)
211 , mHasSelection(false)
212 , mMarcusBains(nullptr)
213 , mActionType(Agenda::NOP)
214 , mItemMoved(false)
215 , mOldLowerScrollValue(0)
216 , mOldUpperScrollValue(0)
217 , mReturnPressed(false)
218 , mIsInteractive(isInteractive)
219 {
220 if (mGridSpacingY < 4 || mGridSpacingY > 30) {
221 mGridSpacingY = 10;
222 }
223 }
224
225public:
226 PrefsPtr preferences() const
227 {
228 return mAgendaView->preferences();
229 }
230
231 bool isQueuedForDeletion(const QString &uid) const
232 {
233 // if mAgendaItemsById contains it it means that a createAgendaItem() was called
234 // before the previous agenda items were deleted.
235 return mItemsQueuedForDeletion.contains(uid) && !mAgendaItemsById.contains(uid);
236 }
237
238 QMultiHash<QString, AgendaItem::QPtr> mAgendaItemsById; // A QMultiHash because recurring incs
239 // might have many agenda items
240 QSet<QString> mItemsQueuedForDeletion;
241
242 AgendaView *mAgendaView = nullptr;
243 QScrollArea *mScrollArea = nullptr;
244
245 bool mAllDayMode;
246
247 // Number of Columns/Rows of agenda grid
248 int mColumns;
249 int mRows;
250
251 // Width and height of agenda cells. mDesiredGridSpacingY is the height
252 // set in the config. The actual height might be larger since otherwise
253 // more than 24 hours might be displayed.
254 double mGridSpacingX;
255 double mGridSpacingY;
256 double mDesiredGridSpacingY;
257
258 Akonadi::IncidenceChanger *mChanger = nullptr;
259
260 // size of border, where mouse action will resize the AgendaItem
261 int mResizeBorderWidth;
262
263 // size of border, where mouse mve will cause a scroll of the agenda
264 int mScrollBorderWidth;
265 int mScrollDelay;
266 int mScrollOffset;
267
268 QTimer mScrollUpTimer;
269 QTimer mScrollDownTimer;
270
271 // Cells to store Move and Resize coordinates while performing the action
272 QPoint mStartCell;
273 QPoint mEndCell;
274
275 // Working Hour coordinates
276 bool mWorkingHoursEnable;
277 QList<bool> *mHolidayMask = nullptr;
278 int mWorkingHoursYTop;
279 int mWorkingHoursYBottom;
280
281 // Selection
282 bool mHasSelection;
283 QPoint mSelectionStartPoint;
284 QPoint mSelectionStartCell;
285 QPoint mSelectionEndCell;
286
287 // List of dates to be displayed
288 KCalendarCore::DateList mSelectedDates;
289
290 // The AgendaItem, which has been right-clicked last
291 QPointer<AgendaItem> mClickedItem;
292
293 // The AgendaItem, which is being moved/resized
294 QPointer<AgendaItem> mActionItem;
295
296 // Currently selected item
297 QPointer<AgendaItem> mSelectedItem;
298 // Uid of the last selected incidence. Used for reselecting in situations
299 // where the selected item points to a no longer valid incidence, for
300 // example during resource reload.
301 QString mSelectedId;
302
303 // The Marcus Bains Line widget.
304 MarcusBains *mMarcusBains = nullptr;
305
306 Agenda::MouseActionType mActionType;
307
308 bool mItemMoved;
309
310 // List of all Items contained in agenda
311 QList<AgendaItem::QPtr> mItems;
312 QList<AgendaItem::QPtr> mItemsToDelete;
313
314 int mOldLowerScrollValue;
315 int mOldUpperScrollValue;
316
317 bool mReturnPressed;
318 bool mIsInteractive;
319
320 MultiViewCalendar::Ptr mCalendar;
321};
322
323/*
324 Create an agenda widget with rows rows and columns columns.
325*/
326Agenda::Agenda(AgendaView *agendaView, QScrollArea *scrollArea, int columns, int rows, int rowSize, bool isInteractive)
327 : QWidget(scrollArea)
328 , d(new AgendaPrivate(agendaView, scrollArea, columns, rows, rowSize, isInteractive))
329{
330 setMouseTracking(true);
331
332 init();
333}
334
335/*
336 Create an agenda widget with columns columns and one row. This is used for
337 all-day events.
338*/
339Agenda::Agenda(AgendaView *agendaView, QScrollArea *scrollArea, int columns, bool isInteractive)
340 : QWidget(scrollArea)
341 , d(new AgendaPrivate(agendaView, scrollArea, columns, 1, 24, isInteractive))
342{
343 d->mAllDayMode = true;
344
345 init();
346}
347
348Agenda::~Agenda()
349{
350 delete d->mMarcusBains;
351}
352
353KCalendarCore::Incidence::Ptr Agenda::selectedIncidence() const
354{
355 return d->mSelectedItem ? d->mSelectedItem->incidence() : KCalendarCore::Incidence::Ptr();
356}
357
358QDate Agenda::selectedIncidenceDate() const
359{
360 return d->mSelectedItem ? d->mSelectedItem->occurrenceDate() : QDate();
361}
362
363QString Agenda::lastSelectedItemUid() const
364{
365 return d->mSelectedId;
366}
367
368void Agenda::init()
369{
371
372 d->mGridSpacingX = static_cast<double>(d->mScrollArea->width()) / d->mColumns;
373 d->mDesiredGridSpacingY = d->preferences()->hourSize();
374 if (d->mDesiredGridSpacingY < 4 || d->mDesiredGridSpacingY > 30) {
375 d->mDesiredGridSpacingY = 10;
376 }
377
378 // make sure that there are not more than 24 per day
379 d->mGridSpacingY = static_cast<double>(height()) / d->mRows;
380 if (d->mGridSpacingY < d->mDesiredGridSpacingY) {
381 d->mGridSpacingY = d->mDesiredGridSpacingY;
382 }
383
384 d->mResizeBorderWidth = 12;
385 d->mScrollBorderWidth = 12;
386 d->mScrollDelay = 30;
387 d->mScrollOffset = 10;
388
389 // Grab key strokes for keyboard navigation of agenda. Seems to have no
390 // effect. Has to be fixed.
392
393 connect(&d->mScrollUpTimer, &QTimer::timeout, this, &Agenda::scrollUp);
394 connect(&d->mScrollDownTimer, &QTimer::timeout, this, &Agenda::scrollDown);
395
396 d->mStartCell = QPoint(0, 0);
397 d->mEndCell = QPoint(0, 0);
398
399 d->mHasSelection = false;
400 d->mSelectionStartPoint = QPoint(0, 0);
401 d->mSelectionStartCell = QPoint(0, 0);
402 d->mSelectionEndCell = QPoint(0, 0);
403
404 d->mOldLowerScrollValue = -1;
405 d->mOldUpperScrollValue = -1;
406
407 d->mClickedItem = nullptr;
408
409 d->mActionItem = nullptr;
410 d->mActionType = NOP;
411 d->mItemMoved = false;
412
413 d->mSelectedItem = nullptr;
414
415 setAcceptDrops(true);
416 installEventFilter(this);
417
418 /* resizeContents(int(mGridSpacingX * mColumns), int(mGridSpacingY * mRows)); */
419
420 d->mScrollArea->viewport()->update();
421 // mScrollArea->viewport()->setAttribute(Qt::WA_NoSystemBackground, true);
422 d->mScrollArea->viewport()->setFocusPolicy(Qt::WheelFocus);
423
424 calculateWorkingHours();
425
426 connect(verticalScrollBar(), &QScrollBar::valueChanged, this, qOverload<int>(&Agenda::checkScrollBoundaries));
427
428 // Create the Marcus Bains line.
429 if (d->mAllDayMode) {
430 d->mMarcusBains = nullptr;
431 } else {
432 d->mMarcusBains = new MarcusBains(d->mAgendaView, this);
433 }
434}
435
436void Agenda::clear()
437{
438 qDeleteAll(d->mItems);
439 qDeleteAll(d->mItemsToDelete);
440 d->mItems.clear();
441 d->mItemsToDelete.clear();
442 d->mAgendaItemsById.clear();
443 d->mItemsQueuedForDeletion.clear();
444
445 d->mSelectedItem = nullptr;
446
447 clearSelection();
448}
449
450void Agenda::clearSelection()
451{
452 d->mHasSelection = false;
453 d->mActionType = NOP;
454 update();
455}
456
457void Agenda::marcus_bains()
458{
459 if (d->mMarcusBains) {
460 d->mMarcusBains->updateLocationRecalc(true);
461 }
462}
463
464void Agenda::changeColumns(int columns)
465{
466 if (columns == 0) {
467 qCDebug(CALENDARVIEW_LOG) << "called with argument 0";
468 return;
469 }
470
471 clear();
472 d->mColumns = columns;
473 // setMinimumSize(mColumns * 10, mGridSpacingY + 1);
474 // init();
475 // update();
476
477 QResizeEvent event(size(), size());
478
480}
481
482int Agenda::columns() const
483{
484 return d->mColumns;
485}
486
487int Agenda::rows() const
488{
489 return d->mRows;
490}
491
492double Agenda::gridSpacingX() const
493{
494 return d->mGridSpacingX;
495}
496
497double Agenda::gridSpacingY() const
498{
499 return d->mGridSpacingY;
500}
501
502/*
503 This is the eventFilter function, which gets all events from the AgendaItems
504 contained in the agenda. It has to handle moving and resizing for all items.
505*/
506bool Agenda::eventFilter(QObject *object, QEvent *event)
507{
508 switch (event->type()) {
513 return eventFilter_mouse(object, static_cast<QMouseEvent *>(event));
514#ifndef QT_NO_WHEELEVENT
515 case QEvent::Wheel:
516 return eventFilter_wheel(object, static_cast<QWheelEvent *>(event));
517#endif
518 case QEvent::KeyPress:
520 return eventFilter_key(object, static_cast<QKeyEvent *>(event));
521
522 case QEvent::Leave:
523#ifndef QT_NO_CURSOR
524 if (!d->mActionItem) {
526 }
527#endif
528
529 if (object == this) {
530 // so timelabels hide the mouse cursor
531 Q_EMIT leaveAgenda();
532 }
533 return true;
534
535 case QEvent::Enter:
536 Q_EMIT enterAgenda();
537 return QWidget::eventFilter(object, event);
538
539#ifndef QT_NO_DRAGANDDROP
541 case QEvent::DragMove:
543 case QEvent::Drop:
544 // case QEvent::DragResponse:
545 return eventFilter_drag(object, static_cast<QDropEvent *>(event));
546#endif
547
548 default:
549 return QWidget::eventFilter(object, event);
550 }
551}
552
553bool Agenda::eventFilter_drag(QObject *obj, QDropEvent *de)
554{
555#ifndef QT_NO_DRAGANDDROP
556 const QMimeData *md = de->mimeData();
557
558 switch (de->type()) {
560 case QEvent::DragMove:
561 if (!CalendarSupport::canDecode(md)) {
562 return false;
563 }
564
565 if (CalendarSupport::mimeDataHasIncidence(md)) {
566 de->accept();
567 } else {
568 de->ignore();
569 }
570 return true;
571 break;
573 return false;
574 break;
575 case QEvent::Drop: {
576 if (!CalendarSupport::canDecode(md)) {
577 return false;
578 }
579
580 const QList<QUrl> incidenceUrls = CalendarSupport::incidenceItemUrls(md);
581 const KCalendarCore::Incidence::List incidences = CalendarSupport::incidences(md);
582
583 Q_ASSERT(!incidenceUrls.isEmpty() || !incidences.isEmpty());
584
586
587 QWidget *dropTarget = qobject_cast<QWidget *>(obj);
588 QPoint dropPosition = de->position().toPoint();
589 if (dropTarget && dropTarget != this) {
590 dropPosition = dropTarget->mapTo(this, dropPosition);
591 }
592
593 const QPoint gridPosition = contentsToGrid(dropPosition);
594 if (!incidenceUrls.isEmpty()) {
595 Q_EMIT droppedIncidences(incidenceUrls, gridPosition, d->mAllDayMode);
596 } else {
597 Q_EMIT droppedIncidences(incidences, gridPosition, d->mAllDayMode);
598 }
599 return true;
600 }
601
603 default:
604 break;
605 }
606#endif
607 return false;
608}
609
610#ifndef QT_NO_WHEELEVENT
611bool Agenda::eventFilter_wheel(QObject *object, QWheelEvent *e)
612{
613 QPoint viewportPos;
614 bool accepted = false;
615 const QPoint pos = e->position().toPoint();
617 if (object != this) {
618 viewportPos = ((QWidget *)object)->mapToParent(pos);
619 } else {
620 viewportPos = pos;
621 }
622 // qCDebug(CALENDARVIEW_LOG) << type:" << e->type() << "angleDelta:" << e->angleDelta();
623 Q_EMIT zoomView(-e->angleDelta().y(), contentsToGrid(viewportPos), Qt::Horizontal);
624 accepted = true;
625 }
626
628 if (object != this) {
629 viewportPos = ((QWidget *)object)->mapToParent(pos);
630 } else {
631 viewportPos = pos;
632 }
633 Q_EMIT zoomView(-e->angleDelta().y(), contentsToGrid(viewportPos), Qt::Vertical);
634 Q_EMIT mousePosSignal(gridToContents(contentsToGrid(viewportPos)));
635 accepted = true;
636 }
637 if (accepted) {
638 e->accept();
639 }
640 return accepted;
641}
642
643#endif
644
645bool Agenda::eventFilter_key(QObject *, QKeyEvent *ke)
646{
647 return d->mAgendaView->processKeyEvent(ke);
648}
649
650bool Agenda::eventFilter_mouse(QObject *object, QMouseEvent *me)
651{
652 QPoint viewportPos;
653 if (object != this) {
654 viewportPos = static_cast<QWidget *>(object)->mapToParent(me->pos());
655 } else {
656 viewportPos = me->pos();
657 }
658
659 switch (me->type()) {
661 if (object != this) {
662 if (me->button() == Qt::RightButton) {
663 d->mClickedItem = qobject_cast<AgendaItem *>(object);
664 if (d->mClickedItem) {
665 selectItem(d->mClickedItem);
666 Q_EMIT showIncidencePopupSignal(d->mClickedItem->incidence(), d->mClickedItem->occurrenceDate());
667 }
668 } else {
669 AgendaItem::QPtr item = qobject_cast<AgendaItem *>(object);
670 if (item) {
671 KCalendarCore::Incidence::Ptr incidence = item->incidence();
672 if (incidence->isReadOnly()) {
673 d->mActionItem = nullptr;
674 } else {
675 d->mActionItem = item;
676 startItemAction(viewportPos);
677 }
678 // Warning: do selectItem() as late as possible, since all
679 // sorts of things happen during this call. Some can lead to
680 // this filter being run again and mActionItem being set to
681 // null.
682 selectItem(item);
683 }
684 }
685 } else {
686 if (me->button() == Qt::RightButton) {
687 // if mouse pointer is not in selection, select the cell below the cursor
688 QPoint gpos = contentsToGrid(viewportPos);
689 if (!ptInSelection(gpos)) {
690 d->mSelectionStartCell = gpos;
691 d->mSelectionEndCell = gpos;
692 d->mHasSelection = true;
693 Q_EMIT newStartSelectSignal();
694 Q_EMIT newTimeSpanSignal(d->mSelectionStartCell, d->mSelectionEndCell);
695 // updateContents();
696 }
697 Q_EMIT showNewEventPopupSignal();
698 } else {
699 selectItem(nullptr);
700 d->mActionItem = nullptr;
701#ifndef QT_NO_CURSOR
703#endif
704 startSelectAction(viewportPos);
705 update();
706 }
707 }
708 break;
709
711 if (d->mActionItem) {
712 endItemAction();
713 } else if (d->mActionType == SELECT) {
714 endSelectAction(viewportPos);
715 }
716 // This nasty gridToContents(contentsToGrid(..)) is needed to
717 // avoid an offset of a few pixels. Don't ask me why...
718 Q_EMIT mousePosSignal(gridToContents(contentsToGrid(viewportPos)));
719 break;
720
721 case QEvent::MouseMove: {
722 if (!d->mIsInteractive) {
723 return true;
724 }
725
726 // This nasty gridToContents(contentsToGrid(..)) is needed todos
727 // avoid an offset of a few pixels. Don't ask me why...
728 QPoint indicatorPos = gridToContents(contentsToGrid(viewportPos));
729 if (object != this) {
730 AgendaItem::QPtr moveItem = qobject_cast<AgendaItem *>(object);
731 KCalendarCore::Incidence::Ptr incidence = moveItem ? moveItem->incidence() : KCalendarCore::Incidence::Ptr();
732 if (incidence && !incidence->isReadOnly()) {
733 if (!d->mActionItem) {
734 setNoActionCursor(moveItem, viewportPos);
735 } else {
736 performItemAction(viewportPos);
737
738 if (d->mActionType == MOVE) {
739 // show cursor at the current begin of the item
740 AgendaItem::QPtr firstItem = d->mActionItem->firstMultiItem();
741 if (!firstItem) {
742 firstItem = d->mActionItem;
743 }
744 indicatorPos = gridToContents(QPoint(firstItem->cellXLeft(), firstItem->cellYTop()));
745 } else if (d->mActionType == RESIZEBOTTOM) {
746 // RESIZETOP is handled correctly, only resizebottom works differently
747 indicatorPos = gridToContents(QPoint(d->mActionItem->cellXLeft(), d->mActionItem->cellYBottom() + 1));
748 }
749 } // If we have an action item
750 } // If move item && !read only
751 } else {
752 if (d->mActionType == SELECT) {
753 performSelectAction(viewportPos);
754
755 // show cursor at end of timespan
756 if (((d->mStartCell.y() < d->mEndCell.y()) && (d->mEndCell.x() >= d->mStartCell.x())) || (d->mEndCell.x() > d->mStartCell.x())) {
757 indicatorPos = gridToContents(QPoint(d->mEndCell.x(), d->mEndCell.y() + 1));
758 } else {
759 indicatorPos = gridToContents(d->mEndCell);
760 }
761 }
762 }
763 Q_EMIT mousePosSignal(indicatorPos);
764 break;
765 }
766
768 if (object == this) {
769 selectItem(nullptr);
770 Q_EMIT newEventSignal();
771 } else {
772 AgendaItem::QPtr doubleClickedItem = qobject_cast<AgendaItem *>(object);
773 if (doubleClickedItem) {
774 selectItem(doubleClickedItem);
775 Q_EMIT editIncidenceSignal(doubleClickedItem->incidence());
776 }
777 }
778 break;
779
780 default:
781 break;
782 }
783
784 return true;
785}
786
787bool Agenda::ptInSelection(QPoint gpos) const
788{
789 if (!d->mHasSelection) {
790 return false;
791 } else if (gpos.x() < d->mSelectionStartCell.x() || gpos.x() > d->mSelectionEndCell.x()) {
792 return false;
793 } else if ((gpos.x() == d->mSelectionStartCell.x()) && (gpos.y() < d->mSelectionStartCell.y())) {
794 return false;
795 } else if ((gpos.x() == d->mSelectionEndCell.x()) && (gpos.y() > d->mSelectionEndCell.y())) {
796 return false;
797 }
798 return true;
799}
800
801void Agenda::startSelectAction(QPoint viewportPos)
802{
803 Q_EMIT newStartSelectSignal();
804
805 d->mActionType = SELECT;
806 d->mSelectionStartPoint = viewportPos;
807 d->mHasSelection = true;
808
809 QPoint pos = viewportPos;
810 QPoint gpos = contentsToGrid(pos);
811
812 // Store new selection
813 d->mStartCell = gpos;
814 d->mEndCell = gpos;
815 d->mSelectionStartCell = gpos;
816 d->mSelectionEndCell = gpos;
817
818 // updateContents();
819}
820
821void Agenda::performSelectAction(QPoint pos)
822{
823 const QPoint gpos = contentsToGrid(pos);
824
825 // Scroll if cursor was moved to upper or lower end of agenda.
826 if (pos.y() - contentsY() < d->mScrollBorderWidth && contentsY() > 0) {
827 d->mScrollUpTimer.start(d->mScrollDelay);
828 } else if (contentsY() + d->mScrollArea->viewport()->height() - d->mScrollBorderWidth < pos.y()) {
829 d->mScrollDownTimer.start(d->mScrollDelay);
830 } else {
831 d->mScrollUpTimer.stop();
832 d->mScrollDownTimer.stop();
833 }
834
835 if (gpos != d->mEndCell) {
836 d->mEndCell = gpos;
837 if (d->mStartCell.x() > d->mEndCell.x() || (d->mStartCell.x() == d->mEndCell.x() && d->mStartCell.y() > d->mEndCell.y())) {
838 // backward selection
839 d->mSelectionStartCell = d->mEndCell;
840 d->mSelectionEndCell = d->mStartCell;
841 } else {
842 d->mSelectionStartCell = d->mStartCell;
843 d->mSelectionEndCell = d->mEndCell;
844 }
845
846 update();
847 }
848}
849
850void Agenda::endSelectAction(const QPoint &currentPos)
851{
852 d->mScrollUpTimer.stop();
853 d->mScrollDownTimer.stop();
854
855 d->mActionType = NOP;
856
857 Q_EMIT newTimeSpanSignal(d->mSelectionStartCell, d->mSelectionEndCell);
858
859 if (d->preferences()->selectionStartsEditor()) {
860 if ((d->mSelectionStartPoint - currentPos).manhattanLength() > QApplication::startDragDistance()) {
861 Q_EMIT newEventSignal();
862 }
863 }
864}
865
866Agenda::MouseActionType Agenda::isInResizeArea(bool horizontal, QPoint pos, const AgendaItem::QPtr &item)
867{
868 if (!item) {
869 return NOP;
870 }
871 QPoint gridpos = contentsToGrid(pos);
872 QPoint contpos = gridToContents(gridpos + QPoint((QApplication::isRightToLeft()) ? 1 : 0, 0));
873
874 if (horizontal) {
875 int clXLeft = item->cellXLeft();
876 int clXRight = item->cellXRight();
878 int tmp = clXLeft;
879 clXLeft = clXRight;
880 clXRight = tmp;
881 }
882 int gridDistanceX = int(pos.x() - contpos.x());
883 if (gridDistanceX < d->mResizeBorderWidth && clXLeft == gridpos.x()) {
885 return RESIZERIGHT;
886 } else {
887 return RESIZELEFT;
888 }
889 } else if ((d->mGridSpacingX - gridDistanceX) < d->mResizeBorderWidth && clXRight == gridpos.x()) {
891 return RESIZELEFT;
892 } else {
893 return RESIZERIGHT;
894 }
895 } else {
896 return MOVE;
897 }
898 } else {
899 int gridDistanceY = int(pos.y() - contpos.y());
900 if (gridDistanceY < d->mResizeBorderWidth && item->cellYTop() == gridpos.y() && !item->firstMultiItem()) {
901 return RESIZETOP;
902 } else if ((d->mGridSpacingY - gridDistanceY) < d->mResizeBorderWidth && item->cellYBottom() == gridpos.y() && !item->lastMultiItem()) {
903 return RESIZEBOTTOM;
904 } else {
905 return MOVE;
906 }
907 }
908}
909
910void Agenda::startItemAction(const QPoint &pos)
911{
912 Q_ASSERT(d->mActionItem);
913
914 d->mStartCell = contentsToGrid(pos);
915 d->mEndCell = d->mStartCell;
916
917 bool noResize = CalendarSupport::hasTodo(d->mActionItem->incidence());
918
919 d->mActionType = MOVE;
920 if (!noResize) {
921 d->mActionType = isInResizeArea(d->mAllDayMode, pos, d->mActionItem);
922 }
923
924 d->mActionItem->startMove();
925 setActionCursor(d->mActionType, true);
926}
927
928void Agenda::performItemAction(QPoint pos)
929{
930 QPoint gpos = contentsToGrid(pos);
931
932 // Cursor left active agenda area.
933 // This starts a drag.
934 if (pos.y() < 0 || pos.y() >= contentsY() + d->mScrollArea->viewport()->height() || pos.x() < 0 || pos.x() >= width()) {
935 if (d->mActionType == MOVE) {
936 d->mScrollUpTimer.stop();
937 d->mScrollDownTimer.stop();
938 d->mActionItem->resetMove();
939 placeSubCells(d->mActionItem);
940 Q_EMIT startDragSignal(d->mActionItem->incidence());
941#ifndef QT_NO_CURSOR
943#endif
944 if (d->mChanger) {
945 // d->mChanger->cancelChange(d->mActionItem->incidence());
946 }
947 d->mActionItem = nullptr;
948 d->mActionType = NOP;
949 d->mItemMoved = false;
950 return;
951 }
952 } else {
953 setActionCursor(d->mActionType, true);
954 }
955
956 // Scroll if item was moved to upper or lower end of agenda.
957 const int distanceToTop = pos.y() - contentsY();
958 if (distanceToTop < d->mScrollBorderWidth && distanceToTop > -d->mScrollBorderWidth) {
959 d->mScrollUpTimer.start(d->mScrollDelay);
960 } else if (contentsY() + d->mScrollArea->viewport()->height() - d->mScrollBorderWidth < pos.y()) {
961 d->mScrollDownTimer.start(d->mScrollDelay);
962 } else {
963 d->mScrollUpTimer.stop();
964 d->mScrollDownTimer.stop();
965 }
966
967 // Move or resize item if necessary
968 if (d->mEndCell != gpos) {
969 if (!d->mItemMoved) {
970 if (!d->mChanger) {
972 i18n("Unable to lock item for modification. "
973 "You cannot make any changes."),
974 i18nc("@title:window", "Locking Failed"),
975 QStringLiteral("AgendaLockingFailed"));
976 d->mScrollUpTimer.stop();
977 d->mScrollDownTimer.stop();
978 d->mActionItem->resetMove();
979 placeSubCells(d->mActionItem);
980#ifndef QT_NO_CURSOR
982#endif
983 d->mActionItem = nullptr;
984 d->mActionType = NOP;
985 d->mItemMoved = false;
986 return;
987 }
988 d->mItemMoved = true;
989 }
990 d->mActionItem->raise();
991 if (d->mActionType == MOVE) {
992 // Move all items belonging to a multi item
993 AgendaItem::QPtr firstItem = d->mActionItem->firstMultiItem();
994 if (!firstItem) {
995 firstItem = d->mActionItem;
996 }
997 AgendaItem::QPtr lastItem = d->mActionItem->lastMultiItem();
998 if (!lastItem) {
999 lastItem = d->mActionItem;
1000 }
1001 QPoint deltapos = gpos - d->mEndCell;
1002 AgendaItem::QPtr moveItem = firstItem;
1003 while (moveItem) {
1004 bool changed = false;
1005 if (deltapos.x() != 0) {
1006 moveItem->moveRelative(deltapos.x(), 0);
1007 changed = true;
1008 }
1009 // in all day view don't try to move multi items, since there are none
1010 if (moveItem == firstItem && !d->mAllDayMode) { // is the first item
1011 int newY = deltapos.y() + moveItem->cellYTop();
1012 // If event start moved earlier than 0:00, it starts the previous day
1013 if (newY < 0 && newY > d->mScrollBorderWidth) {
1014 moveItem->expandTop(-moveItem->cellYTop());
1015 // prepend a new item at (x-1, rows()+newY to rows())
1016 AgendaItem::QPtr newFirst = firstItem->prevMoveItem();
1017 // cell's y values are first and last cell of the bar,
1018 // so if newY=-1, they need to be the same
1019 if (newFirst) {
1020 newFirst->setCellXY(moveItem->cellXLeft() - 1, rows() + newY, rows() - 1);
1021 d->mItems.append(newFirst);
1022 moveItem->resize(int(d->mGridSpacingX * newFirst->cellWidth()), int(d->mGridSpacingY * newFirst->cellHeight()));
1023 QPoint cpos = gridToContents(QPoint(newFirst->cellXLeft(), newFirst->cellYTop()));
1024 newFirst->setParent(this);
1025 newFirst->move(cpos.x(), cpos.y());
1026 } else {
1027 newFirst = insertItem(moveItem->incidence(),
1028 moveItem->occurrenceDateTime(),
1029 moveItem->cellXLeft() - 1,
1030 rows() + newY,
1031 rows() - 1,
1032 moveItem->itemPos(),
1033 moveItem->itemCount(),
1034 false);
1035 }
1036 if (newFirst) {
1037 newFirst->show();
1038 }
1039 moveItem->prependMoveItem(newFirst);
1040 firstItem = newFirst;
1041 } else if (newY >= rows()) {
1042 // If event start is moved past 24:00, it starts the next day
1043 // erase current item (i.e. remove it from the multiItem list)
1044 firstItem = moveItem->nextMultiItem();
1045 moveItem->hide();
1046 d->mItems.removeAll(moveItem);
1047 // removeChild(moveItem);
1048 d->mActionItem->removeMoveItem(moveItem);
1049 moveItem = firstItem;
1050 // adjust next day's item
1051 if (moveItem) {
1052 moveItem->expandTop(rows() - newY);
1053 }
1054 } else {
1055 moveItem->expandTop(deltapos.y(), true);
1056 }
1057 changed = true;
1058 }
1059 if (moveItem && !moveItem->lastMultiItem() && !d->mAllDayMode) { // is the last item
1060 int newY = deltapos.y() + moveItem->cellYBottom();
1061 if (newY < 0) {
1062 // erase current item
1063 lastItem = moveItem->prevMultiItem();
1064 moveItem->hide();
1065 d->mItems.removeAll(moveItem);
1066 // removeChild(moveItem);
1067 moveItem->removeMoveItem(moveItem);
1068 moveItem = lastItem;
1069 moveItem->expandBottom(newY + 1);
1070 } else if (newY >= rows()) {
1071 moveItem->expandBottom(rows() - moveItem->cellYBottom() - 1);
1072 // append item at (x+1, 0 to newY-rows())
1073 AgendaItem::QPtr newLast = lastItem->nextMoveItem();
1074 if (newLast) {
1075 newLast->setCellXY(moveItem->cellXLeft() + 1, 0, newY - rows() - 1);
1076 d->mItems.append(newLast);
1077 moveItem->resize(int(d->mGridSpacingX * newLast->cellWidth()), int(d->mGridSpacingY * newLast->cellHeight()));
1078 QPoint cpos = gridToContents(QPoint(newLast->cellXLeft(), newLast->cellYTop()));
1079 newLast->setParent(this);
1080 newLast->move(cpos.x(), cpos.y());
1081 } else {
1082 newLast = insertItem(moveItem->incidence(),
1083 moveItem->occurrenceDateTime(),
1084 moveItem->cellXLeft() + 1,
1085 0,
1086 newY - rows() - 1,
1087 moveItem->itemPos(),
1088 moveItem->itemCount(),
1089 false);
1090 }
1091 moveItem->appendMoveItem(newLast);
1092 newLast->show();
1093 lastItem = newLast;
1094 } else {
1095 moveItem->expandBottom(deltapos.y());
1096 }
1097 changed = true;
1098 }
1099 if (changed) {
1100 adjustItemPosition(moveItem);
1101 }
1102 if (moveItem) {
1103 moveItem = moveItem->nextMultiItem();
1104 }
1105 }
1106 } else if (d->mActionType == RESIZETOP) {
1107 if (d->mEndCell.y() <= d->mActionItem->cellYBottom()) {
1108 d->mActionItem->expandTop(gpos.y() - d->mEndCell.y());
1109 adjustItemPosition(d->mActionItem);
1110 }
1111 } else if (d->mActionType == RESIZEBOTTOM) {
1112 if (d->mEndCell.y() >= d->mActionItem->cellYTop()) {
1113 d->mActionItem->expandBottom(gpos.y() - d->mEndCell.y());
1114 adjustItemPosition(d->mActionItem);
1115 }
1116 } else if (d->mActionType == RESIZELEFT) {
1117 if (d->mEndCell.x() <= d->mActionItem->cellXRight()) {
1118 d->mActionItem->expandLeft(gpos.x() - d->mEndCell.x());
1119 adjustItemPosition(d->mActionItem);
1120 }
1121 } else if (d->mActionType == RESIZERIGHT) {
1122 if (d->mEndCell.x() >= d->mActionItem->cellXLeft()) {
1123 d->mActionItem->expandRight(gpos.x() - d->mEndCell.x());
1124 adjustItemPosition(d->mActionItem);
1125 }
1126 }
1127 d->mEndCell = gpos;
1128 }
1129}
1130
1131void Agenda::endItemAction()
1132{
1133 // PENDING(AKONADI_PORT) review all this cloning and changer calls
1134 d->mActionType = NOP;
1135 d->mScrollUpTimer.stop();
1136 d->mScrollDownTimer.stop();
1137#ifndef QT_NO_CURSOR
1139#endif
1140
1141 if (!d->mChanger) {
1142 qCCritical(CALENDARVIEW_LOG) << "No IncidenceChanger set";
1143 return;
1144 }
1145
1146 bool multiModify = false;
1147 // FIXME: do the cloning here...
1148 KCalendarCore::Incidence::Ptr incidence = d->mActionItem->incidence();
1149 const auto recurrenceId = d->mActionItem->occurrenceDateTime();
1150
1151 d->mItemMoved = d->mItemMoved && !(d->mStartCell.x() == d->mEndCell.x() && d->mStartCell.y() == d->mEndCell.y());
1152
1153 if (d->mItemMoved) {
1154 bool addIncidence = false;
1155 bool modify = false;
1156
1157 // get the main event and not the exception
1158 if (incidence->hasRecurrenceId() && !incidence->recurs()) {
1159 KCalendarCore::Incidence::Ptr mainIncidence;
1160 KCalendarCore::Calendar::Ptr cal = d->mCalendar->findCalendar(incidence)->getCalendar();
1161 if (CalendarSupport::hasEvent(incidence)) {
1162 mainIncidence = cal->event(incidence->uid());
1163 } else if (CalendarSupport::hasTodo(incidence)) {
1164 mainIncidence = cal->todo(incidence->uid());
1165 }
1166 incidence = mainIncidence;
1167 }
1168
1169 Akonadi::Item item = d->mCalendar->item(incidence);
1170 if (incidence && incidence->recurs()) {
1171 const int res = d->mAgendaView->showMoveRecurDialog(incidence, recurrenceId.date());
1172
1173 if (!d->mActionItem) {
1174 qCWarning(CALENDARVIEW_LOG) << "mActionItem was reset while the 'move' dialog was active";
1175 d->mItemMoved = false;
1176 return;
1177 }
1178
1179 switch (res) {
1180 case KCalUtils::RecurrenceActions::AllOccurrences: // All occurrences
1181 // Moving the whole sequence of events is handled by the itemModified below.
1182 modify = true;
1183 break;
1186 const bool thisAndFuture = (res == KCalUtils::RecurrenceActions::FutureOccurrences);
1187 modify = true;
1188 multiModify = true;
1189 d->mChanger->startAtomicOperation(i18n("Dissociate event from recurrence"));
1190 KCalendarCore::Incidence::Ptr newInc(KCalendarCore::Calendar::createException(incidence, recurrenceId, thisAndFuture));
1191 if (newInc) {
1192 newInc->removeCustomProperty("VOLATILE", "AKONADI-ID");
1193 Akonadi::Item newItem = d->mCalendar->item(newInc);
1194
1195 if (newItem.isValid() && newItem != item) { // it is not a new exception
1196 item = newItem;
1197 newInc->setCustomProperty("VOLATILE", "AKONADI-ID", QString::number(newItem.id()));
1198 addIncidence = false;
1199 } else {
1200 addIncidence = true;
1201 }
1202 // don't recreate items, they already have the correct position
1203 d->mAgendaView->enableAgendaUpdate(false);
1204
1205 d->mActionItem->setIncidence(newInc);
1206 d->mActionItem->dissociateFromMultiItem();
1207
1208 d->mAgendaView->enableAgendaUpdate(true);
1209 } else {
1210 KMessageBox::error(this,
1211 i18n("Unable to add the exception item to the calendar. "
1212 "No change will be done."),
1213 i18nc("@title:window", "Error Occurred"));
1214 }
1215 break;
1216 }
1217 default:
1218 modify = false;
1219 d->mActionItem->resetMove();
1220 placeSubCells(d->mActionItem); // PENDING(AKONADI_PORT) should this be done after
1221 // the new item was asynchronously added?
1222 }
1223 }
1224
1225 AgendaItem::QPtr placeItem = d->mActionItem->firstMultiItem();
1226 if (!placeItem) {
1227 placeItem = d->mActionItem;
1228 }
1229
1230 Akonadi::Collection::Id saveCollection = -1;
1231
1232 if (item.isValid()) {
1233 saveCollection = item.parentCollection().id();
1234
1235 // if parent collection is only a search collection for example
1237 saveCollection = item.storageCollectionId();
1238 }
1239 }
1240
1241 if (modify) {
1242 d->mActionItem->endMove();
1243
1244 AgendaItem::QPtr modif = placeItem;
1245
1246 QList<AgendaItem::QPtr> oldconflictItems = placeItem->conflictItems();
1247 QList<AgendaItem::QPtr>::iterator it;
1248 for (it = oldconflictItems.begin(); it != oldconflictItems.end(); ++it) {
1249 if (*it) {
1250 placeSubCells(*it);
1251 }
1252 }
1253 while (placeItem) {
1254 placeSubCells(placeItem);
1255 placeItem = placeItem->nextMultiItem();
1256 }
1257
1258 // Notify about change
1259 // The agenda view will apply the changes to the actual Incidence*!
1260 // Bug #228696 don't call endChanged now it's async in Akonadi so it can
1261 // be called before that modified item was done. And endChange is
1262 // calling when we move item.
1263 // Not perfect need to improve it!
1264 // mChanger->endChange(inc);
1265 if (item.isValid()) {
1266 d->mAgendaView->updateEventDates(modif, addIncidence, saveCollection);
1267 }
1268 if (addIncidence) {
1269 // delete the one we dragged, there's a new one being added async, due to dissociation.
1270 delete modif;
1271 }
1272 } else {
1273 // the item was moved, but not further modified, since it's not recurring
1274 // make sure the view updates anyhow, with the right item
1275 if (item.isValid()) {
1276 d->mAgendaView->updateEventDates(placeItem, addIncidence, saveCollection);
1277 }
1278 }
1279 }
1280
1281 d->mActionItem = nullptr;
1282 d->mItemMoved = false;
1283
1284 if (multiModify) {
1285 d->mChanger->endAtomicOperation();
1286 }
1287}
1288
1289void Agenda::setActionCursor(int actionType, bool acting)
1290{
1291#ifndef QT_NO_CURSOR
1292 switch (actionType) {
1293 case MOVE:
1294 if (acting) {
1296 } else {
1298 }
1299 break;
1300 case RESIZETOP:
1301 case RESIZEBOTTOM:
1303 break;
1304 case RESIZELEFT:
1305 case RESIZERIGHT:
1307 break;
1308 default:
1310 }
1311#endif
1312}
1313
1314void Agenda::setNoActionCursor(const AgendaItem::QPtr &moveItem, QPoint pos)
1315{
1316 const KCalendarCore::Incidence::Ptr item = moveItem ? moveItem->incidence() : KCalendarCore::Incidence::Ptr();
1317
1318 const bool noResize = CalendarSupport::hasTodo(item);
1319
1320 Agenda::MouseActionType resizeType = MOVE;
1321 if (!noResize) {
1322 resizeType = isInResizeArea(d->mAllDayMode, pos, moveItem);
1323 }
1324 setActionCursor(resizeType);
1325}
1326
1327/** calculate the width of the column subcells of the given item
1328 */
1329double Agenda::calcSubCellWidth(const AgendaItem::QPtr &item)
1330{
1331 QPoint pt;
1332 QPoint pt1;
1333 pt = gridToContents(QPoint(item->cellXLeft(), item->cellYTop()));
1334 pt1 = gridToContents(QPoint(item->cellXLeft(), item->cellYTop()) + QPoint(1, 1));
1335 pt1 -= pt;
1336 int maxSubCells = item->subCells();
1337 double newSubCellWidth;
1338 if (d->mAllDayMode) {
1339 newSubCellWidth = static_cast<double>(pt1.y()) / maxSubCells;
1340 } else {
1341 newSubCellWidth = static_cast<double>(pt1.x()) / maxSubCells;
1342 }
1343 return newSubCellWidth;
1344}
1345
1346void Agenda::adjustItemPosition(const AgendaItem::QPtr &item)
1347{
1348 if (!item) {
1349 return;
1350 }
1351 item->resize(int(d->mGridSpacingX * item->cellWidth()), int(d->mGridSpacingY * item->cellHeight()));
1352 int clXLeft = item->cellXLeft();
1354 clXLeft = item->cellXRight() + 1;
1355 }
1356 QPoint cpos = gridToContents(QPoint(clXLeft, item->cellYTop()));
1357 item->move(cpos.x(), cpos.y());
1358}
1359
1360void Agenda::placeAgendaItem(const AgendaItem::QPtr &item, double subCellWidth)
1361{
1362 // "left" upper corner, no subcells yet, RTL layouts have right/left
1363 // switched, widths are negative then
1364 QPoint pt = gridToContents(QPoint(item->cellXLeft(), item->cellYTop()));
1365 // right lower corner
1366 QPoint pt1 = gridToContents(QPoint(item->cellXLeft() + item->cellWidth(), item->cellYBottom() + 1));
1367
1368 double subCellPos = item->subCell() * subCellWidth;
1369
1370 // we need to add 0.01 to make sure we don't loose one pixed due to numerics
1371 // (i.e. if it would be x.9998, we want the integer, not rounded down.
1372 double delta = 0.01;
1373 if (subCellWidth < 0) {
1374 delta = -delta;
1375 }
1376 int height;
1377 int width;
1378 int xpos;
1379 int ypos;
1380 if (d->mAllDayMode) {
1381 width = pt1.x() - pt.x();
1382 height = int(subCellPos + subCellWidth + delta) - int(subCellPos);
1383 xpos = pt.x();
1384 ypos = pt.y() + int(subCellPos);
1385 } else {
1386 width = int(subCellPos + subCellWidth + delta) - int(subCellPos);
1387 height = pt1.y() - pt.y();
1388 xpos = pt.x() + int(subCellPos);
1389 ypos = pt.y();
1390 }
1391 if (QApplication::isRightToLeft()) { // RTL language/layout
1392 xpos += width;
1393 width = -width;
1394 }
1395 if (height < 0) { // BTT (bottom-to-top) layout ?!?
1396 ypos += height;
1397 height = -height;
1398 }
1399 item->resize(width, height);
1400 item->move(xpos, ypos);
1401}
1402
1403/*
1404 Place item in cell and take care that multiple items using the same cell do
1405 not overlap. This method is not yet optimal. It doesn't use the maximum space
1406 it can get in all cases.
1407 At the moment the method has a bug: When an item is placed only the sub cell
1408 widths of the items are changed, which are within the Y region the item to
1409 place spans. When the sub cell width change of one of this items affects a
1410 cell, where other items are, which do not overlap in Y with the item to
1411 place, the display gets corrupted, although the corruption looks quite nice.
1412*/
1413void Agenda::placeSubCells(const AgendaItem::QPtr &placeItem)
1414{
1415#if 0
1416 qCDebug(CALENDARVIEW_LOG);
1417 if (placeItem) {
1418 KCalendarCore::Incidence::Ptr event = placeItem->incidence();
1419 if (!event) {
1420 qCDebug(CALENDARVIEW_LOG) << " event is 0";
1421 } else {
1422 qCDebug(CALENDARVIEW_LOG) << " event:" << event->summary();
1423 }
1424 } else {
1425 qCDebug(CALENDARVIEW_LOG) << " placeItem is 0";
1426 }
1427 qCDebug(CALENDARVIEW_LOG) << "Agenda::placeSubCells()...";
1428#endif
1429
1430 QList<CalendarSupport::CellItem *> cells;
1431 for (CalendarSupport::CellItem *item : std::as_const(d->mItems)) {
1432 if (item) {
1433 cells.append(item);
1434 }
1435 }
1436
1437 QList<CalendarSupport::CellItem *> items = CalendarSupport::CellItem::placeItem(cells, placeItem);
1438
1439 placeItem->setConflictItems(QList<AgendaItem::QPtr>());
1440 double newSubCellWidth = calcSubCellWidth(placeItem);
1441 QList<CalendarSupport::CellItem *>::iterator it;
1442 for (it = items.begin(); it != items.end(); ++it) {
1443 if (*it) {
1444 AgendaItem::QPtr item = static_cast<AgendaItem *>(*it);
1445 placeAgendaItem(item, newSubCellWidth);
1446 item->addConflictItem(placeItem);
1447 placeItem->addConflictItem(item);
1448 }
1449 }
1450 if (items.isEmpty()) {
1451 placeAgendaItem(placeItem, newSubCellWidth);
1452 }
1453 placeItem->update();
1454}
1455
1456int Agenda::columnWidth(int column) const
1457{
1458 int start = gridToContents(QPoint(column, 0)).x();
1460 column--;
1461 } else {
1462 column++;
1463 }
1464 int end = gridToContents(QPoint(column, 0)).x();
1465 return end - start;
1466}
1467
1468void Agenda::paintEvent(QPaintEvent *)
1469{
1470 QPainter p(this);
1471 drawContents(&p, 0, -y(), d->mGridSpacingX * d->mColumns, d->mGridSpacingY * d->mRows + y());
1472}
1473
1474/*
1475 Draw grid in the background of the agenda.
1476*/
1477void Agenda::drawContents(QPainter *p, int cx, int cy, int cw, int ch)
1478{
1479 QPixmap db(cw, ch);
1480 db.fill(); // We don't want to see leftovers from previous paints
1481 QPainter dbp(&db);
1482 // TODO: CHECK THIS
1483 // if (! d->preferences()->agendaGridBackgroundImage().isEmpty()) {
1484 // QPixmap bgImage(d->preferences()->agendaGridBackgroundImage());
1485 // dbp.drawPixmap(0, 0, cw, ch, bgImage); FIXME
1486 // }
1487 if (!d->preferences()->useSystemColor()) {
1488 dbp.fillRect(0, 0, cw, ch, d->preferences()->agendaGridBackgroundColor());
1489 } else {
1490 dbp.fillRect(0, 0, cw, ch, palette().color(QPalette::Window));
1491 }
1492
1493 dbp.translate(-cx, -cy);
1494
1495 double lGridSpacingY = d->mGridSpacingY * 2;
1496
1497 // If work day, use work color
1498 // If busy day, use busy color
1499 // if work and busy day, mix both, and busy color has alpha
1500
1501 const QList<bool> busyDayMask = d->mAgendaView->busyDayMask();
1502
1503 // Highlight working hours
1504 if (d->mWorkingHoursEnable && d->mHolidayMask) {
1505 QColor workColor;
1506 if (!d->preferences()->useSystemColor()) {
1507 workColor = d->preferences()->workingHoursColor();
1508 } else {
1509 workColor = palette().color(QPalette::Base);
1510 }
1511
1512 QPoint pt1(cx, d->mWorkingHoursYTop);
1513 QPoint pt2(cx + cw, d->mWorkingHoursYBottom);
1514 if (pt2.x() >= pt1.x() /*&& pt2.y() >= pt1.y()*/) {
1515 int gxStart = contentsToGrid(pt1).x();
1516 int gxEnd = contentsToGrid(pt2).x();
1517 // correct start/end for rtl layouts
1518 if (gxStart > gxEnd) {
1519 int tmp = gxStart;
1520 gxStart = gxEnd;
1521 gxEnd = tmp;
1522 }
1523 int xoffset = (QApplication::isRightToLeft() ? 1 : 0);
1524 while (gxStart <= gxEnd) {
1525 int xStart = gridToContents(QPoint(gxStart + xoffset, 0)).x();
1526 int xWidth = columnWidth(gxStart) + 1;
1527
1528 if (pt2.y() < pt1.y()) {
1529 // overnight working hours
1530 if (((gxStart == 0) && !d->mHolidayMask->at(d->mHolidayMask->count() - 1))
1531 || ((gxStart > 0) && (gxStart < int(d->mHolidayMask->count())) && (!d->mHolidayMask->at(gxStart - 1)))) {
1532 if (pt2.y() > cy) {
1533 dbp.fillRect(xStart, cy, xWidth, pt2.y() - cy + 1, workColor);
1534 }
1535 }
1536 if ((gxStart < int(d->mHolidayMask->count() - 1)) && (!d->mHolidayMask->at(gxStart))) {
1537 if (pt1.y() < cy + ch - 1) {
1538 dbp.fillRect(xStart, pt1.y(), xWidth, cy + ch - pt1.y() + 1, workColor);
1539 }
1540 }
1541 } else {
1542 // last entry in holiday mask denotes the previous day not visible
1543 // (needed for overnight shifts)
1544 if (gxStart < int(d->mHolidayMask->count() - 1) && !d->mHolidayMask->at(gxStart)) {
1545 dbp.fillRect(xStart, pt1.y(), xWidth, pt2.y() - pt1.y() + 1, workColor);
1546 }
1547 }
1548 ++gxStart;
1549 }
1550 }
1551 }
1552
1553 // busy days
1554 if (d->preferences()->colorAgendaBusyDays() && !d->mAllDayMode) {
1555 for (int i = 0; i < busyDayMask.count(); ++i) {
1556 if (busyDayMask[i]) {
1557 const QPoint pt1(cx + d->mGridSpacingX * i, 0);
1558 // const QPoint pt2(cx + mGridSpacingX * (i+1), ch);
1559 QColor busyColor;
1560 if (!d->preferences()->useSystemColor()) {
1561 busyColor = d->preferences()->viewBgBusyColor();
1562 } else {
1563 busyColor = palette().color(QPalette::Window);
1564 if ((busyColor.blue() + busyColor.red() + busyColor.green()) > (256 / 2 * 3)) {
1565 // dark
1566 busyColor = busyColor.lighter(140);
1567 } else {
1568 // light
1569 busyColor = busyColor.darker(140);
1570 }
1571 }
1572 busyColor.setAlpha(EventViews::BUSY_BACKGROUND_ALPHA);
1573 dbp.fillRect(pt1.x(), pt1.y(), d->mGridSpacingX, cy + ch, busyColor);
1574 }
1575 }
1576 }
1577
1578 // draw selection
1579 if (d->mHasSelection && d->mAgendaView->dateRangeSelectionEnabled()) {
1580 QPoint pt;
1581 QPoint pt1;
1582 QColor highlightColor;
1583 if (!d->preferences()->useSystemColor()) {
1584 highlightColor = d->preferences()->agendaGridHighlightColor();
1585 } else {
1586 highlightColor = palette().color(QPalette::Highlight);
1587 }
1588
1589 if (d->mSelectionEndCell.x() > d->mSelectionStartCell.x()) { // multi day selection
1590 // draw start day
1591 pt = gridToContents(d->mSelectionStartCell);
1592 pt1 = gridToContents(QPoint(d->mSelectionStartCell.x() + 1, d->mRows + 1));
1593 dbp.fillRect(QRect(pt, pt1), highlightColor);
1594 // draw all other days between the start day and the day of the selection end
1595 for (int c = d->mSelectionStartCell.x() + 1; c < d->mSelectionEndCell.x(); ++c) {
1596 pt = gridToContents(QPoint(c, 0));
1597 pt1 = gridToContents(QPoint(c + 1, d->mRows + 1));
1598 dbp.fillRect(QRect(pt, pt1), highlightColor);
1599 }
1600 // draw end day
1601 pt = gridToContents(QPoint(d->mSelectionEndCell.x(), 0));
1602 pt1 = gridToContents(d->mSelectionEndCell + QPoint(1, 1));
1603 dbp.fillRect(QRect(pt, pt1), highlightColor);
1604 } else { // single day selection
1605 pt = gridToContents(d->mSelectionStartCell);
1606 pt1 = gridToContents(d->mSelectionEndCell + QPoint(1, 1));
1607 dbp.fillRect(QRect(pt, pt1), highlightColor);
1608 }
1609 }
1610
1611 // Compute the grid line color for both the hour and half-hour
1612 // The grid colors are always computed as a function of the palette's windowText color.
1613 QPen hourPen;
1614 QPen halfHourPen;
1615
1616 const QColor windowTextColor = palette().color(QPalette::WindowText);
1617 if (windowTextColor.red() + windowTextColor.green() + windowTextColor.blue() < (256 / 2 * 3)) {
1618 // dark grey line
1619 hourPen = windowTextColor.lighter(200);
1620 halfHourPen = windowTextColor.lighter(500);
1621 } else {
1622 // light grey line
1623 hourPen = windowTextColor.darker(150);
1624 halfHourPen = windowTextColor.darker(200);
1625 }
1626
1627 dbp.setPen(hourPen);
1628
1629 // Draw vertical lines of grid, start with the last line not yet visible
1630 double x = (int(cx / d->mGridSpacingX)) * d->mGridSpacingX;
1631 while (x < cx + cw) {
1632 dbp.drawLine(int(x), cy, int(x), cy + ch);
1633 x += d->mGridSpacingX;
1634 }
1635
1636 // Draw horizontal lines of grid
1637 double y = (int(cy / (2 * lGridSpacingY))) * 2 * lGridSpacingY;
1638 while (y < cy + ch) {
1639 dbp.drawLine(cx, int(y), cx + cw, int(y));
1640 y += 2 * lGridSpacingY;
1641 }
1642 y = (2 * int(cy / (2 * lGridSpacingY)) + 1) * lGridSpacingY;
1643 dbp.setPen(halfHourPen);
1644 while (y < cy + ch) {
1645 dbp.drawLine(cx, int(y), cx + cw, int(y));
1646 y += 2 * lGridSpacingY;
1647 }
1648 p->drawPixmap(cx, cy, db);
1649}
1650
1651/*
1652 Convert srcollview contents coordinates to agenda grid coordinates.
1653*/
1654QPoint Agenda::contentsToGrid(QPoint pos) const
1655{
1656 int gx = int(QApplication::isRightToLeft() ? d->mColumns - pos.x() / d->mGridSpacingX : pos.x() / d->mGridSpacingX);
1657 int gy = int(pos.y() / d->mGridSpacingY);
1658 return {gx, gy};
1659}
1660
1661/*
1662 Convert agenda grid coordinates to scrollview contents coordinates.
1663*/
1664QPoint Agenda::gridToContents(QPoint gpos) const
1665{
1666 int x = int(QApplication::isRightToLeft() ? (d->mColumns - gpos.x()) * d->mGridSpacingX : gpos.x() * d->mGridSpacingX);
1667 int y = int(gpos.y() * d->mGridSpacingY);
1668 return {x, y};
1669}
1670
1671/*
1672 Return Y coordinate corresponding to time. Coordinates are rounded to
1673 fit into the grid.
1674*/
1675int Agenda::timeToY(QTime time) const
1676{
1677 int minutesPerCell = 24 * 60 / d->mRows;
1678 int timeMinutes = time.hour() * 60 + time.minute();
1679 int Y = (timeMinutes + (minutesPerCell / 2)) / minutesPerCell;
1680
1681 return Y;
1682}
1683
1684/*
1685 Return time corresponding to cell y coordinate. Coordinates are rounded to
1686 fit into the grid.
1687*/
1688QTime Agenda::gyToTime(int gy) const
1689{
1690 int secondsPerCell = 24 * 60 * 60 / d->mRows;
1691 int timeSeconds = secondsPerCell * gy;
1692
1693 QTime time(0, 0, 0);
1694 if (timeSeconds < 24 * 60 * 60) {
1695 time = time.addSecs(timeSeconds);
1696 } else {
1697 time.setHMS(23, 59, 59);
1698 }
1699 return time;
1700}
1701
1702QList<int> Agenda::minContentsY() const
1703{
1704 QList<int> minArray;
1705 minArray.fill(timeToY(QTime(23, 59)), d->mSelectedDates.count());
1706 for (const AgendaItem::QPtr &item : std::as_const(d->mItems)) {
1707 if (item) {
1708 int ymin = item->cellYTop();
1709 int index = item->cellXLeft();
1710 if (index >= 0 && index < (int)(d->mSelectedDates.count())) {
1711 if (ymin < minArray[index] && !d->mItemsToDelete.contains(item)) {
1712 minArray[index] = ymin;
1713 }
1714 }
1715 }
1716 }
1717
1718 return minArray;
1719}
1720
1721QList<int> Agenda::maxContentsY() const
1722{
1723 QList<int> maxArray;
1724 maxArray.fill(timeToY(QTime(0, 0)), d->mSelectedDates.count());
1725 for (const AgendaItem::QPtr &item : std::as_const(d->mItems)) {
1726 if (item) {
1727 int ymax = item->cellYBottom();
1728
1729 int index = item->cellXLeft();
1730 if (index >= 0 && index < (int)(d->mSelectedDates.count())) {
1731 if (ymax > maxArray[index] && !d->mItemsToDelete.contains(item)) {
1732 maxArray[index] = ymax;
1733 }
1734 }
1735 }
1736 }
1737
1738 return maxArray;
1739}
1740
1741void Agenda::setStartTime(QTime startHour)
1742{
1743 const double startPos = (startHour.hour() / 24. + startHour.minute() / 1440. + startHour.second() / 86400.) * d->mRows * gridSpacingY();
1744
1745 verticalScrollBar()->setValue(startPos);
1746}
1747
1748/*
1749 Insert AgendaItem into agenda.
1750*/
1751AgendaItem::QPtr Agenda::insertItem(const KCalendarCore::Incidence::Ptr &incidence,
1752 const QDateTime &recurrenceId,
1753 int X,
1754 int YTop,
1755 int YBottom,
1756 int itemPos,
1757 int itemCount,
1758 bool isSelected)
1759{
1760 if (d->mAllDayMode) {
1761 qCDebug(CALENDARVIEW_LOG) << "using this in all-day mode is illegal.";
1762 return nullptr;
1763 }
1764
1765 d->mActionType = NOP;
1766
1767 AgendaItem::QPtr agendaItem = createAgendaItem(incidence, itemPos, itemCount, recurrenceId, isSelected);
1768 if (!agendaItem) {
1769 return {};
1770 }
1771
1772 if (YTop >= d->mRows) {
1773 YBottom -= YTop - (d->mRows - 1); // Slide the item up into view.
1774 YTop = d->mRows - 1;
1775 }
1776 if (YBottom <= YTop) {
1777 qCDebug(CALENDARVIEW_LOG) << "Text:" << agendaItem->text() << " YSize<0";
1778 YBottom = YTop;
1779 }
1780
1781 agendaItem->resize(int((X + 1) * d->mGridSpacingX) - int(X * d->mGridSpacingX), int(YTop * d->mGridSpacingY) - int((YBottom + 1) * d->mGridSpacingY));
1782 agendaItem->setCellXY(X, YTop, YBottom);
1783 agendaItem->setCellXRight(X);
1784 agendaItem->setResourceColor(d->mCalendar->resourceColor(incidence));
1785 agendaItem->installEventFilter(this);
1786
1787 agendaItem->move(int(X * d->mGridSpacingX), int(YTop * d->mGridSpacingY));
1788
1789 d->mItems.append(agendaItem);
1790
1791 placeSubCells(agendaItem);
1792
1793 agendaItem->show();
1794
1795 marcus_bains();
1796
1797 return agendaItem;
1798}
1799
1800/*
1801 Insert all-day AgendaItem into agenda.
1802*/
1803AgendaItem::QPtr Agenda::insertAllDayItem(const KCalendarCore::Incidence::Ptr &incidence, const QDateTime &recurrenceId, int XBegin, int XEnd, bool isSelected)
1804{
1805 if (!d->mAllDayMode) {
1806 qCCritical(CALENDARVIEW_LOG) << "using this in non all-day mode is illegal.";
1807 return nullptr;
1808 }
1809
1810 d->mActionType = NOP;
1811
1812 AgendaItem::QPtr agendaItem = createAgendaItem(incidence, 1, 1, recurrenceId, isSelected);
1813 if (!agendaItem) {
1814 return {};
1815 }
1816
1817 agendaItem->setCellXY(XBegin, 0, 0);
1818 agendaItem->setCellXRight(XEnd);
1819
1820 const double startIt = d->mGridSpacingX * (agendaItem->cellXLeft());
1821 const double endIt = d->mGridSpacingX * (agendaItem->cellWidth() + agendaItem->cellXLeft());
1822
1823 agendaItem->resize(int(endIt) - int(startIt), int(d->mGridSpacingY));
1824
1825 agendaItem->installEventFilter(this);
1826 agendaItem->setResourceColor(d->mCalendar->resourceColor(incidence));
1827 agendaItem->move(int(XBegin * d->mGridSpacingX), 0);
1828 d->mItems.append(agendaItem);
1829
1830 placeSubCells(agendaItem);
1831
1832 agendaItem->show();
1833
1834 return agendaItem;
1835}
1836
1837AgendaItem::QPtr
1838Agenda::createAgendaItem(const KCalendarCore::Incidence::Ptr &incidence, int itemPos, int itemCount, const QDateTime &recurrenceId, bool isSelected)
1839{
1840 if (!incidence) {
1841 qCWarning(CALENDARVIEW_LOG) << "Agenda::createAgendaItem() item is invalid.";
1842 return {};
1843 }
1844
1845 AgendaItem::QPtr agendaItem = new AgendaItem(d->mAgendaView, d->mCalendar, incidence, itemPos, itemCount, recurrenceId, isSelected, this);
1846
1847 connect(agendaItem.data(), &AgendaItem::removeAgendaItem, this, &Agenda::removeAgendaItem);
1848 connect(agendaItem.data(), &AgendaItem::showAgendaItem, this, &Agenda::showAgendaItem);
1849
1850 d->mAgendaItemsById.insert(incidence->uid(), agendaItem);
1851
1852 return agendaItem;
1853}
1854
1855void Agenda::insertMultiItem(const KCalendarCore::Incidence::Ptr &event,
1856 const QDateTime &recurrenceId,
1857 int XBegin,
1858 int XEnd,
1859 int YTop,
1860 int YBottom,
1861 bool isSelected)
1862{
1863 KCalendarCore::Event::Ptr ev = CalendarSupport::event(event);
1864 Q_ASSERT(ev);
1865 if (d->mAllDayMode) {
1866 qCDebug(CALENDARVIEW_LOG) << "using this in all-day mode is illegal.";
1867 return;
1868 }
1869
1870 d->mActionType = NOP;
1871 int cellX;
1872 int cellYTop;
1873 int cellYBottom;
1874 QString newtext;
1875 int width = XEnd - XBegin + 1;
1876 int count = 0;
1877 AgendaItem::QPtr current = nullptr;
1878 QList<AgendaItem::QPtr> multiItems;
1879 int visibleCount = d->mSelectedDates.first().daysTo(d->mSelectedDates.last());
1880 for (cellX = XBegin; cellX <= XEnd; ++cellX) {
1881 ++count;
1882 // Only add the items that are visible.
1883 if (cellX >= 0 && cellX <= visibleCount) {
1884 if (cellX == XBegin) {
1885 cellYTop = YTop;
1886 } else {
1887 cellYTop = 0;
1888 }
1889 if (cellX == XEnd) {
1890 cellYBottom = YBottom;
1891 } else {
1892 cellYBottom = rows() - 1;
1893 }
1894 newtext = QStringLiteral("(%1/%2): ").arg(count).arg(width);
1895 newtext.append(ev->summary());
1896
1897 current = insertItem(event, recurrenceId, cellX, cellYTop, cellYBottom, count, width, isSelected);
1898 Q_ASSERT(current);
1899 current->setText(newtext);
1900 multiItems.append(current);
1901 }
1902 }
1903
1904 QList<AgendaItem::QPtr>::iterator it = multiItems.begin();
1905 QList<AgendaItem::QPtr>::iterator e = multiItems.end();
1906
1907 if (it != e) { // .first asserts if the list is empty
1908 AgendaItem::QPtr first = multiItems.first();
1909 AgendaItem::QPtr last = multiItems.last();
1910 AgendaItem::QPtr prev = nullptr;
1911 AgendaItem::QPtr next = nullptr;
1912
1913 while (it != e) {
1914 AgendaItem::QPtr item = *it;
1915 ++it;
1916 next = (it == e) ? nullptr : (*it);
1917 if (item) {
1918 item->setMultiItem((item == first) ? nullptr : first, prev, next, (item == last) ? nullptr : last);
1919 }
1920 prev = item;
1921 }
1922 }
1923
1924 marcus_bains();
1925}
1926
1927void Agenda::removeIncidence(const KCalendarCore::Incidence::Ptr &incidence)
1928{
1929 if (!incidence) {
1930 qCWarning(CALENDARVIEW_LOG) << "Agenda::removeIncidence() incidence is invalid" << incidence->uid();
1931 return;
1932 }
1933
1934 if (d->isQueuedForDeletion(incidence->uid())) {
1935 return; // It's already queued for deletion
1936 }
1937
1938 const AgendaItem::List agendaItems = d->mAgendaItemsById.values(incidence->uid());
1939 if (agendaItems.isEmpty()) {
1940 // We're not displaying such item
1941 // qCDebug(CALENDARVIEW_LOG) << "Ignoring";
1942 return;
1943 }
1944 for (const AgendaItem::QPtr &agendaItem : agendaItems) {
1945 if (agendaItem) {
1946 if (incidence->instanceIdentifier() != agendaItem->incidence()->instanceIdentifier()) {
1947 continue;
1948 }
1949 if (!removeAgendaItem(agendaItem)) {
1950 qCWarning(CALENDARVIEW_LOG) << "Agenda::removeIncidence() Failed to remove " << incidence->uid();
1951 }
1952 }
1953 }
1954}
1955
1956void Agenda::showAgendaItem(const AgendaItem::QPtr &agendaItem)
1957{
1958 if (!agendaItem) {
1959 qCCritical(CALENDARVIEW_LOG) << "Show what?";
1960 return;
1961 }
1962
1963 agendaItem->hide();
1964
1965 agendaItem->setParent(this);
1966
1967 if (!d->mItems.contains(agendaItem)) {
1968 d->mItems.append(agendaItem);
1969 }
1970 placeSubCells(agendaItem);
1971
1972 agendaItem->show();
1973}
1974
1975bool Agenda::removeAgendaItem(const AgendaItem::QPtr &agendaItem)
1976{
1977 Q_ASSERT(agendaItem);
1978 // we found the item. Let's remove it and update the conflicts
1979 QList<AgendaItem::QPtr> conflictItems = agendaItem->conflictItems();
1980 // removeChild(thisItem);
1981
1982 bool taken = d->mItems.removeAll(agendaItem) > 0;
1983 d->mAgendaItemsById.remove(agendaItem->incidence()->uid(), agendaItem);
1984
1985 QList<AgendaItem::QPtr>::iterator it;
1986 for (it = conflictItems.begin(); it != conflictItems.end(); ++it) {
1987 if (*it) {
1988 (*it)->setSubCells((*it)->subCells() - 1);
1989 }
1990 }
1991
1992 for (it = conflictItems.begin(); it != conflictItems.end(); ++it) {
1993 // the item itself is also in its own conflictItems list!
1994 if (*it && *it != agendaItem) {
1995 placeSubCells(*it);
1996 }
1997 }
1998 d->mItemsToDelete.append(agendaItem);
1999 d->mItemsQueuedForDeletion.insert(agendaItem->incidence()->uid());
2000 agendaItem->setVisible(false);
2001 QTimer::singleShot(0, this, &Agenda::deleteItemsToDelete);
2002 return taken;
2003}
2004
2005void Agenda::deleteItemsToDelete()
2006{
2007 qDeleteAll(d->mItemsToDelete);
2008 d->mItemsToDelete.clear();
2009 d->mItemsQueuedForDeletion.clear();
2010}
2011
2012/*QSizePolicy Agenda::sizePolicy() const
2013{
2014 // Thought this would make the all-day event agenda minimum size and the
2015 // normal agenda take the remaining space. But it doesn't work. The QSplitter
2016 // don't seem to think that an Expanding widget needs more space than a
2017 // Preferred one.
2018 // But it doesn't hurt, so it stays.
2019 if (mAllDayMode) {
2020 return QSizePolicy(QSizePolicy::Expanding,QSizePolicy::Preferred);
2021 } else {
2022 return QSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding);
2023 }
2024}*/
2025
2026/*
2027 Overridden from QScrollView to provide proper resizing of AgendaItems.
2028*/
2029void Agenda::resizeEvent(QResizeEvent *ev)
2030{
2031 QSize newSize(ev->size());
2032
2033 if (d->mAllDayMode) {
2034 d->mGridSpacingX = static_cast<double>(newSize.width()) / d->mColumns;
2035 d->mGridSpacingY = newSize.height();
2036 } else {
2037 d->mGridSpacingX = static_cast<double>(newSize.width()) / d->mColumns;
2038 // make sure that there are not more than 24 per day
2039 d->mGridSpacingY = static_cast<double>(newSize.height()) / d->mRows;
2040 if (d->mGridSpacingY < d->mDesiredGridSpacingY) {
2041 d->mGridSpacingY = d->mDesiredGridSpacingY;
2042 }
2043 }
2044 calculateWorkingHours();
2045
2046 QTimer::singleShot(0, this, &Agenda::resizeAllContents);
2047 Q_EMIT gridSpacingYChanged(d->mGridSpacingY * 4);
2048
2051}
2052
2053void Agenda::resizeAllContents()
2054{
2055 double subCellWidth;
2056 for (const AgendaItem::QPtr &item : std::as_const(d->mItems)) {
2057 if (item) {
2058 subCellWidth = calcSubCellWidth(item);
2059 placeAgendaItem(item, subCellWidth);
2060 }
2061 }
2062 /*
2063 if (d->mAllDayMode) {
2064 foreach (const AgendaItem::QPtr &item, d->mItems) {
2065 if (item) {
2066 subCellWidth = calcSubCellWidth(item);
2067 placeAgendaItem(item, subCellWidth);
2068 }
2069 }
2070 } else {
2071 foreach (const AgendaItem::QPtr &item, d->mItems) {
2072 if (item) {
2073 subCellWidth = calcSubCellWidth(item);
2074 placeAgendaItem(item, subCellWidth);
2075 }
2076 }
2077 }
2078 */
2079 checkScrollBoundaries();
2080 marcus_bains();
2081 update();
2082}
2083
2084void Agenda::scrollUp()
2085{
2086 int currentValue = verticalScrollBar()->value();
2087 verticalScrollBar()->setValue(currentValue - d->mScrollOffset);
2088}
2089
2090void Agenda::scrollDown()
2091{
2092 int currentValue = verticalScrollBar()->value();
2093 verticalScrollBar()->setValue(currentValue + d->mScrollOffset);
2094}
2095
2096QSize Agenda::minimumSize() const
2097{
2098 return sizeHint();
2099}
2100
2101QSize Agenda::minimumSizeHint() const
2102{
2103 return sizeHint();
2104}
2105
2106int Agenda::minimumHeight() const
2107{
2108 // all day agenda never has scrollbars and the scrollarea will
2109 // resize it to fit exactly on the viewport.
2110
2111 if (d->mAllDayMode) {
2112 return 0;
2113 } else {
2114 return d->mGridSpacingY * d->mRows;
2115 }
2116}
2117
2118void Agenda::updateConfig()
2119{
2120 const double oldGridSpacingY = d->mGridSpacingY;
2121
2122 if (!d->mAllDayMode) {
2123 d->mDesiredGridSpacingY = d->preferences()->hourSize();
2124 if (d->mDesiredGridSpacingY < 4 || d->mDesiredGridSpacingY > 30) {
2125 d->mDesiredGridSpacingY = 10;
2126 }
2127
2128 /*
2129 // make sure that there are not more than 24 per day
2130 d->mGridSpacingY = static_cast<double>(height()) / d->mRows;
2131 if (d->mGridSpacingY < d->mDesiredGridSpacingY || true) {
2132 d->mGridSpacingY = d->mDesiredGridSpacingY;
2133 }
2134 */
2135
2136 // can be two doubles equal?, it's better to compare them with an epsilon
2137 if (fabs(oldGridSpacingY - d->mDesiredGridSpacingY) > 0.1) {
2138 d->mGridSpacingY = d->mDesiredGridSpacingY;
2140 }
2141 }
2142
2143 calculateWorkingHours();
2144
2145 marcus_bains();
2146}
2147
2148void Agenda::checkScrollBoundaries()
2149{
2150 // Invalidate old values to force update
2151 d->mOldLowerScrollValue = -1;
2152 d->mOldUpperScrollValue = -1;
2153
2154 checkScrollBoundaries(verticalScrollBar()->value());
2155}
2156
2157void Agenda::checkScrollBoundaries(int v)
2158{
2159 int yMin = int((v) / d->mGridSpacingY);
2160 int yMax = int((v + d->mScrollArea->height()) / d->mGridSpacingY);
2161
2162 if (yMin != d->mOldLowerScrollValue) {
2163 d->mOldLowerScrollValue = yMin;
2164 Q_EMIT lowerYChanged(yMin);
2165 }
2166 if (yMax != d->mOldUpperScrollValue) {
2167 d->mOldUpperScrollValue = yMax;
2168 Q_EMIT upperYChanged(yMax);
2169 }
2170}
2171
2172int Agenda::visibleContentsYMin() const
2173{
2174 int v = verticalScrollBar()->value();
2175 return int(v / d->mGridSpacingY);
2176}
2177
2178int Agenda::visibleContentsYMax() const
2179{
2180 int v = verticalScrollBar()->value();
2181 return int((v + d->mScrollArea->height()) / d->mGridSpacingY);
2182}
2183
2184void Agenda::deselectItem()
2185{
2186 if (d->mSelectedItem.isNull()) {
2187 return;
2188 }
2189
2190 const KCalendarCore::Incidence::Ptr selectedItem = d->mSelectedItem->incidence();
2191
2192 for (AgendaItem::QPtr item : std::as_const(d->mItems)) {
2193 if (item) {
2194 const KCalendarCore::Incidence::Ptr itemInc = item->incidence();
2195 if (itemInc && selectedItem && itemInc->uid() == selectedItem->uid()) {
2196 item->select(false);
2197 }
2198 }
2199 }
2200
2201 d->mSelectedItem = nullptr;
2202}
2203
2204void Agenda::selectItem(const AgendaItem::QPtr &item)
2205{
2206 if ((AgendaItem::QPtr)d->mSelectedItem == item) {
2207 return;
2208 }
2209 deselectItem();
2210 if (item == nullptr) {
2211 Q_EMIT incidenceSelected(KCalendarCore::Incidence::Ptr(), QDate());
2212 return;
2213 }
2214 d->mSelectedItem = item;
2215 d->mSelectedItem->select();
2216 Q_ASSERT(d->mSelectedItem->incidence());
2217 d->mSelectedId = d->mSelectedItem->incidence()->uid();
2218
2219 for (AgendaItem::QPtr item : std::as_const(d->mItems)) {
2220 if (item && item->incidence()->uid() == d->mSelectedId) {
2221 item->select();
2222 }
2223 }
2224 Q_EMIT incidenceSelected(d->mSelectedItem->incidence(), d->mSelectedItem->occurrenceDate());
2225}
2226
2227void Agenda::selectIncidenceByUid(const QString &uid)
2228{
2229 for (const AgendaItem::QPtr &item : std::as_const(d->mItems)) {
2230 if (item && item->incidence()->uid() == uid) {
2231 selectItem(item);
2232 break;
2233 }
2234 }
2235}
2236
2237void Agenda::selectItem(const Akonadi::Item &item)
2238{
2239 selectIncidenceByUid(Akonadi::CalendarUtils::incidence(item)->uid());
2240}
2241
2242// This function seems never be called.
2243void Agenda::keyPressEvent(QKeyEvent *kev)
2244{
2245 switch (kev->key()) {
2246 case Qt::Key_PageDown:
2247 verticalScrollBar()->triggerAction(QAbstractSlider::SliderPageStepAdd);
2248 break;
2249 case Qt::Key_PageUp:
2250 verticalScrollBar()->triggerAction(QAbstractSlider::SliderPageStepSub);
2251 break;
2252 case Qt::Key_Down:
2253 verticalScrollBar()->triggerAction(QAbstractSlider::SliderSingleStepAdd);
2254 break;
2255 case Qt::Key_Up:
2256 verticalScrollBar()->triggerAction(QAbstractSlider::SliderSingleStepSub);
2257 break;
2258 default:;
2259 }
2260}
2261
2262void Agenda::calculateWorkingHours()
2263{
2264 d->mWorkingHoursEnable = !d->mAllDayMode;
2265
2266 QTime tmp = d->preferences()->workingHoursStart().time();
2267 d->mWorkingHoursYTop = int(4 * d->mGridSpacingY * (tmp.hour() + tmp.minute() / 60. + tmp.second() / 3600.));
2268 tmp = d->preferences()->workingHoursEnd().time();
2269 d->mWorkingHoursYBottom = int(4 * d->mGridSpacingY * (tmp.hour() + tmp.minute() / 60. + tmp.second() / 3600.) - 1);
2270}
2271
2272void Agenda::setDateList(const KCalendarCore::DateList &selectedDates)
2273{
2274 d->mSelectedDates = selectedDates;
2275 marcus_bains();
2276}
2277
2278KCalendarCore::DateList Agenda::dateList() const
2279{
2280 return d->mSelectedDates;
2281}
2282
2283void Agenda::setCalendar(const MultiViewCalendar::Ptr &cal)
2284{
2285 d->mCalendar = cal;
2286}
2287
2288void Agenda::setIncidenceChanger(Akonadi::IncidenceChanger *changer)
2289{
2290 d->mChanger = changer;
2291}
2292
2293void Agenda::setHolidayMask(QList<bool> *mask)
2294{
2295 d->mHolidayMask = mask;
2296}
2297
2298void Agenda::contentsMousePressEvent(QMouseEvent *event)
2299{
2300 Q_UNUSED(event)
2301}
2302
2303QSize Agenda::sizeHint() const
2304{
2305 if (d->mAllDayMode) {
2306 return QWidget::sizeHint();
2307 } else {
2308 return {parentWidget()->width(), static_cast<int>(d->mGridSpacingY * d->mRows)};
2309 }
2310}
2311
2312QScrollBar *Agenda::verticalScrollBar() const
2313{
2314 return d->mScrollArea->verticalScrollBar();
2315}
2316
2317QScrollArea *Agenda::scrollArea() const
2318{
2319 return d->mScrollArea;
2320}
2321
2322AgendaItem::List Agenda::agendaItems(const QString &uid) const
2323{
2324 return d->mAgendaItemsById.values(uid);
2325}
2326
2327AgendaScrollArea::AgendaScrollArea(bool isAllDay, AgendaView *agendaView, bool isInteractive, QWidget *parent)
2328 : QScrollArea(parent)
2329{
2330 if (isAllDay) {
2331 mAgenda = new Agenda(agendaView, this, 1, isInteractive);
2332 setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
2333 } else {
2334 mAgenda = new Agenda(agendaView, this, 1, 96, agendaView->preferences()->hourSize(), isInteractive);
2335 }
2336
2337 setWidgetResizable(true);
2338 setWidget(mAgenda);
2339 setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
2340
2341 mAgenda->setStartTime(agendaView->preferences()->dayBegins().time());
2342}
2343
2344AgendaScrollArea::~AgendaScrollArea() = default;
2345
2346Agenda *AgendaScrollArea::agenda() const
2347{
2348 return mAgenda;
2349}
2350
2351#include "moc_agenda.cpp"
Rights rights() const
Collection & parentCollection()
Id id() const
Collection::Id storageCollectionId() const
bool isValid() const
AgendaView is the agenda-like view that displays events in a single or multi-day view.
Definition agendaview.h:70
EventView is the abstract base class from which all other calendar views for event data are derived.
Definition eventview.h:69
QSharedPointer< Calendar > Ptr
static Incidence::Ptr createException(const Incidence::Ptr &incidence, const QDateTime &recurrenceId, bool thisAndFuture=false)
QSharedPointer< Event > Ptr
QSharedPointer< Incidence > Ptr
Q_SCRIPTABLE Q_NOREPLY void start()
QString i18nc(const char *context, const char *text, const TYPE &arg...)
QString i18n(const char *text, const TYPE &arg...)
AKONADI_CALENDAR_EXPORT KCalendarCore::Incidence::Ptr incidence(const Akonadi::Item &item)
char * toString(const EngineQuery &query)
Namespace EventViews provides facilities for displaying incidences, including events,...
Definition agenda.h:33
QList< QDate > DateList
void init(KXmlGuiWindow *window, KGameDifficulty *difficulty=nullptr)
QAction * end(const QObject *recvr, const char *slot, QObject *parent)
void information(QWidget *parent, const QString &text, const QString &title=QString(), const QString &dontShowAgainName=QString(), Options options=Notify)
void error(QWidget *parent, const QString &text, const QString &title, const KGuiItem &buttonOk, Options options=Notify)
const QList< QKeySequence > & next()
const QList< QKeySequence > & preferences()
void valueChanged(int value)
int blue() const const
QColor darker(int factor) const const
int green() const const
QColor lighter(int factor) const const
int red() const const
void setAlpha(int alpha)
bool sendEvent(QObject *receiver, QEvent *event)
QDate currentDate()
QDateTime currentDateTime()
QDate date() const const
QTime time() const const
const QMimeData * mimeData() const const
QPointF position() const const
void setDropAction(Qt::DropAction action)
MouseButtonPress
void accept()
void ignore()
Type type() const const
void setLineWidth(int)
void setFrameStyle(int style)
bool isRightToLeft()
Qt::KeyboardModifiers modifiers() const const
int key() const const
void append(QList< T > &&value)
iterator begin()
qsizetype count() const const
iterator end()
QList< T > & fill(parameter_type value, qsizetype size)
T & first()
bool isEmpty() const const
T & last()
QLocale system()
QPoint pos() const const
Q_EMITQ_EMIT
QMetaObject::Connection connect(const QObject *sender, PointerToMemberFunction signal, Functor functor)
virtual bool event(QEvent *e)
virtual bool eventFilter(QObject *watched, QEvent *event)
void installEventFilter(QObject *filterObj)
T qobject_cast(QObject *object)
void drawPixmap(const QPoint &point, const QPixmap &pixmap)
void setColor(ColorGroup group, ColorRole role, const QColor &color)
int x() const const
int y() const const
T * data() const const
QPoint toPoint() const const
const QSize & size() const const
Qt::MouseButton button() const const
QPointF position() const const
QString & append(QChar ch)
QString arg(Args &&... args) const const
QString number(double n, char format, int precision)
AlignRight
ArrowCursor
MoveAction
WheelFocus
Key_PageDown
ShiftModifier
RightButton
Horizontal
ScrollBarAlwaysOff
WA_OpaquePaintEvent
QFuture< ArgsType< Signal > > connect(Sender *sender, Signal signal)
QTime addSecs(int s) const const
int hour() const const
int minute() const const
int second() const const
bool setHMS(int h, int m, int s, int ms)
void timeout()
QPoint angleDelta() const const
QWidget(QWidget *parent, Qt::WindowFlags f)
void setAcceptDrops(bool on)
void setCursor(const QCursor &)
virtual bool event(QEvent *event) override
void setFocusPolicy(Qt::FocusPolicy policy)
void hide()
bool isHidden() const const
QPoint mapTo(const QWidget *parent, const QPoint &pos) const const
QPoint mapToParent(const QPoint &pos) const const
QRegion mask() const const
QWidget * parentWidget() const const
void move(const QPoint &)
void raise()
virtual void resizeEvent(QResizeEvent *event)
void setAttribute(Qt::WidgetAttribute attribute, bool on)
void setFixedSize(const QSize &s)
void show()
void update()
void updateGeometry()
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Mar 28 2025 11:49:04 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.