/* This file is part of KOrganizer. Copyright (c) 2001 Eitzenberger Thomas <thomas.eitzenberger@siemens.at> Parts of the source code have been copied from kdpdatebutton.cpp Copyright (c) 2003 Cornelius Schumacher <schumacher@kde.org> Copyright (C) 2003-2004 Reinhold Kainhofer <reinhold@kainhofer.com> This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. As a special exception, permission is given to link this program with any edition of TQt, and distribute the resulting executable, without including the source code for TQt in the source distribution. */ #include <tqevent.h> #include <tqpainter.h> #include <tqptrlist.h> #include <tdeglobal.h> #include <kdebug.h> #include <tdelocale.h> #include <kiconloader.h> #include <libkcal/vcaldrag.h> #include <libkcal/icaldrag.h> #include <libkcal/dndfactory.h> #include <libkcal/calendarresources.h> #include <libkcal/resourcecalendar.h> #include <kcalendarsystem.h> #include "koprefs.h" #include "koglobals.h" #include "kodialogmanager.h" #include "kodaymatrix.h" #include "kodaymatrix.moc" #ifndef NODND #include <tqcursor.h> #include <tdepopupmenu.h> #include <X11/Xlib.h> #undef FocusIn #undef KeyPress #undef None #undef Status #endif // ============================================================================ // D Y N A M I C T I P // ============================================================================ DynamicTip::DynamicTip( TQWidget * parent ) : TQToolTip( parent ) { mMatrix = static_cast<KODayMatrix *>( parent ); } void DynamicTip::maybeTip( const TQPoint &pos ) { //calculate which cell of the matrix the mouse is in TQRect sz = mMatrix->frameRect(); int dheight = sz.height() * 7 / 42; int dwidth = sz.width() / 7; int row = pos.y() / dheight; int col = pos.x() / dwidth; TQRect rct( col * dwidth, row * dheight, dwidth, dheight ); // kdDebug(5850) << "DynamicTip::maybeTip matrix cell index [" << // col << "][" << row << "] => " <<(col+row*7) << endl; //show holiday names only TQString str = mMatrix->getHolidayLabel( col + row * 7 ); if ( str.isEmpty() ) return; tip( rct, str ); } // ============================================================================ // K O D A Y M A T R I X // ============================================================================ const int KODayMatrix::NOSELECTION = -1000; const int KODayMatrix::NUMDAYS = 42; KODayMatrix::KODayMatrix( TQWidget *parent, const char *name ) : TQFrame( parent, name ), mCalendar( 0 ), mStartDate(), mPendingChanges( false ) { // initialize dynamic arrays mDays = new TQDate[ NUMDAYS ]; mDayLabels = new TQString[ NUMDAYS ]; mEvents = new int[ NUMDAYS ]; mToolTip = new DynamicTip( this ); mTodayMarginWidth = 2; mSelEnd = mSelStart = NOSELECTION; setBackgroundMode( NoBackground ); recalculateToday(); } void KODayMatrix::setCalendar( Calendar *cal ) { if ( mCalendar ) { mCalendar->unregisterObserver( this ); mCalendar->disconnect( this ); } mCalendar = cal; mCalendar->registerObserver( this ); CalendarResources *calres = dynamic_cast<CalendarResources*>( cal ); if ( calres ) { connect( calres, TQ_SIGNAL(signalResourceAdded(ResourceCalendar *)), TQ_SLOT(resourcesChanged()) ); connect( calres, TQ_SIGNAL(signalResourceModified( ResourceCalendar *)), TQ_SLOT(resourcesChanged()) ); connect( calres, TQ_SIGNAL(signalResourceDeleted(ResourceCalendar *)), TQ_SLOT(resourcesChanged()) ); } setAcceptDrops( mCalendar ); updateEvents(); } TQColor KODayMatrix::getShadedColor( const TQColor &color ) { TQColor shaded; int h = 0; int s = 0; int v = 0; color.hsv( &h, &s, &v ); s = s / 4; v = 192 + v / 4; shaded.setHsv( h, s, v ); return shaded; } KODayMatrix::~KODayMatrix() { if ( mCalendar ) mCalendar->unregisterObserver( this ); delete [] mDays; delete [] mDayLabels; delete [] mEvents; delete mToolTip; } void KODayMatrix::addSelectedDaysTo( DateList &selDays ) { kdDebug(5850) << "KODayMatrix::addSelectedDaysTo() - " << "mSelStart:" << mSelStart << endl; if ( mSelStart == NOSELECTION ) { return; } // cope with selection being out of matrix limits at top (< 0) int i0 = mSelStart; if ( i0 < 0 ) { for ( int i = i0; i < 0; i++ ) { selDays.append( mDays[ 0 ].addDays( i ) ); } i0 = 0; } // cope with selection being out of matrix limits at bottom (> NUMDAYS-1) if ( mSelEnd > NUMDAYS-1 ) { for ( int i = i0; i <= NUMDAYS - 1; i++ ) { selDays.append( mDays[ i ] ); } for ( int i = NUMDAYS; i < mSelEnd; i++ ) { selDays.append( mDays[ 0 ].addDays( i ) ); } } else { // apply normal routine to selection being entirely within matrix limits for ( int i = i0; i <= mSelEnd; i++ ) { selDays.append( mDays[ i ] ); } } } void KODayMatrix::setSelectedDaysFrom( const TQDate &start, const TQDate &end ) { if ( mStartDate.isValid() ) { mSelStart = mStartDate.daysTo( start ); mSelEnd = mStartDate.daysTo( end ); } } void KODayMatrix::clearSelection() { mSelEnd = mSelStart = NOSELECTION; } void KODayMatrix::recalculateToday() { if ( !mStartDate.isValid() ) return; mToday = -1; for ( int i = 0; i < NUMDAYS; i++ ) { mDays[ i ] = mStartDate.addDays( i ); mDayLabels[ i ] = TQString::number( KOGlobals::self()->calendarSystem()->day( mDays[i] )); // if today is in the currently displayed month, hilight today if ( mDays[ i ].year() == TQDate::currentDate().year() && mDays[ i ].month() == TQDate::currentDate().month() && mDays[ i ].day() == TQDate::currentDate().day() ) { mToday = i; } } // kdDegug(5850) << "Today is visible at "<< today << "." << endl; } /* slot */ void KODayMatrix::updateView() { updateView( mStartDate ); } void KODayMatrix::setUpdateNeeded() { mPendingChanges = true; } void KODayMatrix::updateView( const TQDate &actdate ) { kdDebug(5850) << "KODayMatrix::updateView() " << actdate << ", day start="<<mStartDate<< endl; if ( !actdate.isValid() ) return; //flag to indicate if the starting day of the matrix has changed by this call bool daychanged = false; // if a new startdate is to be set then apply Cornelius's calculation // of the first day to be shown if ( actdate != mStartDate ) { // reset index of selection according to shift of starting date from startdate to actdate if ( mSelStart != NOSELECTION ) { int tmp = actdate.daysTo( mStartDate ); //kdDebug(5850) << "Shift of Selection1: " << mSelStart << " - " << mSelEnd << " -> " << tmp << "(" << offset << ")" << endl; // shift selection if new one would be visible at least partly ! if ( mSelStart + tmp < NUMDAYS && mSelEnd + tmp >= 0 ) { // nested if is required for next X display pushed from a different month - correction required // otherwise, for month forward and backward, it must be avoided if( mSelStart > NUMDAYS || mSelStart < 0 ) mSelStart = mSelStart + tmp; if( mSelEnd > NUMDAYS || mSelEnd < 0 ) mSelEnd = mSelEnd + tmp; } } mStartDate = actdate; daychanged = true; } if ( daychanged ) { recalculateToday(); } // the calendar hasn't changed in the meantime and the selected range is still the same // so we can safe the expensive updateEvents() call if ( !daychanged && !mPendingChanges ) return; // TODO_Recurrence: If we just change the selection, but not the data, there's // no need to update the whole list of events... This is just a waste of // computational power (and it takes forever!) updateEvents(); for( int i = 0; i < NUMDAYS; i++ ) { //if it is a holy day then draw it red. Sundays are consider holidays, too TQStringList holidays = KOGlobals::self()->holiday( mDays[ i ] ); TQString holiStr = TQString(); if ( ( KOGlobals::self()->calendarSystem()->dayOfWeek( mDays[ i ] ) == KOGlobals::self()->calendarSystem()->weekDayOfPray() ) || !holidays.isEmpty() ) { if ( !holidays.isEmpty() ) holiStr = holidays.join( i18n("delimiter for joining holiday names", ", " ) ); if ( holiStr.isNull() ) holiStr = ""; } mHolidays[ i ] = holiStr; } } void KODayMatrix::updateEvents() { kdDebug( 5850 ) << k_funcinfo << endl; if ( !mCalendar ) return; for( int i = 0; i < NUMDAYS; i++ ) { // if events are set for the day then remember to draw it bold Event::List eventlist = mCalendar->events( mDays[ i ] ); int numEvents = eventlist.count(); Event::List::ConstIterator it; for( it = eventlist.begin(); it != eventlist.end(); ++it ) { Event *event = *it; ushort recurType = event->recurrenceType(); if ( ( recurType == Recurrence::rDaily && !KOPrefs::instance()->mDailyRecur ) || ( recurType == Recurrence::rWeekly && !KOPrefs::instance()->mWeeklyRecur ) ) { numEvents--; } } mEvents[ i ] = numEvents; } mPendingChanges = false; } const TQDate& KODayMatrix::getDate( int offset ) { if ( offset < 0 || offset > NUMDAYS - 1 ) { kdDebug(5850) << "Wrong offset (" << offset << ") in KODayMatrix::getDate(int)" << endl; return mDays[ 0 ]; } return mDays[ offset ]; } TQString KODayMatrix::getHolidayLabel( int offset ) { if ( offset < 0 || offset > NUMDAYS - 1 ) { kdDebug(5850) << "Wrong offset (" << offset << ") in KODayMatrix::getHolidayLabel(int)" << endl; return 0; } return mHolidays[ offset ]; } int KODayMatrix::getDayIndexFrom( int x, int y ) { return 7 * ( y / mDaySize.height() ) + ( KOGlobals::self()->reverseLayout() ? 6 - x / mDaySize.width() : x / mDaySize.width() ); } void KODayMatrix::calendarIncidenceAdded(Incidence * incidence) { Q_UNUSED( incidence ); mPendingChanges = true; } void KODayMatrix::calendarIncidenceChanged(Incidence * incidence) { Q_UNUSED( incidence ); mPendingChanges = true; } void KODayMatrix::calendarIncidenceDeleted(Incidence * incidence) { Q_UNUSED( incidence ); mPendingChanges = true; } void KODayMatrix::resourcesChanged() { mPendingChanges = true; } // ---------------------------------------------------------------------------- // M O U S E E V E N T H A N D L I N G // ---------------------------------------------------------------------------- void KODayMatrix::mousePressEvent( TQMouseEvent *e ) { mSelStart = getDayIndexFrom(e->x(), e->y()); if (mSelStart > NUMDAYS-1) mSelStart=NUMDAYS-1; mSelInit = mSelStart; } void KODayMatrix::mouseReleaseEvent( TQMouseEvent *e ) { int tmp = getDayIndexFrom(e->x(), e->y()); if (tmp > NUMDAYS-1) tmp=NUMDAYS-1; if (mSelInit > tmp) { mSelEnd = mSelInit; if (tmp != mSelStart) { mSelStart = tmp; repaint(); } } else { mSelStart = mSelInit; //repaint only if selection has changed if (tmp != mSelEnd) { mSelEnd = tmp; repaint(); } } DateList daylist; if ( mSelStart < 0 ) mSelStart = 0; for ( int i = mSelStart; i <= mSelEnd; ++i ) { daylist.append( mDays[i] ); } emit selected((const DateList)daylist); } void KODayMatrix::mouseMoveEvent( TQMouseEvent *e ) { int tmp = getDayIndexFrom(e->x(), e->y()); if (tmp > NUMDAYS-1) tmp=NUMDAYS-1; if (mSelInit > tmp) { mSelEnd = mSelInit; if ( tmp != mSelStart ) { mSelStart = tmp; repaint(); } } else { mSelStart = mSelInit; //repaint only if selection has changed if ( tmp != mSelEnd ) { mSelEnd = tmp; repaint(); } } } // ---------------------------------------------------------------------------- // D R A G ' N D R O P H A N D L I N G // ---------------------------------------------------------------------------- //----------------------------------------------------------------------------- // Drag and Drop handling -- based on the Troll Tech dirview example enum { DRAG_COPY = 0, DRAG_MOVE = 1, DRAG_CANCEL = 2 }; void KODayMatrix::dragEnterEvent( TQDragEnterEvent *e ) { #ifndef KORG_NODND if ( !ICalDrag::canDecode( e ) && !VCalDrag::canDecode( e ) ) { e->ignore(); return; } // some visual feedback // oldPalette = palette(); // setPalette(my_HilitePalette); // update(); #endif } void KODayMatrix::dragMoveEvent( TQDragMoveEvent *e ) { #ifndef KORG_NODND if ( !ICalDrag::canDecode( e ) && !VCalDrag::canDecode( e ) ) { e->ignore(); return; } e->accept(); #endif } void KODayMatrix::dragLeaveEvent( TQDragLeaveEvent * /*dl*/ ) { #ifndef KORG_NODND // setPalette(oldPalette); // update(); #endif } void KODayMatrix::dropEvent( TQDropEvent *e ) { #ifndef KORG_NODND kdDebug(5850) << "KODayMatrix::dropEvent(e) begin" << endl; if ( !mCalendar || ( !ICalDrag::canDecode( e ) && !VCalDrag::canDecode( e ) ) ) { e->ignore(); return; } DndFactory factory( mCalendar ); Event *event = factory.createDrop( e ); Todo *todo = factory.createDropTodo( e ); if ( !event && !todo ) { e->ignore(); return; } Todo *existingTodo = 0; Event *existingEvent = 0; // Find the incidence in the calendar, then we don't need the drag object any more if ( event ) existingEvent = mCalendar->event( event->uid() ); if ( todo ) existingTodo = mCalendar->todo( todo->uid() ); int action = DRAG_CANCEL; int root_x, root_y, win_x, win_y; uint keybstate; Window rootw, childw; XQueryPointer( tqt_xdisplay(), tqt_xrootwin(), &rootw, &childw, &root_x, &root_y, &win_x, &win_y, &keybstate ); if ( keybstate & ControlMask ) { action = DRAG_COPY; } else if ( keybstate & ShiftMask ) { action = DRAG_MOVE; } else { TDEPopupMenu *menu = new TDEPopupMenu( this ); if ( existingEvent || existingTodo ) { menu->insertItem( i18n("Move"), DRAG_MOVE, 0 ); if (existingEvent) menu->insertItem( KOGlobals::self()->smallIcon("edit-copy"), i18n("Copy"), DRAG_COPY, 1 ); } else { menu->insertItem( i18n("Add"), DRAG_MOVE, 0 ); } menu->insertSeparator(); menu->insertItem( KOGlobals::self()->smallIcon("cancel"), i18n("Cancel"), DRAG_CANCEL, 3 ); action = menu->exec( TQCursor::pos(), 0 ); } if ( action == DRAG_COPY || action == DRAG_MOVE ) { e->accept(); int idx = getDayIndexFrom( e->pos().x(), e->pos().y() ); if ( action == DRAG_COPY ) { if ( event ) emit incidenceDropped( event, mDays[idx] ); if ( todo ) emit incidenceDropped( todo, mDays[idx] ); } else if ( action == DRAG_MOVE ) { if ( event ) emit incidenceDroppedMove( event, mDays[idx] ); if ( todo ) emit incidenceDroppedMove( todo, mDays[idx] ); } } delete event; delete todo; #endif } // ---------------------------------------------------------------------------- // P A I N T E V E N T H A N D L I N G // ---------------------------------------------------------------------------- void KODayMatrix::paintEvent( TQPaintEvent * ) { // kdDebug(5850) << "KODayMatrix::paintEvent() BEGIN" << endl; TQPainter p; TQRect sz = frameRect(); TQPixmap pm( sz.size() ); int dheight = mDaySize.height(); int dwidth = mDaySize.width(); int row,col; int selw, selh; bool isRTL = KOGlobals::self()->reverseLayout(); TQColorGroup cg = palette().active(); p.begin( &pm, this ); pm.fill( cg.base() ); // draw topleft frame p.setPen( cg.mid() ); p.drawRect(0, 0, sz.width()-1, sz.height()-1); // don't paint over borders p.translate(1,1); // draw selected days with highlighted background color if (mSelStart != NOSELECTION) { row = mSelStart/7; // fix larger selections starting in the previous month if ( row < 0 && mSelEnd > 0 ) row = 0; col = mSelStart -row*7; TQColor selcol = KOPrefs::instance()->mHighlightColor; if ( row < 6 && row >= 0 ) { if (row == mSelEnd/7) { // Single row selection p.fillRect(isRTL ? (7 - (mSelEnd-mSelStart+1) - col)*dwidth : col*dwidth, row*dheight, (mSelEnd-mSelStart+1)*dwidth, dheight, selcol); } else { // draw first row to the right p.fillRect(isRTL ? 0 : col*dwidth, row*dheight, (7-col)*dwidth, dheight, selcol); // draw full block till last line selh = mSelEnd/7-row; if ( selh + row >= 6 ) selh = 6-row; if ( selh > 1 ) { p.fillRect(0, (row+1)*dheight, 7*dwidth, (selh-1)*dheight,selcol); } // draw last block from left to mSelEnd if ( mSelEnd/7 < 6 ) { selw = mSelEnd-7*(mSelEnd/7)+1; p.fillRect(isRTL ? (7-selw)*dwidth : 0, (row+selh)*dheight, selw*dwidth, dheight, selcol); } } } } // iterate over all days in the matrix and draw the day label in appropriate colors TQColor textColor = cg.text(); TQColor textColorShaded = getShadedColor( textColor ); TQColor actcol = textColorShaded; p.setPen(actcol); TQPen tmppen; for ( int i = 0; i < NUMDAYS; ++i ) { row = i/7; col = isRTL ? 6-(i-row*7) : i-row*7; // if it is the first day of a month switch color from normal to shaded and vice versa if ( KOGlobals::self()->calendarSystem()->day( mDays[i] ) == 1) { if (actcol == textColorShaded) { actcol = textColor; } else { actcol = textColorShaded; } p.setPen(actcol); } //Reset pen color after selected days block if (i == mSelEnd+1) { p.setPen(actcol); } bool holiday = ! KOGlobals::self()->isWorkDay( mDays[ i ] ); TQColor holidayColorShaded = getShadedColor( KOPrefs::instance()->mHolidayColor ); // if today then draw rectangle around day if (mToday == i) { tmppen = p.pen(); TQPen mTodayPen(p.pen()); mTodayPen.setWidth(mTodayMarginWidth); //draw red rectangle for holidays if (holiday) { if (actcol == textColor) { mTodayPen.setColor(KOPrefs::instance()->mHolidayColor); } else { mTodayPen.setColor(holidayColorShaded); } } //draw gray rectangle for today if in selection if (i >= mSelStart && i <= mSelEnd) { TQColor grey("grey"); mTodayPen.setColor(grey); } p.setPen(mTodayPen); p.drawRect(col*dwidth, row*dheight, dwidth, dheight); p.setPen(tmppen); } // if any events are on that day then draw it using a bold font if (mEvents[i] > 0) { TQFont myFont = font(); myFont.setBold(true); p.setFont(myFont); } // if it is a holiday then use the default holiday color if (holiday) { if (actcol == textColor) { p.setPen(KOPrefs::instance()->mHolidayColor); } else { p.setPen(holidayColorShaded); } } // draw selected days with special color if ( i >= mSelStart && i <= mSelEnd && !holiday ) { p.setPen( TQColor( "white" ) ); } p.drawText(col*dwidth, row*dheight, dwidth, dheight, TQt::AlignHCenter | TQt::AlignVCenter, mDayLabels[i]); // reset color to actual color if ( holiday ) { p.setPen(actcol); } // reset bold font to plain font if (mEvents[i] > 0) { TQFont myFont = font(); myFont.setBold(false); p.setFont(myFont); } } p.end(); bitBlt( this, 0, 0, &pm ); } // ---------------------------------------------------------------------------- // R E SI Z E E V E N T H A N D L I N G // ---------------------------------------------------------------------------- void KODayMatrix::resizeEvent( TQResizeEvent * ) { TQRect sz = frameRect(); mDaySize.setHeight( sz.height() * 7 / NUMDAYS ); mDaySize.setWidth( sz.width() / 7 ); } /* static */ TQPair<TQDate,TQDate> KODayMatrix::matrixLimits( const TQDate &month ) { const KCalendarSystem *calSys = KOGlobals::self()->calendarSystem(); TQDate d = month; calSys->setYMD( d, calSys->year( month ), calSys->month( month ), 1 ); const int dayOfWeek = calSys->dayOfWeek( d ); const int weekstart = TDEGlobal::locale()->weekStartDay(); d = d.addDays( weekstart - dayOfWeek ); if ( dayOfWeek == weekstart ) { d = d.addDays( -7 ); // Start on the second line } return qMakePair( d, d.addDays( NUMDAYS-1 ) ); }