You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2062 lines
64 KiB
2062 lines
64 KiB
/*
|
|
This file is part of KOrganizer.
|
|
Copyright (c) 2001 Cornelius Schumacher <schumacher@kde.org>
|
|
Copyright (C) 2003-2004 Reinhold Kainhofer <reinhold@kainhofer.com>
|
|
|
|
Marcus Bains line.
|
|
Copyright (c) 2001 Ali Rahimi
|
|
|
|
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 <assert.h>
|
|
|
|
#include <tqintdict.h>
|
|
#include <tqdatetime.h>
|
|
#include <tqapplication.h>
|
|
#include <tqpopupmenu.h>
|
|
#include <tqcursor.h>
|
|
#include <tqpainter.h>
|
|
#include <tqlabel.h>
|
|
|
|
#include <kdebug.h>
|
|
#include <tdelocale.h>
|
|
#include <kiconloader.h>
|
|
#include <tdeglobal.h>
|
|
#include <tdemessagebox.h>
|
|
|
|
#include "koagendaitem.h"
|
|
#include "koprefs.h"
|
|
#include "koglobals.h"
|
|
#include "komessagebox.h"
|
|
#include "incidencechanger.h"
|
|
#include "kohelper.h"
|
|
|
|
#include "koagenda.h"
|
|
#include "koagenda.moc"
|
|
#include <korganizer/baseview.h>
|
|
|
|
#include <libkcal/event.h>
|
|
#include <libkcal/todo.h>
|
|
#include <libkcal/dndfactory.h>
|
|
#include <libkcal/icaldrag.h>
|
|
#include <libkcal/vcaldrag.h>
|
|
#include <libkcal/calendar.h>
|
|
#include <libkcal/calendarresources.h>
|
|
#include <libkcal/calhelper.h>
|
|
#include <math.h>
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
MarcusBains::MarcusBains(KOAgenda *_agenda,const char *name )
|
|
: TQFrame(_agenda->viewport(), name), agenda(_agenda)
|
|
{
|
|
setLineWidth(0);
|
|
setMargin(0);
|
|
setBackgroundColor(TQt::red);
|
|
minutes = new TQTimer(this);
|
|
connect(minutes, TQ_SIGNAL(timeout()), this, TQ_SLOT(updateLocation()));
|
|
minutes->start(0, true);
|
|
|
|
mTimeBox = new TQLabel(this);
|
|
mTimeBox->setAlignment(TQt::AlignRight | TQt::AlignBottom);
|
|
TQPalette pal = mTimeBox->palette();
|
|
pal.setColor(TQColorGroup::Foreground, TQt::red);
|
|
mTimeBox->setPalette(pal);
|
|
mTimeBox->setAutoMask(true);
|
|
|
|
agenda->addChild(mTimeBox);
|
|
|
|
mOldTime = TQTime( 0, 0 );
|
|
mOldToday = -1;
|
|
}
|
|
|
|
MarcusBains::~MarcusBains()
|
|
{
|
|
delete minutes;
|
|
}
|
|
|
|
int MarcusBains::todayColumn()
|
|
{
|
|
TQDate currentDate = TQDate::currentDate();
|
|
|
|
DateList dateList = agenda->dateList();
|
|
DateList::ConstIterator it;
|
|
int col = 0;
|
|
for(it = dateList.begin(); it != dateList.end(); ++it) {
|
|
if((*it) == currentDate)
|
|
return KOGlobals::self()->reverseLayout() ?
|
|
agenda->columns() - 1 - col : col;
|
|
++col;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
void MarcusBains::updateLocation()
|
|
{
|
|
updateLocationRecalc();
|
|
}
|
|
|
|
void MarcusBains::updateLocationRecalc( bool recalculate )
|
|
{
|
|
TQTime tim = TQTime::currentTime();
|
|
if((tim.hour() == 0) && (mOldTime.hour()==23))
|
|
recalculate = true;
|
|
|
|
int mins = tim.hour()*60 + tim.minute();
|
|
int minutesPerCell = 24 * 60 / agenda->rows();
|
|
int y = int( mins * agenda->gridSpacingY() / minutesPerCell );
|
|
int today = recalculate ? todayColumn() : mOldToday;
|
|
int x = int( agenda->gridSpacingX() * today );
|
|
|
|
mOldTime = tim;
|
|
mOldToday = today;
|
|
|
|
bool hideIt = !( KOPrefs::instance()->mMarcusBainsEnabled );
|
|
|
|
if ( !isHidden() && ( hideIt || ( today < 0 ) ) ) {
|
|
hide();
|
|
mTimeBox->hide();
|
|
return;
|
|
}
|
|
|
|
if ( isHidden() && !hideIt ) {
|
|
show();
|
|
mTimeBox->show();
|
|
}
|
|
|
|
if ( recalculate ) setFixedSize( int( agenda->gridSpacingX() ), 1 );
|
|
agenda->moveChild( this, x, y );
|
|
raise();
|
|
|
|
if(recalculate)
|
|
mTimeBox->setFont(KOPrefs::instance()->mMarcusBainsFont);
|
|
|
|
TQString timeStr = TDEGlobal::locale()->formatTime(tim, KOPrefs::instance()->mMarcusBainsShowSeconds);
|
|
TQFontMetrics fm = fontMetrics();
|
|
mTimeBox->setText( timeStr );
|
|
TQSize sz( fm.width( timeStr + ' ' ), fm.height() );
|
|
mTimeBox->setFixedSize( sz );
|
|
|
|
if (y-mTimeBox->height()>=0) y-=mTimeBox->height(); else y++;
|
|
if (x-mTimeBox->width()+agenda->gridSpacingX() > 0)
|
|
x += int( agenda->gridSpacingX() - mTimeBox->width() - 1 );
|
|
else x++;
|
|
agenda->moveChild(mTimeBox,x,y);
|
|
mTimeBox->raise();
|
|
mTimeBox->setAutoMask(true);
|
|
|
|
minutes->start(1000,true);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
/*
|
|
Create an agenda widget with rows rows and columns columns.
|
|
*/
|
|
KOAgenda::KOAgenda( int columns, int rows, int rowSize, CalendarView *calendarView,
|
|
TQWidget *parent, const char *name, WFlags f )
|
|
: TQScrollView( parent, name, f ), mChanger( 0 )
|
|
{
|
|
mColumns = columns;
|
|
mRows = rows;
|
|
mGridSpacingY = rowSize;
|
|
if ( mGridSpacingY < 4 || mGridSpacingY > 30 ) {
|
|
mGridSpacingY = 10;
|
|
}
|
|
|
|
mCalendarView = calendarView;
|
|
|
|
mAllDayMode = false;
|
|
|
|
init();
|
|
|
|
viewport()->setMouseTracking(true);
|
|
}
|
|
|
|
/*
|
|
Create an agenda widget with columns columns and one row. This is used for
|
|
all-day events.
|
|
*/
|
|
KOAgenda::KOAgenda( int columns, CalendarView *calendarView, TQWidget *parent,
|
|
const char *name, WFlags f ) : TQScrollView( parent, name, f )
|
|
{
|
|
mColumns = columns;
|
|
mRows = 1;
|
|
mGridSpacingY = 24;
|
|
mAllDayMode = true;
|
|
mCalendarView = calendarView;
|
|
setVScrollBarMode( AlwaysOff );
|
|
|
|
init();
|
|
}
|
|
|
|
|
|
KOAgenda::~KOAgenda()
|
|
{
|
|
delete mMarcusBains;
|
|
}
|
|
|
|
|
|
Incidence *KOAgenda::selectedIncidence() const
|
|
{
|
|
return ( mSelectedItem ? mSelectedItem->incidence() : 0 );
|
|
}
|
|
|
|
|
|
TQDate KOAgenda::selectedIncidenceDate() const
|
|
{
|
|
return ( mSelectedItem ? mSelectedItem->itemDate() : TQDate() );
|
|
}
|
|
|
|
const TQString KOAgenda::lastSelectedUid() const
|
|
{
|
|
return mSelectedUid;
|
|
}
|
|
|
|
|
|
void KOAgenda::init()
|
|
{
|
|
mGridSpacingX = 100;
|
|
mDesiredGridSpacingY = KOPrefs::instance()->mHourSize;
|
|
if ( mDesiredGridSpacingY < 4 || mDesiredGridSpacingY > 30 ) {
|
|
mDesiredGridSpacingY = 10;
|
|
}
|
|
|
|
// make sure that there are not more than 24 per day
|
|
mGridSpacingY = (double)height() / (double)mRows;
|
|
if ( mGridSpacingY < mDesiredGridSpacingY ) {
|
|
mGridSpacingY = mDesiredGridSpacingY;
|
|
}
|
|
|
|
mResizeBorderWidth = 8;
|
|
mScrollBorderWidth = 8;
|
|
mScrollDelay = 30;
|
|
mScrollOffset = 10;
|
|
|
|
enableClipper( true );
|
|
|
|
// Grab key strokes for keyboard navigation of agenda. Seems to have no
|
|
// effect. Has to be fixed.
|
|
setFocusPolicy( TQWidget::WheelFocus );
|
|
|
|
connect( &mScrollUpTimer, TQ_SIGNAL( timeout() ), TQ_SLOT( scrollUp() ) );
|
|
connect( &mScrollDownTimer, TQ_SIGNAL( timeout() ), TQ_SLOT( scrollDown() ) );
|
|
|
|
mStartCell = TQPoint( 0, 0 );
|
|
mEndCell = TQPoint( 0, 0 );
|
|
|
|
mHasSelection = false;
|
|
mSelectionStartPoint = TQPoint( 0, 0 );
|
|
mSelectionStartCell = TQPoint( 0, 0 );
|
|
mSelectionEndCell = TQPoint( 0, 0 );
|
|
|
|
mOldLowerScrollValue = -1;
|
|
mOldUpperScrollValue = -1;
|
|
|
|
mClickedItem = 0;
|
|
|
|
mActionItem = 0;
|
|
mResPair = qMakePair( static_cast<ResourceCalendar *>( 0 ), TQString() );
|
|
mActionType = NOP;
|
|
mItemMoved = false;
|
|
|
|
mSelectedItem = 0;
|
|
mSelectedUid = TQString();
|
|
|
|
setAcceptDrops( true );
|
|
installEventFilter( this );
|
|
mItems.setAutoDelete( true );
|
|
mItemsToDelete.setAutoDelete( true );
|
|
|
|
resizeContents( int( mGridSpacingX * mColumns ),
|
|
int( mGridSpacingY * mRows ) );
|
|
|
|
viewport()->update();
|
|
viewport()->setBackgroundMode( NoBackground );
|
|
viewport()->setFocusPolicy( TQWidget::WheelFocus );
|
|
|
|
setMinimumSize( 30, int( mGridSpacingY + 1 ) );
|
|
// setMaximumHeight(mGridSpacingY * mRows + 5);
|
|
|
|
// Disable horizontal scrollbar. This is a hack. The geometry should be
|
|
// controlled in a way that the contents horizontally always fits. Then it is
|
|
// not necessary to turn off the scrollbar.
|
|
setHScrollBarMode( AlwaysOff );
|
|
|
|
setStartTime( KOPrefs::instance()->mDayBegins.time() );
|
|
|
|
calculateWorkingHours();
|
|
|
|
connect( verticalScrollBar(), TQ_SIGNAL( valueChanged( int ) ),
|
|
TQ_SLOT( checkScrollBoundaries( int ) ) );
|
|
|
|
// Create the Marcus Bains line.
|
|
if( mAllDayMode ) {
|
|
mMarcusBains = 0;
|
|
} else {
|
|
mMarcusBains = new MarcusBains( this );
|
|
addChild( mMarcusBains );
|
|
}
|
|
|
|
mTypeAhead = false;
|
|
mTypeAheadReceiver = 0;
|
|
|
|
mReturnPressed = false;
|
|
}
|
|
|
|
|
|
void KOAgenda::clear()
|
|
{
|
|
// kdDebug(5850) << "KOAgenda::clear()" << endl;
|
|
|
|
KOAgendaItem *item;
|
|
for ( item = mItems.first(); item != 0; item = mItems.next() ) {
|
|
removeChild( item );
|
|
}
|
|
mItems.clear();
|
|
mItemsToDelete.clear();
|
|
|
|
mSelectedItem = 0;
|
|
|
|
clearSelection();
|
|
}
|
|
|
|
|
|
void KOAgenda::clearSelection()
|
|
{
|
|
mHasSelection = false;
|
|
mActionType = NOP;
|
|
updateContents();
|
|
}
|
|
|
|
void KOAgenda::marcus_bains()
|
|
{
|
|
if(mMarcusBains) mMarcusBains->updateLocationRecalc( true );
|
|
}
|
|
|
|
|
|
void KOAgenda::changeColumns(int columns)
|
|
{
|
|
if (columns == 0) {
|
|
kdDebug(5850) << "KOAgenda::changeColumns() called with argument 0" << endl;
|
|
return;
|
|
}
|
|
|
|
clear();
|
|
mColumns = columns;
|
|
// setMinimumSize(mColumns * 10, mGridSpacingY + 1);
|
|
// init();
|
|
// update();
|
|
|
|
TQResizeEvent event( size(), size() );
|
|
|
|
TQApplication::sendEvent( this, &event );
|
|
}
|
|
|
|
/*
|
|
This is the eventFilter function, which gets all events from the KOAgendaItems
|
|
contained in the agenda. It has to handle moving and resizing for all items.
|
|
*/
|
|
bool KOAgenda::eventFilter ( TQObject *object, TQEvent *event )
|
|
{
|
|
// kdDebug(5850) << "KOAgenda::eventFilter() " << int( event->type() ) << endl;
|
|
|
|
switch( event->type() ) {
|
|
case TQEvent::MouseButtonPress:
|
|
case TQEvent::MouseButtonDblClick:
|
|
case TQEvent::MouseButtonRelease:
|
|
case TQEvent::MouseMove:
|
|
return eventFilter_mouse( object, static_cast<TQMouseEvent*>( event ) );
|
|
#ifndef TQT_NO_WHEELEVENT
|
|
case TQEvent::Wheel:
|
|
return eventFilter_wheel( object, static_cast<TQWheelEvent*>( event ) );
|
|
#endif
|
|
case TQEvent::KeyPress:
|
|
case TQEvent::KeyRelease:
|
|
return eventFilter_key( object, static_cast<TQKeyEvent*>( event ) );
|
|
|
|
case ( TQEvent::Leave ):
|
|
if ( !mActionItem )
|
|
setCursor( arrowCursor );
|
|
if ( object == viewport() )
|
|
emit leaveAgenda();
|
|
return true;
|
|
|
|
case TQEvent::Enter:
|
|
emit enterAgenda();
|
|
return TQScrollView::eventFilter( object, event );
|
|
|
|
#ifndef KORG_NODND
|
|
case TQEvent::DragEnter:
|
|
case TQEvent::DragMove:
|
|
case TQEvent::DragLeave:
|
|
case TQEvent::Drop:
|
|
// case TQEvent::DragResponse:
|
|
return eventFilter_drag(object, static_cast<TQDropEvent*>(event));
|
|
#endif
|
|
|
|
default:
|
|
return TQScrollView::eventFilter( object, event );
|
|
}
|
|
}
|
|
|
|
bool KOAgenda::eventFilter_drag( TQObject *object, TQDropEvent *de )
|
|
{
|
|
#ifndef KORG_NODND
|
|
TQPoint viewportPos;
|
|
if ( object != viewport() && object != this ) {
|
|
viewportPos = static_cast<TQWidget*>( object )->mapToParent( de->pos() );
|
|
} else {
|
|
viewportPos = de->pos();
|
|
}
|
|
|
|
switch ( de->type() ) {
|
|
case TQEvent::DragEnter:
|
|
case TQEvent::DragMove:
|
|
if ( ICalDrag::canDecode( de ) || VCalDrag::canDecode( de ) ) {
|
|
|
|
DndFactory factory( mCalendar );
|
|
Todo *todo = factory.createDropTodo( de );
|
|
if ( todo ) {
|
|
de->accept();
|
|
delete todo;
|
|
} else {
|
|
de->ignore();
|
|
}
|
|
return true;
|
|
} else return false;
|
|
break;
|
|
case TQEvent::DragLeave:
|
|
return false;
|
|
break;
|
|
case TQEvent::Drop:
|
|
{
|
|
if ( !ICalDrag::canDecode( de ) && !VCalDrag::canDecode( de ) ) {
|
|
return false;
|
|
}
|
|
|
|
DndFactory factory( mCalendar );
|
|
Todo *todo = factory.createDropTodo( de );
|
|
|
|
if ( todo ) {
|
|
de->acceptAction();
|
|
TQPoint pos;
|
|
// FIXME: This is a bad hack, as the viewportToContents seems to be off by
|
|
// 2000 (which is the left upper corner of the viewport). It works correctly
|
|
// for agendaItems.
|
|
if ( object == this ) {
|
|
pos = viewportPos + TQPoint( contentsX(), contentsY() );
|
|
} else {
|
|
pos = viewportToContents( viewportPos );
|
|
}
|
|
TQPoint gpos = contentsToGrid( pos );
|
|
emit droppedToDo( todo, gpos, mAllDayMode );
|
|
return true;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case TQEvent::DragResponse:
|
|
default:
|
|
break;
|
|
}
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
bool KOAgenda::eventFilter_key( TQObject *, TQKeyEvent *ke )
|
|
{
|
|
// kdDebug(5850) << "KOAgenda::eventFilter_key() " << ke->type() << endl;
|
|
|
|
// If Return is pressed bring up an editor for the current selected time span.
|
|
if ( ke->key() == Key_Return ) {
|
|
if ( ke->type() == TQEvent::KeyPress ) mReturnPressed = true;
|
|
else if ( ke->type() == TQEvent::KeyRelease ) {
|
|
if ( mReturnPressed ) {
|
|
emitNewEventForSelection();
|
|
mReturnPressed = false;
|
|
return true;
|
|
} else {
|
|
mReturnPressed = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Ignore all input that does not produce any output
|
|
if ( ke->text().isEmpty() ) return false;
|
|
|
|
if ( ke->type() == TQEvent::KeyPress || ke->type() == TQEvent::KeyRelease ) {
|
|
switch ( ke->key() ) {
|
|
case Key_Escape:
|
|
case Key_Return:
|
|
case Key_Enter:
|
|
case Key_Tab:
|
|
case Key_Backtab:
|
|
case Key_Left:
|
|
case Key_Right:
|
|
case Key_Up:
|
|
case Key_Down:
|
|
case Key_Backspace:
|
|
case Key_Delete:
|
|
case Key_Prior:
|
|
case Key_Next:
|
|
case Key_Home:
|
|
case Key_End:
|
|
case Key_Control:
|
|
case Key_Meta:
|
|
case Key_Alt:
|
|
break;
|
|
default:
|
|
mTypeAheadEvents.append( new TQKeyEvent( ke->type(), ke->key(),
|
|
ke->ascii(), ke->state(),
|
|
ke->text(), ke->isAutoRepeat(),
|
|
ke->count() ) );
|
|
if ( !mTypeAhead ) {
|
|
mTypeAhead = true;
|
|
emitNewEventForSelection();
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void KOAgenda::emitNewEventForSelection()
|
|
{
|
|
TQPair<ResourceCalendar *, TQString>p = mCalendarView->viewSubResourceCalendar();
|
|
emit newEventSignal( p.first, p.second );
|
|
}
|
|
|
|
void KOAgenda::finishTypeAhead()
|
|
{
|
|
// kdDebug(5850) << "KOAgenda::finishTypeAhead()" << endl;
|
|
if ( typeAheadReceiver() ) {
|
|
for( TQEvent *e = mTypeAheadEvents.first(); e;
|
|
e = mTypeAheadEvents.next() ) {
|
|
// kdDebug(5850) << "sendEvent() " << int( typeAheadReceiver() ) << endl;
|
|
TQApplication::sendEvent( typeAheadReceiver(), e );
|
|
}
|
|
}
|
|
mTypeAheadEvents.clear();
|
|
mTypeAhead = false;
|
|
}
|
|
#ifndef TQT_NO_WHEELEVENT
|
|
bool KOAgenda::eventFilter_wheel ( TQObject *object, TQWheelEvent *e )
|
|
{
|
|
TQPoint viewportPos;
|
|
bool accepted=false;
|
|
if ( ( e->state() & ShiftButton) == ShiftButton ) {
|
|
if ( object != viewport() ) {
|
|
viewportPos = ( (TQWidget *) object )->mapToParent( e->pos() );
|
|
} else {
|
|
viewportPos = e->pos();
|
|
}
|
|
//kdDebug(5850)<< "KOAgenda::eventFilter_wheel: type:"<<
|
|
// e->type()<<" delta: "<< e->delta()<< endl;
|
|
emit zoomView( -e->delta() ,
|
|
contentsToGrid( viewportToContents( viewportPos ) ),
|
|
TQt::Horizontal );
|
|
accepted=true;
|
|
}
|
|
|
|
if ( ( e->state() & ControlButton ) == ControlButton ){
|
|
if ( object != viewport() ) {
|
|
viewportPos = ( (TQWidget *)object )->mapToParent( e->pos() );
|
|
} else {
|
|
viewportPos = e->pos();
|
|
}
|
|
emit zoomView( -e->delta() ,
|
|
contentsToGrid( viewportToContents( viewportPos ) ),
|
|
TQt::Vertical );
|
|
emit mousePosSignal(gridToContents(contentsToGrid(viewportToContents( viewportPos ))));
|
|
accepted=true;
|
|
}
|
|
if (accepted ) e->accept();
|
|
return accepted;
|
|
}
|
|
#endif
|
|
bool KOAgenda::eventFilter_mouse(TQObject *object, TQMouseEvent *me)
|
|
{
|
|
TQPoint viewportPos;
|
|
if (object != viewport()) {
|
|
viewportPos = ((TQWidget *)object)->mapToParent(me->pos());
|
|
} else {
|
|
viewportPos = me->pos();
|
|
}
|
|
|
|
switch (me->type()) {
|
|
case TQEvent::MouseButtonPress:
|
|
// kdDebug(5850) << "koagenda: filtered button press" << endl;
|
|
if (object != viewport()) {
|
|
if (me->button() == TQt::RightButton) {
|
|
mClickedItem = dynamic_cast<KOAgendaItem *>(object);
|
|
if (mClickedItem) {
|
|
selectItem(mClickedItem);
|
|
emit showIncidencePopupSignal( mCalendar,
|
|
mClickedItem->incidence(),
|
|
mClickedItem->itemDate() );
|
|
} else {
|
|
return TQScrollView::eventFilter( object, me ); // pass through for use by multiagenda
|
|
}
|
|
} else {
|
|
KOAgendaItem* item = dynamic_cast<KOAgendaItem *>(object);
|
|
if (item) {
|
|
Incidence *incidence = item->incidence();
|
|
if ( incidence->isReadOnly() ) {
|
|
mActionItem = 0;
|
|
mResPair = qMakePair( static_cast<ResourceCalendar *>( 0 ), TQString() );
|
|
} else {
|
|
mActionItem = item;
|
|
mResPair = CalHelper::incSubResourceCalendar( mCalendar, incidence );
|
|
startItemAction(viewportPos);
|
|
}
|
|
// Warning: do selectItem() as late as possible, since all
|
|
// sorts of things happen during this call. Some can lead to
|
|
// this filter being run again and mActionItem being set to
|
|
// null.
|
|
selectItem( item );
|
|
} else {
|
|
return TQScrollView::eventFilter( object, me ); // pass through for use by multiagenda
|
|
}
|
|
}
|
|
} else {
|
|
if ( me->button() == TQt::RightButton ) {
|
|
// if mouse pointer is not in selection, select the cell below the cursor
|
|
TQPoint gpos = contentsToGrid( viewportToContents( viewportPos ) );
|
|
if ( !ptInSelection( gpos ) ) {
|
|
mSelectionStartCell = gpos;
|
|
mSelectionEndCell = gpos;
|
|
mHasSelection = true;
|
|
emit newStartSelectSignal();
|
|
emit newTimeSpanSignal( mSelectionStartCell, mSelectionEndCell );
|
|
updateContents();
|
|
}
|
|
showNewEventPopupSignal();
|
|
} else {
|
|
// if mouse pointer is in selection, don't change selection
|
|
TQPoint gpos = contentsToGrid( viewportToContents( viewportPos ) );
|
|
if ( !ptInSelection( gpos ) ) {
|
|
selectItem(0);
|
|
mActionItem = 0;
|
|
mResPair = qMakePair( static_cast<ResourceCalendar *>( 0 ), TQString() );
|
|
setCursor(arrowCursor);
|
|
startSelectAction(viewportPos);
|
|
}
|
|
}
|
|
return TQScrollView::eventFilter( object, me ); // pass through for use by multiagenda
|
|
}
|
|
break;
|
|
|
|
case TQEvent::MouseButtonRelease:
|
|
if (mActionItem) {
|
|
endItemAction();
|
|
} else if ( mActionType == SELECT ) {
|
|
endSelectAction( viewportPos );
|
|
}
|
|
// This nasty gridToContents(contentsToGrid(..)) is needed to
|
|
// avoid an offset of a few pixels. Don't ask me why...
|
|
emit mousePosSignal( gridToContents(contentsToGrid(
|
|
viewportToContents( viewportPos ) ) ));
|
|
break;
|
|
|
|
case TQEvent::MouseMove: {
|
|
// This nasty gridToContents(contentsToGrid(..)) is needed to
|
|
// avoid an offset of a few pixels. Don't ask me why...
|
|
TQPoint indicatorPos = gridToContents(contentsToGrid(
|
|
viewportToContents( viewportPos )));
|
|
if (object != viewport()) {
|
|
KOAgendaItem *moveItem = dynamic_cast<KOAgendaItem *>(object);
|
|
if (moveItem && !moveItem->incidence()->isReadOnly() ) {
|
|
if (!mActionItem)
|
|
setNoActionCursor(moveItem,viewportPos);
|
|
else {
|
|
performItemAction(viewportPos);
|
|
|
|
if ( mActionType == MOVE ) {
|
|
// show cursor at the current begin of the item
|
|
KOAgendaItem *firstItem = mActionItem->firstMultiItem();
|
|
if (!firstItem) firstItem = mActionItem;
|
|
indicatorPos = gridToContents( TQPoint( firstItem->cellXLeft(),
|
|
firstItem->cellYTop() ) );
|
|
|
|
} else if ( mActionType == RESIZEBOTTOM ) {
|
|
// RESIZETOP is handled correctly, only resizebottom works differently
|
|
indicatorPos = gridToContents( TQPoint( mActionItem->cellXLeft(),
|
|
mActionItem->cellYBottom()+1 ) );
|
|
}
|
|
|
|
} // If we have an action item
|
|
} // If move item && !read only
|
|
} else {
|
|
if ( mActionType == SELECT ) {
|
|
performSelectAction( viewportPos );
|
|
|
|
// show cursor at end of timespan
|
|
if ( ((mStartCell.y() < mEndCell.y()) && (mEndCell.x() >= mStartCell.x())) ||
|
|
(mEndCell.x() > mStartCell.x()) )
|
|
indicatorPos = gridToContents( TQPoint(mEndCell.x(), mEndCell.y()+1) );
|
|
else
|
|
indicatorPos = gridToContents( mEndCell );
|
|
}
|
|
}
|
|
emit mousePosSignal( indicatorPos );
|
|
break; }
|
|
|
|
case TQEvent::MouseButtonDblClick:
|
|
if (object == viewport()) {
|
|
selectItem(0);
|
|
TQPair<ResourceCalendar *, TQString>p = mCalendarView->viewSubResourceCalendar();
|
|
emit newEventSignal( p.first, p.second );
|
|
} else {
|
|
KOAgendaItem *doubleClickedItem = dynamic_cast<KOAgendaItem *>( object );
|
|
if ( doubleClickedItem ) {
|
|
selectItem( doubleClickedItem );
|
|
emit editIncidenceSignal( doubleClickedItem->incidence(), doubleClickedItem->itemDate() );
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool KOAgenda::ptInSelection( TQPoint gpos ) const
|
|
{
|
|
if ( !mHasSelection ) {
|
|
return false;
|
|
} else if ( gpos.x()<mSelectionStartCell.x() || gpos.x()>mSelectionEndCell.x() ) {
|
|
return false;
|
|
} else if ( (gpos.x()==mSelectionStartCell.x()) && (gpos.y()<mSelectionStartCell.y()) ) {
|
|
return false;
|
|
} else if ( (gpos.x()==mSelectionEndCell.x()) && (gpos.y()>mSelectionEndCell.y()) ) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void KOAgenda::startSelectAction( const TQPoint &viewportPos )
|
|
{
|
|
emit newStartSelectSignal();
|
|
|
|
mActionType = SELECT;
|
|
mSelectionStartPoint = viewportPos;
|
|
mHasSelection = true;
|
|
|
|
TQPoint pos = viewportToContents( viewportPos );
|
|
TQPoint gpos = contentsToGrid( pos );
|
|
|
|
// Store new selection
|
|
mStartCell = gpos;
|
|
mEndCell = gpos;
|
|
mSelectionStartCell = gpos;
|
|
mSelectionEndCell = gpos;
|
|
|
|
updateContents();
|
|
}
|
|
|
|
void KOAgenda::performSelectAction(const TQPoint& viewportPos)
|
|
{
|
|
TQPoint pos = viewportToContents( viewportPos );
|
|
TQPoint gpos = contentsToGrid( pos );
|
|
|
|
TQPoint clipperPos = clipper()->
|
|
mapFromGlobal(viewport()->mapToGlobal(viewportPos));
|
|
|
|
// Scroll if cursor was moved to upper or lower end of agenda.
|
|
if (clipperPos.y() < mScrollBorderWidth) {
|
|
mScrollUpTimer.start(mScrollDelay);
|
|
} else if (visibleHeight() - clipperPos.y() <
|
|
mScrollBorderWidth) {
|
|
mScrollDownTimer.start(mScrollDelay);
|
|
} else {
|
|
mScrollUpTimer.stop();
|
|
mScrollDownTimer.stop();
|
|
}
|
|
|
|
if ( gpos != mEndCell ) {
|
|
mEndCell = gpos;
|
|
if ( mStartCell.x()>mEndCell.x() ||
|
|
( mStartCell.x()==mEndCell.x() && mStartCell.y()>mEndCell.y() ) ) {
|
|
// backward selection
|
|
mSelectionStartCell = mEndCell;
|
|
mSelectionEndCell = mStartCell;
|
|
} else {
|
|
mSelectionStartCell = mStartCell;
|
|
mSelectionEndCell = mEndCell;
|
|
}
|
|
|
|
updateContents();
|
|
}
|
|
}
|
|
|
|
void KOAgenda::endSelectAction( const TQPoint ¤tPos )
|
|
{
|
|
mScrollUpTimer.stop();
|
|
mScrollDownTimer.stop();
|
|
|
|
mActionType = NOP;
|
|
|
|
emit newTimeSpanSignal( mSelectionStartCell, mSelectionEndCell );
|
|
|
|
if ( KOPrefs::instance()->mSelectionStartsEditor ) {
|
|
if ( ( mSelectionStartPoint - currentPos ).manhattanLength() >
|
|
TQApplication::startDragDistance() ) {
|
|
emitNewEventForSelection();
|
|
}
|
|
}
|
|
}
|
|
|
|
KOAgenda::MouseActionType KOAgenda::isInResizeArea( bool horizontal,
|
|
const TQPoint &pos, KOAgendaItem*item )
|
|
{
|
|
if (!item) return NOP;
|
|
TQPoint gridpos = contentsToGrid( pos );
|
|
TQPoint contpos = gridToContents( gridpos +
|
|
TQPoint( (KOGlobals::self()->reverseLayout())?1:0, 0 ) );
|
|
|
|
//kdDebug(5850)<<"contpos="<<contpos<<", pos="<<pos<<", gpos="<<gpos<<endl;
|
|
//kdDebug(5850)<<"clXLeft="<<clXLeft<<", clXRight="<<clXRight<<endl;
|
|
|
|
if ( horizontal ) {
|
|
int clXLeft = item->cellXLeft();
|
|
int clXRight = item->cellXRight();
|
|
if ( KOGlobals::self()->reverseLayout() ) {
|
|
int tmp = clXLeft;
|
|
clXLeft = clXRight;
|
|
clXRight = tmp;
|
|
}
|
|
int gridDistanceX = int( pos.x() - contpos.x() );
|
|
if (gridDistanceX < mResizeBorderWidth && clXLeft == gridpos.x() ) {
|
|
if ( KOGlobals::self()->reverseLayout() ) return RESIZERIGHT;
|
|
else return RESIZELEFT;
|
|
} else if ((mGridSpacingX - gridDistanceX) < mResizeBorderWidth &&
|
|
clXRight == gridpos.x() ) {
|
|
if ( KOGlobals::self()->reverseLayout() ) return RESIZELEFT;
|
|
else return RESIZERIGHT;
|
|
} else {
|
|
return MOVE;
|
|
}
|
|
} else {
|
|
int gridDistanceY = int( pos.y() - contpos.y() );
|
|
if (gridDistanceY < mResizeBorderWidth &&
|
|
item->cellYTop() == gridpos.y() &&
|
|
!item->firstMultiItem() ) {
|
|
return RESIZETOP;
|
|
} else if ((mGridSpacingY - gridDistanceY) < mResizeBorderWidth &&
|
|
item->cellYBottom() == gridpos.y() &&
|
|
!item->lastMultiItem() ) {
|
|
return RESIZEBOTTOM;
|
|
} else {
|
|
return MOVE;
|
|
}
|
|
}
|
|
}
|
|
|
|
void KOAgenda::startItemAction(const TQPoint& viewportPos)
|
|
{
|
|
TQPoint pos = viewportToContents( viewportPos );
|
|
mStartCell = contentsToGrid( pos );
|
|
mEndCell = mStartCell;
|
|
|
|
bool noResize = ( mActionItem->incidence()->type() == "Todo");
|
|
|
|
mActionType = MOVE;
|
|
if ( !noResize ) {
|
|
mActionType = isInResizeArea( mAllDayMode, pos, mActionItem );
|
|
}
|
|
|
|
|
|
mActionItem->startMove();
|
|
setActionCursor( mActionType, true );
|
|
}
|
|
|
|
void KOAgenda::performItemAction(const TQPoint& viewportPos)
|
|
{
|
|
// kdDebug(5850) << "viewportPos: " << viewportPos.x() << "," << viewportPos.y() << endl;
|
|
// TQPoint point = viewport()->mapToGlobal(viewportPos);
|
|
// kdDebug(5850) << "Global: " << point.x() << "," << point.y() << endl;
|
|
// point = clipper()->mapFromGlobal(point);
|
|
// kdDebug(5850) << "clipper: " << point.x() << "," << point.y() << endl;
|
|
// kdDebug(5850) << "visible height: " << visibleHeight() << endl;
|
|
TQPoint pos = viewportToContents( viewportPos );
|
|
// kdDebug(5850) << "contents: " << x << "," << y << "\n" << endl;
|
|
TQPoint gpos = contentsToGrid( pos );
|
|
TQPoint clipperPos = clipper()->
|
|
mapFromGlobal(viewport()->mapToGlobal(viewportPos));
|
|
|
|
// Cursor left active agenda area.
|
|
// This starts a drag.
|
|
if ( clipperPos.y() < 0 || clipperPos.y() > visibleHeight() ||
|
|
clipperPos.x() < 0 || clipperPos.x() > visibleWidth() ) {
|
|
if ( mActionType == MOVE ) {
|
|
mScrollUpTimer.stop();
|
|
mScrollDownTimer.stop();
|
|
mActionItem->resetMove();
|
|
placeSubCells( mActionItem );
|
|
emit startDragSignal( mActionItem->incidence() );
|
|
setCursor( arrowCursor );
|
|
mActionItem = 0;
|
|
mResPair = qMakePair( static_cast<ResourceCalendar *>( 0 ), TQString() );
|
|
mActionType = NOP;
|
|
mItemMoved = false;
|
|
return;
|
|
}
|
|
} else {
|
|
setActionCursor( mActionType );
|
|
}
|
|
|
|
// Scroll if item was moved to upper or lower end of agenda.
|
|
if (clipperPos.y() < mScrollBorderWidth) {
|
|
mScrollUpTimer.start(mScrollDelay);
|
|
} else if (visibleHeight() - clipperPos.y() <
|
|
mScrollBorderWidth) {
|
|
mScrollDownTimer.start(mScrollDelay);
|
|
} else {
|
|
mScrollUpTimer.stop();
|
|
mScrollDownTimer.stop();
|
|
}
|
|
|
|
// Move or resize item if necessary
|
|
if ( mEndCell != gpos ) {
|
|
if ( !mItemMoved ) {
|
|
if ( !mChanger ||
|
|
!mChanger->beginChange( mActionItem->incidence(), mResPair.first, mResPair.second ) ) {
|
|
KMessageBox::information( this, i18n("Unable to lock item for "
|
|
"modification. You cannot make any changes."),
|
|
i18n("Locking Failed"), "AgendaLockingFailed" );
|
|
mScrollUpTimer.stop();
|
|
mScrollDownTimer.stop();
|
|
mActionItem->resetMove();
|
|
placeSubCells( mActionItem );
|
|
setCursor( arrowCursor );
|
|
mActionItem = 0;
|
|
mResPair = qMakePair( static_cast<ResourceCalendar *>( 0 ), TQString() );
|
|
mActionType = NOP;
|
|
mItemMoved = false;
|
|
return;
|
|
}
|
|
mItemMoved = true;
|
|
}
|
|
mActionItem->raise();
|
|
if (mActionType == MOVE) {
|
|
// Move all items belonging to a multi item
|
|
KOAgendaItem *firstItem = mActionItem->firstMultiItem();
|
|
if (!firstItem) firstItem = mActionItem;
|
|
KOAgendaItem *lastItem = mActionItem->lastMultiItem();
|
|
if (!lastItem) lastItem = mActionItem;
|
|
TQPoint deltapos = gpos - mEndCell;
|
|
KOAgendaItem *moveItem = firstItem;
|
|
while (moveItem) {
|
|
bool changed=false;
|
|
if ( deltapos.x()!=0 ) {
|
|
moveItem->moveRelative( deltapos.x(), 0 );
|
|
changed=true;
|
|
}
|
|
// in agenda's all day view don't try to move multi items, since there are none
|
|
if ( moveItem==firstItem && !mAllDayMode ) { // is the first item
|
|
int newY = deltapos.y() + moveItem->cellYTop();
|
|
// If event start moved earlier than 0:00, it starts the previous day
|
|
if ( newY<0 ) {
|
|
moveItem->expandTop( -moveItem->cellYTop() );
|
|
// prepend a new item at ( x-1, rows()+newY to rows() )
|
|
KOAgendaItem *newFirst = firstItem->prevMoveItem();
|
|
// cell's y values are first and last cell of the bar, so if newY=-1, they need to be the same
|
|
if (newFirst) {
|
|
newFirst->setCellXY(moveItem->cellXLeft()-1, rows()+newY, rows()-1);
|
|
mItems.append( newFirst );
|
|
moveItem->resize( int( mGridSpacingX * newFirst->cellWidth() ),
|
|
int( mGridSpacingY * newFirst->cellHeight() ));
|
|
TQPoint cpos = gridToContents( TQPoint( newFirst->cellXLeft(), newFirst->cellYTop() ) );
|
|
addChild( newFirst, cpos.x(), cpos.y() );
|
|
} else {
|
|
newFirst = insertItem( moveItem->incidence(), moveItem->itemDate(),
|
|
moveItem->cellXLeft()-1, rows()+newY, rows()-1, moveItem->itemPos(), moveItem->itemCount() ) ;
|
|
}
|
|
if (newFirst) newFirst->show();
|
|
moveItem->prependMoveItem(newFirst);
|
|
firstItem=newFirst;
|
|
} else if ( newY>=rows() ) {
|
|
// If event start is moved past 24:00, it starts the next day
|
|
// erase current item (i.e. remove it from the multiItem list)
|
|
firstItem = moveItem->nextMultiItem();
|
|
moveItem->hide();
|
|
mItems.take( mItems.find( moveItem ) );
|
|
removeChild( moveItem );
|
|
mActionItem->removeMoveItem(moveItem);
|
|
moveItem=firstItem;
|
|
// adjust next day's item
|
|
if (moveItem) moveItem->expandTop( rows()-newY );
|
|
} else {
|
|
moveItem->expandTop(deltapos.y());
|
|
}
|
|
changed=true;
|
|
}
|
|
if ( !moveItem->lastMultiItem() && !mAllDayMode ) { // is the last item
|
|
int newY = deltapos.y()+moveItem->cellYBottom();
|
|
if (newY<0) {
|
|
// erase current item
|
|
lastItem = moveItem->prevMultiItem();
|
|
moveItem->hide();
|
|
mItems.take( mItems.find(moveItem) );
|
|
removeChild( moveItem );
|
|
moveItem->removeMoveItem( moveItem );
|
|
moveItem = lastItem;
|
|
moveItem->expandBottom(newY+1);
|
|
} else if (newY>=rows()) {
|
|
moveItem->expandBottom( rows()-moveItem->cellYBottom()-1 );
|
|
// append item at ( x+1, 0 to newY-rows() )
|
|
KOAgendaItem *newLast = lastItem->nextMoveItem();
|
|
if (newLast) {
|
|
newLast->setCellXY( moveItem->cellXLeft()+1, 0, newY-rows()-1 );
|
|
mItems.append(newLast);
|
|
moveItem->resize( int( mGridSpacingX * newLast->cellWidth() ),
|
|
int( mGridSpacingY * newLast->cellHeight() ));
|
|
TQPoint cpos = gridToContents( TQPoint( newLast->cellXLeft(), newLast->cellYTop() ) ) ;
|
|
addChild( newLast, cpos.x(), cpos.y() );
|
|
} else {
|
|
newLast = insertItem( moveItem->incidence(), moveItem->itemDate(),
|
|
moveItem->cellXLeft()+1, 0, newY-rows()-1, moveItem->itemPos(), moveItem->itemCount() ) ;
|
|
}
|
|
moveItem->appendMoveItem( newLast );
|
|
newLast->show();
|
|
lastItem = newLast;
|
|
} else {
|
|
moveItem->expandBottom( deltapos.y() );
|
|
}
|
|
changed=true;
|
|
}
|
|
if (changed) {
|
|
adjustItemPosition( moveItem );
|
|
}
|
|
moveItem = moveItem->nextMultiItem();
|
|
}
|
|
} else if (mActionType == RESIZETOP) {
|
|
if (mEndCell.y() <= mActionItem->cellYBottom()) {
|
|
mActionItem->expandTop(gpos.y() - mEndCell.y());
|
|
adjustItemPosition( mActionItem );
|
|
}
|
|
} else if (mActionType == RESIZEBOTTOM) {
|
|
if (mEndCell.y() >= mActionItem->cellYTop()) {
|
|
mActionItem->expandBottom(gpos.y() - mEndCell.y());
|
|
adjustItemPosition( mActionItem );
|
|
}
|
|
} else if (mActionType == RESIZELEFT) {
|
|
if (mEndCell.x() <= mActionItem->cellXRight()) {
|
|
mActionItem->expandLeft( gpos.x() - mEndCell.x() );
|
|
adjustItemPosition( mActionItem );
|
|
}
|
|
} else if (mActionType == RESIZERIGHT) {
|
|
if (mEndCell.x() >= mActionItem->cellXLeft()) {
|
|
mActionItem->expandRight(gpos.x() - mEndCell.x());
|
|
adjustItemPosition( mActionItem );
|
|
}
|
|
}
|
|
mEndCell = gpos;
|
|
}
|
|
}
|
|
|
|
void KOAgenda::endItemAction()
|
|
{
|
|
// kdDebug(5850) << "KOAgenda::endItemAction() " << endl;
|
|
mActionType = NOP;
|
|
mScrollUpTimer.stop();
|
|
mScrollDownTimer.stop();
|
|
setCursor( arrowCursor );
|
|
bool multiModify = false;
|
|
// FIXME: do the cloning here...
|
|
Incidence* inc = mActionItem->incidence();
|
|
|
|
if ( mStartCell.x() == mEndCell.x() && mStartCell.y() == mEndCell.y() ) {
|
|
// not really moved, so stop any change
|
|
if ( mItemMoved ) {
|
|
mItemMoved = false;
|
|
mChanger->endChange( inc, mResPair.first, mResPair.second );
|
|
}
|
|
}
|
|
|
|
if ( mItemMoved ) {
|
|
Incidence *incToChange = inc;
|
|
if ( mActionItem->incidence()->doesRecur() ) {
|
|
Incidence* oldIncSaved = inc->clone();
|
|
KOGlobals::WhichOccurrences chosenOption;
|
|
incToChange = mCalendarView->singleOccurrenceOrAll( inc,
|
|
KOGlobals::EDIT,
|
|
chosenOption,
|
|
mActionItem->itemDate() );
|
|
|
|
if ( chosenOption == KOGlobals::ONLY_THIS_ONE ||
|
|
chosenOption == KOGlobals::ONLY_FUTURE ) {
|
|
|
|
// FIXME Prompt for this...it is quite possible that the user does not want to broadcast the change
|
|
// That prompting dialog will require the ability to suppress/override the mChanger->endChange GroupWare communication though.
|
|
int autoAnswerGroupWare = 1; // Send all possible GroupWare messages without prompting
|
|
|
|
// Store modification information in case it is needed to recreate the changes with a new actionitem...
|
|
int mai_xl = mActionItem->cellXLeft();
|
|
int mai_xr = mActionItem->cellXRight();
|
|
int mai_yt = mActionItem->cellYTop();
|
|
int mai_yb = mActionItem->cellYBottom();
|
|
|
|
multiModify = true;
|
|
emit startMultiModify( i18n("Dissociate event from recurrence") );
|
|
enableAgendaUpdate( false );
|
|
|
|
mChanger->addIncidence( incToChange, mResPair.first, mResPair.second, this, autoAnswerGroupWare );
|
|
enableAgendaUpdate( true );
|
|
KOGlobals::WhatChanged wc = chosenOption == KOGlobals::ONLY_THIS_ONE ?
|
|
KOGlobals::RECURRENCE_MODIFIED_ONE_ONLY :
|
|
KOGlobals::RECURRENCE_MODIFIED_ALL_FUTURE;
|
|
|
|
mChanger->changeIncidence( oldIncSaved, inc, wc, this, autoAnswerGroupWare );
|
|
|
|
// mActionItem does not exist any more, seeing as we just got done deleting it
|
|
// (by deleting/replacing the original incidence it was created from through
|
|
// user modification of said incidence) above!
|
|
// Therefore we have to find the new KOAgendaItem that matches the new incidence
|
|
// Then we can apply the saved X/Y settings from the original move operation as shown.
|
|
|
|
KOAgendaItem *koai_insertedItem;
|
|
for ( koai_insertedItem = mItems.first(); koai_insertedItem; koai_insertedItem = mItems.next() ) {
|
|
if (koai_insertedItem->incidence() == incToChange) {
|
|
selectItem( koai_insertedItem );
|
|
mSelectedItem->startMove();
|
|
mSelectedItem->setCellY(mai_yt, mai_yb);
|
|
mSelectedItem->setCellX(mai_xl, mai_xr);
|
|
mActionItem = mSelectedItem;
|
|
//mSelectedItem->endMove();
|
|
break;
|
|
}
|
|
}
|
|
|
|
mActionItem->dissociateFromMultiItem();
|
|
mActionItem->setIncidence( incToChange );
|
|
}
|
|
}
|
|
|
|
if ( incToChange ) {
|
|
mActionItem->endMove();
|
|
KOAgendaItem *placeItem = mActionItem->firstMultiItem();
|
|
if ( !placeItem ) {
|
|
placeItem = mActionItem;
|
|
}
|
|
|
|
KOAgendaItem *modif = placeItem;
|
|
|
|
TQPtrList<KOAgendaItem> oldconflictItems = placeItem->conflictItems();
|
|
KOAgendaItem *item;
|
|
for ( item = oldconflictItems.first(); item != 0;
|
|
item = oldconflictItems.next() ) {
|
|
placeSubCells( item );
|
|
}
|
|
while ( placeItem ) {
|
|
placeSubCells( placeItem );
|
|
placeItem = placeItem->nextMultiItem();
|
|
}
|
|
|
|
// Notify about change
|
|
// the agenda view will apply the changes to the actual Incidence*!
|
|
mChanger->endChange( inc, mResPair.first, mResPair.second );
|
|
emit itemModified( modif );
|
|
} else {
|
|
|
|
mActionItem->resetMove();
|
|
placeSubCells( mActionItem );
|
|
|
|
// the item was moved, but not further modified, since it's not recurring
|
|
// make sure the view updates anyhow, with the right item
|
|
mChanger->endChange( inc, mResPair.first, mResPair.second );
|
|
emit itemModified( mActionItem );
|
|
}
|
|
}
|
|
|
|
mActionItem = 0;
|
|
mResPair = qMakePair( static_cast<ResourceCalendar *>( 0 ), TQString() );
|
|
mItemMoved = false;
|
|
|
|
if ( multiModify ) {
|
|
emit endMultiModify();
|
|
}
|
|
|
|
kdDebug(5850) << "KOAgenda::endItemAction() done" << endl;
|
|
}
|
|
|
|
void KOAgenda::setActionCursor( int actionType, bool acting )
|
|
{
|
|
switch ( actionType ) {
|
|
case MOVE:
|
|
if (acting) setCursor( sizeAllCursor );
|
|
else setCursor( arrowCursor );
|
|
break;
|
|
case RESIZETOP:
|
|
case RESIZEBOTTOM:
|
|
setCursor( sizeVerCursor );
|
|
break;
|
|
case RESIZELEFT:
|
|
case RESIZERIGHT:
|
|
setCursor( sizeHorCursor );
|
|
break;
|
|
default:
|
|
setCursor( arrowCursor );
|
|
}
|
|
}
|
|
|
|
void KOAgenda::setNoActionCursor( KOAgendaItem *moveItem, const TQPoint& viewportPos )
|
|
{
|
|
// kdDebug(5850) << "viewportPos: " << viewportPos.x() << "," << viewportPos.y() << endl;
|
|
// TQPoint point = viewport()->mapToGlobal(viewportPos);
|
|
// kdDebug(5850) << "Global: " << point.x() << "," << point.y() << endl;
|
|
// point = clipper()->mapFromGlobal(point);
|
|
// kdDebug(5850) << "clipper: " << point.x() << "," << point.y() << endl;
|
|
|
|
TQPoint pos = viewportToContents( viewportPos );
|
|
bool noResize = (moveItem && moveItem->incidence() &&
|
|
moveItem->incidence()->type() == "Todo");
|
|
|
|
KOAgenda::MouseActionType resizeType = MOVE;
|
|
if ( !noResize ) resizeType = isInResizeArea( mAllDayMode, pos , moveItem);
|
|
setActionCursor( resizeType );
|
|
}
|
|
|
|
|
|
/** calculate the width of the column subcells of the given item
|
|
*/
|
|
double KOAgenda::calcSubCellWidth( KOAgendaItem *item )
|
|
{
|
|
TQPoint pt, pt1;
|
|
pt = gridToContents( TQPoint( item->cellXLeft(), item->cellYTop() ) );
|
|
pt1 = gridToContents( TQPoint( item->cellXLeft(), item->cellYTop() ) +
|
|
TQPoint( 1, 1 ) );
|
|
pt1 -= pt;
|
|
int maxSubCells = item->subCells();
|
|
double newSubCellWidth;
|
|
if ( mAllDayMode ) {
|
|
newSubCellWidth = double( pt1.y() ) / maxSubCells;
|
|
} else {
|
|
newSubCellWidth = double( pt1.x() ) / maxSubCells;
|
|
}
|
|
return newSubCellWidth;
|
|
}
|
|
|
|
void KOAgenda::adjustItemPosition( KOAgendaItem *item )
|
|
{
|
|
if (!item) return;
|
|
item->resize( int( mGridSpacingX * item->cellWidth() ),
|
|
int( mGridSpacingY * item->cellHeight() ) );
|
|
int clXLeft = item->cellXLeft();
|
|
if ( KOGlobals::self()->reverseLayout() )
|
|
clXLeft = item->cellXRight() + 1;
|
|
TQPoint cpos = gridToContents( TQPoint( clXLeft, item->cellYTop() ) );
|
|
moveChild( item, cpos.x(), cpos.y() );
|
|
}
|
|
|
|
void KOAgenda::placeAgendaItem( KOAgendaItem *item, double subCellWidth )
|
|
{
|
|
// kdDebug(5850) << "KOAgenda::placeAgendaItem(): " << item->incidence()->summary()
|
|
// << " subCellWidth: " << subCellWidth << endl;
|
|
|
|
// "left" upper corner, no subcells yet, RTL layouts have right/left switched, widths are negative then
|
|
TQPoint pt = gridToContents( TQPoint( item->cellXLeft(), item->cellYTop() ) );
|
|
// right lower corner
|
|
TQPoint pt1 = gridToContents( TQPoint( item->cellXLeft() + item->cellWidth(),
|
|
item->cellYBottom()+1 ) );
|
|
|
|
double subCellPos = item->subCell() * subCellWidth;
|
|
|
|
// we need to add 0.01 to make sure we don't loose one pixed due to
|
|
// numerics (i.e. if it would be x.9998, we want the integer, not rounded down.
|
|
double delta=0.01;
|
|
if (subCellWidth<0) delta=-delta;
|
|
int height, width, xpos, ypos;
|
|
if (mAllDayMode) {
|
|
width = pt1.x()-pt.x();
|
|
height = int( subCellPos + subCellWidth + delta ) - int( subCellPos );
|
|
xpos = pt.x();
|
|
ypos = pt.y() + int( subCellPos );
|
|
} else {
|
|
width = int( subCellPos + subCellWidth + delta ) - int( subCellPos );
|
|
height = pt1.y()-pt.y();
|
|
xpos = pt.x() + int( subCellPos );
|
|
ypos = pt.y();
|
|
}
|
|
if ( KOGlobals::self()->reverseLayout() ) { // RTL language/layout
|
|
xpos += width;
|
|
width = -width;
|
|
}
|
|
if ( height<0 ) { // BTT (bottom-to-top) layout ?!?
|
|
ypos += height;
|
|
height = -height;
|
|
}
|
|
item->resize( width, height );
|
|
moveChild( item, xpos, ypos );
|
|
}
|
|
|
|
/*
|
|
Place item in cell and take care that multiple items using the same cell do
|
|
not overlap. This method is not yet optimal. It doesn't use the maximum space
|
|
it can get in all cases.
|
|
At the moment the method has a bug: When an item is placed only the sub cell
|
|
widths of the items are changed, which are within the Y region the item to
|
|
place spans. When the sub cell width change of one of this items affects a
|
|
cell, where other items are, which do not overlap in Y with the item to place,
|
|
the display gets corrupted, although the corruption looks quite nice.
|
|
*/
|
|
void KOAgenda::placeSubCells( KOAgendaItem *placeItem )
|
|
{
|
|
#if 0
|
|
kdDebug(5850) << "KOAgenda::placeSubCells()" << endl;
|
|
if ( placeItem ) {
|
|
Incidence *event = placeItem->incidence();
|
|
if ( !event ) {
|
|
kdDebug(5850) << " event is 0" << endl;
|
|
} else {
|
|
kdDebug(5850) << " event: " << event->summary() << endl;
|
|
}
|
|
} else {
|
|
kdDebug(5850) << " placeItem is 0" << endl;
|
|
}
|
|
kdDebug(5850) << "KOAgenda::placeSubCells()..." << endl;
|
|
#endif
|
|
|
|
TQPtrList<KOrg::CellItem> cells;
|
|
KOAgendaItem *item;
|
|
for ( item = mItems.first(); item != 0; item = mItems.next() ) {
|
|
cells.append( item );
|
|
}
|
|
|
|
TQPtrList<KOrg::CellItem> items = KOrg::CellItem::placeItem( cells,
|
|
placeItem );
|
|
|
|
placeItem->setConflictItems( TQPtrList<KOAgendaItem>() );
|
|
double newSubCellWidth = calcSubCellWidth( placeItem );
|
|
KOrg::CellItem *i;
|
|
for ( i = items.first(); i; i = items.next() ) {
|
|
item = static_cast<KOAgendaItem *>( i );
|
|
placeAgendaItem( item, newSubCellWidth );
|
|
item->addConflictItem( placeItem );
|
|
placeItem->addConflictItem( item );
|
|
}
|
|
if ( items.isEmpty() ) {
|
|
placeAgendaItem( placeItem, newSubCellWidth );
|
|
}
|
|
placeItem->update();
|
|
}
|
|
|
|
int KOAgenda::columnWidth( int column )
|
|
{
|
|
int start = gridToContents( TQPoint( column, 0 ) ).x();
|
|
if (KOGlobals::self()->reverseLayout() )
|
|
column--;
|
|
else
|
|
column++;
|
|
int end = gridToContents( TQPoint( column, 0 ) ).x();
|
|
return end - start;
|
|
}
|
|
/*
|
|
Draw grid in the background of the agenda.
|
|
*/
|
|
void KOAgenda::drawContents(TQPainter* p, int cx, int cy, int cw, int ch)
|
|
{
|
|
TQPixmap db(cw, ch);
|
|
db.fill(KOPrefs::instance()->mAgendaBgColor);
|
|
TQPainter dbp(&db);
|
|
dbp.translate(-cx,-cy);
|
|
|
|
// kdDebug(5850) << "KOAgenda::drawContents()" << endl;
|
|
double lGridSpacingY = mGridSpacingY*2;
|
|
|
|
// Highlight working hours
|
|
if (mWorkingHoursEnable) {
|
|
TQPoint pt1( cx, mWorkingHoursYTop );
|
|
TQPoint pt2( cx+cw, mWorkingHoursYBottom );
|
|
if ( pt2.x() >= pt1.x() /*&& pt2.y() >= pt1.y()*/) {
|
|
int gxStart = contentsToGrid( pt1 ).x();
|
|
int gxEnd = contentsToGrid( pt2 ).x();
|
|
// correct start/end for rtl layouts
|
|
if ( gxStart > gxEnd ) {
|
|
int tmp = gxStart;
|
|
gxStart = gxEnd;
|
|
gxEnd = tmp;
|
|
}
|
|
int xoffset = ( KOGlobals::self()->reverseLayout()?1:0 );
|
|
while( gxStart <= gxEnd ) {
|
|
int xStart = gridToContents( TQPoint( gxStart+xoffset, 0 ) ).x();
|
|
int xWidth = columnWidth( gxStart ) + 1;
|
|
if ( pt2.y() < pt1.y() ) {
|
|
// overnight working hours
|
|
if ( ( (gxStart==0) && !mHolidayMask->at(mHolidayMask->count()-1) ) ||
|
|
( (gxStart>0) && (gxStart<int(mHolidayMask->count())) && (!mHolidayMask->at(gxStart-1) ) ) ) {
|
|
if ( pt2.y() > cy ) {
|
|
dbp.fillRect( xStart, cy, xWidth, pt2.y() - cy + 1,
|
|
KOPrefs::instance()->mWorkingHoursColor);
|
|
}
|
|
}
|
|
if ( (gxStart < int(mHolidayMask->count()-1)) && (!mHolidayMask->at(gxStart)) ) {
|
|
if ( pt1.y() < cy + ch - 1 ) {
|
|
dbp.fillRect( xStart, pt1.y(), xWidth, cy + ch - pt1.y() + 1,
|
|
KOPrefs::instance()->mWorkingHoursColor);
|
|
}
|
|
}
|
|
} else {
|
|
// last entry in holiday mask denotes the previous day not visible (needed for overnight shifts)
|
|
if ( gxStart < int(mHolidayMask->count()-1) && !mHolidayMask->at(gxStart)) {
|
|
dbp.fillRect( xStart, pt1.y(), xWidth, pt2.y() - pt1.y() + 1,
|
|
KOPrefs::instance()->mWorkingHoursColor );
|
|
}
|
|
}
|
|
++gxStart;
|
|
}
|
|
}
|
|
}
|
|
|
|
// draw selection
|
|
if ( mHasSelection ) {
|
|
TQPoint pt, pt1;
|
|
|
|
if ( mSelectionEndCell.x() > mSelectionStartCell.x() ) { // multi day selection
|
|
// draw start day
|
|
pt = gridToContents( mSelectionStartCell );
|
|
pt1 = gridToContents( TQPoint( mSelectionStartCell.x() + 1, mRows + 1 ) );
|
|
dbp.fillRect( TQRect( pt, pt1 ), KOPrefs::instance()->mHighlightColor );
|
|
// draw all other days between the start day and the day of the selection end
|
|
for ( int c = mSelectionStartCell.x() + 1; c < mSelectionEndCell.x(); ++c ) {
|
|
pt = gridToContents( TQPoint( c, 0 ) );
|
|
pt1 = gridToContents( TQPoint( c + 1, mRows + 1 ) );
|
|
dbp.fillRect( TQRect( pt, pt1 ), KOPrefs::instance()->mHighlightColor );
|
|
}
|
|
// draw end day
|
|
pt = gridToContents( TQPoint( mSelectionEndCell.x(), 0 ) );
|
|
pt1 = gridToContents( mSelectionEndCell + TQPoint(1,1) );
|
|
dbp.fillRect( TQRect( pt, pt1), KOPrefs::instance()->mHighlightColor );
|
|
} else { // single day selection
|
|
pt = gridToContents( mSelectionStartCell );
|
|
pt1 = gridToContents( mSelectionEndCell + TQPoint(1,1) );
|
|
dbp.fillRect( TQRect( pt, pt1 ), KOPrefs::instance()->mHighlightColor );
|
|
}
|
|
}
|
|
|
|
TQPen hourPen( KOPrefs::instance()->mAgendaBgColor.dark( 150 ) );
|
|
TQPen halfHourPen( KOPrefs::instance()->mAgendaBgColor.dark( 125 ) );
|
|
dbp.setPen( hourPen );
|
|
|
|
// Draw vertical lines of grid, start with the last line not yet visible
|
|
// kdDebug(5850) << "drawContents cx: " << cx << " cy: " << cy << " cw: " << cw << " ch: " << ch << endl;
|
|
double x = ( int( cx / mGridSpacingX ) ) * mGridSpacingX;
|
|
while (x < cx + cw) {
|
|
dbp.drawLine( int( x ), cy, int( x ), cy + ch );
|
|
x+=mGridSpacingX;
|
|
}
|
|
|
|
// Draw horizontal lines of grid
|
|
double y = ( int( cy / (2*lGridSpacingY) ) ) * 2 * lGridSpacingY;
|
|
while (y < cy + ch) {
|
|
// kdDebug(5850) << " y: " << y << endl;
|
|
dbp.drawLine( cx, int( y ), cx + cw, int( y ) );
|
|
y += 2 * lGridSpacingY;
|
|
}
|
|
y = ( 2 * int( cy / (2*lGridSpacingY) ) + 1) * lGridSpacingY;
|
|
dbp.setPen( halfHourPen );
|
|
while (y < cy + ch) {
|
|
// kdDebug(5850) << " y: " << y << endl;
|
|
dbp.drawLine( cx, int( y ), cx + cw, int( y ) );
|
|
y+=2*lGridSpacingY;
|
|
}
|
|
p->drawPixmap(cx,cy, db);
|
|
}
|
|
|
|
/*
|
|
Convert srcollview contents coordinates to agenda grid coordinates.
|
|
*/
|
|
TQPoint KOAgenda::contentsToGrid ( const TQPoint &pos ) const
|
|
{
|
|
int gx = int( KOGlobals::self()->reverseLayout() ?
|
|
mColumns - pos.x()/mGridSpacingX : pos.x()/mGridSpacingX );
|
|
int gy = int( pos.y()/mGridSpacingY );
|
|
return TQPoint( gx, gy );
|
|
}
|
|
|
|
/*
|
|
Convert agenda grid coordinates to scrollview contents coordinates.
|
|
*/
|
|
TQPoint KOAgenda::gridToContents( const TQPoint &gpos ) const
|
|
{
|
|
int x = int( KOGlobals::self()->reverseLayout() ?
|
|
(mColumns - gpos.x())*mGridSpacingX : gpos.x()*mGridSpacingX );
|
|
int y = int( gpos.y()*mGridSpacingY );
|
|
return TQPoint( x, y );
|
|
}
|
|
|
|
|
|
/*
|
|
Return Y coordinate corresponding to time. Coordinates are rounded to fit into
|
|
the grid.
|
|
*/
|
|
int KOAgenda::timeToY(const TQTime &time)
|
|
{
|
|
// kdDebug(5850) << "Time: " << time.toString() << endl;
|
|
int minutesPerCell = 24 * 60 / mRows;
|
|
// kdDebug(5850) << "minutesPerCell: " << minutesPerCell << endl;
|
|
int timeMinutes = time.hour() * 60 + time.minute();
|
|
// kdDebug(5850) << "timeMinutes: " << timeMinutes << endl;
|
|
int Y = (timeMinutes + (minutesPerCell / 2)) / minutesPerCell;
|
|
// kdDebug(5850) << "y: " << Y << endl;
|
|
// kdDebug(5850) << "\n" << endl;
|
|
return Y;
|
|
}
|
|
|
|
|
|
/*
|
|
Return time corresponding to cell y coordinate. Coordinates are rounded to
|
|
fit into the grid.
|
|
*/
|
|
TQTime KOAgenda::gyToTime(int gy)
|
|
{
|
|
// kdDebug(5850) << "gyToTime: " << gy << endl;
|
|
int secondsPerCell = 24 * 60 * 60/ mRows;
|
|
|
|
int timeSeconds = secondsPerCell * gy;
|
|
|
|
TQTime time( 0, 0, 0 );
|
|
if ( timeSeconds < 24 * 60 * 60 ) {
|
|
time = time.addSecs(timeSeconds);
|
|
} else {
|
|
time.setHMS( 23, 59, 59 );
|
|
}
|
|
// kdDebug(5850) << " gyToTime: " << time.toString() << endl;
|
|
|
|
return time;
|
|
}
|
|
|
|
TQMemArray<int> KOAgenda::minContentsY()
|
|
{
|
|
TQMemArray<int> minArray;
|
|
minArray.fill( timeToY( TQTime(23, 59) ), mSelectedDates.count() );
|
|
for ( KOAgendaItem *item = mItems.first();
|
|
item != 0; item = mItems.next() ) {
|
|
int ymin = item->cellYTop();
|
|
int index = item->cellXLeft();
|
|
if ( index>=0 && index<(int)(mSelectedDates.count()) ) {
|
|
if ( ymin < minArray[index] && mItemsToDelete.findRef( item ) == -1 )
|
|
minArray[index] = ymin;
|
|
}
|
|
}
|
|
|
|
return minArray;
|
|
}
|
|
|
|
TQMemArray<int> KOAgenda::maxContentsY()
|
|
{
|
|
TQMemArray<int> maxArray;
|
|
maxArray.fill( timeToY( TQTime(0, 0) ), mSelectedDates.count() );
|
|
for ( KOAgendaItem *item = mItems.first();
|
|
item != 0; item = mItems.next() ) {
|
|
int ymax = item->cellYBottom();
|
|
int index = item->cellXLeft();
|
|
if ( index>=0 && index<(int)(mSelectedDates.count()) ) {
|
|
if ( ymax > maxArray[index] && mItemsToDelete.findRef( item ) == -1 )
|
|
maxArray[index] = ymax;
|
|
}
|
|
}
|
|
|
|
return maxArray;
|
|
}
|
|
|
|
void KOAgenda::setStartTime( const TQTime &startHour )
|
|
{
|
|
double startPos = ( startHour.hour()/24. + startHour.minute()/1440. +
|
|
startHour.second()/86400. ) * mRows * gridSpacingY();
|
|
setContentsPos( 0, int( startPos ) );
|
|
}
|
|
|
|
|
|
/*
|
|
Insert KOAgendaItem into agenda.
|
|
*/
|
|
KOAgendaItem *KOAgenda::insertItem( Incidence *incidence, const TQDate &qd, int X,
|
|
int YTop, int YBottom, int itemPos, int itemCount )
|
|
{
|
|
if ( mAllDayMode ) {
|
|
kdDebug(5850) << "KOAgenda: calling insertItem in all-day mode is illegal." << endl;
|
|
return 0;
|
|
}
|
|
|
|
mActionType = NOP;
|
|
|
|
KOAgendaItem *agendaItem = new KOAgendaItem( mCalendar, incidence, qd, viewport(), itemPos, itemCount );
|
|
connect( agendaItem, TQ_SIGNAL( removeAgendaItem( KOAgendaItem * ) ),
|
|
TQ_SLOT( removeAgendaItem( KOAgendaItem * ) ) );
|
|
connect( agendaItem, TQ_SIGNAL( showAgendaItem( KOAgendaItem * ) ),
|
|
TQ_SLOT( showAgendaItem( KOAgendaItem * ) ) );
|
|
|
|
if ( YBottom <= YTop ) {
|
|
kdDebug(5850) << "KOAgenda::insertItem(): Text: " << agendaItem->text() << " YSize<0" << endl;
|
|
YBottom = YTop;
|
|
}
|
|
|
|
agendaItem->resize( int( ( X + 1 ) * mGridSpacingX ) -
|
|
int( X * mGridSpacingX ),
|
|
int( YTop * mGridSpacingY ) -
|
|
int( ( YBottom + 1 ) * mGridSpacingY ) );
|
|
agendaItem->setCellXY( X, YTop, YBottom );
|
|
agendaItem->setCellXRight( X );
|
|
agendaItem->setResourceColor( KOHelper::resourceColor( mCalendar, incidence ) );
|
|
agendaItem->installEventFilter( this );
|
|
|
|
addChild( agendaItem, int( X * mGridSpacingX ), int( YTop * mGridSpacingY ) );
|
|
mItems.append( agendaItem );
|
|
|
|
placeSubCells( agendaItem );
|
|
|
|
agendaItem->show();
|
|
|
|
marcus_bains();
|
|
|
|
return agendaItem;
|
|
}
|
|
|
|
/*
|
|
Insert all-day KOAgendaItem into agenda.
|
|
*/
|
|
KOAgendaItem *KOAgenda::insertAllDayItem( Incidence *event, const TQDate &qd,
|
|
int XBegin, int XEnd )
|
|
{
|
|
if ( !mAllDayMode ) {
|
|
kdDebug(5850) << "KOAgenda: calling insertAllDayItem in non all-day mode is illegal." << endl;
|
|
return 0;
|
|
}
|
|
|
|
mActionType = NOP;
|
|
|
|
KOAgendaItem *agendaItem = new KOAgendaItem( mCalendar, event, qd, viewport(), 1, 1 );
|
|
connect( agendaItem, TQ_SIGNAL( removeAgendaItem( KOAgendaItem* ) ),
|
|
TQ_SLOT( removeAgendaItem( KOAgendaItem* ) ) );
|
|
connect( agendaItem, TQ_SIGNAL( showAgendaItem( KOAgendaItem* ) ),
|
|
TQ_SLOT( showAgendaItem( KOAgendaItem* ) ) );
|
|
|
|
agendaItem->setCellXY( XBegin, 0, 0 );
|
|
agendaItem->setCellXRight( XEnd );
|
|
|
|
double startIt = mGridSpacingX * ( agendaItem->cellXLeft() );
|
|
double endIt = mGridSpacingX * ( agendaItem->cellWidth() +
|
|
agendaItem->cellXLeft() );
|
|
|
|
agendaItem->resize( int( endIt ) - int( startIt ), int( mGridSpacingY ) );
|
|
|
|
agendaItem->installEventFilter( this );
|
|
agendaItem->setResourceColor( KOHelper::resourceColor( mCalendar, event ) );
|
|
addChild( agendaItem, int( XBegin * mGridSpacingX ), 0 );
|
|
mItems.append( agendaItem );
|
|
|
|
placeSubCells( agendaItem );
|
|
|
|
agendaItem->show();
|
|
|
|
return agendaItem;
|
|
}
|
|
|
|
|
|
void KOAgenda::insertMultiItem( Event *event, const TQDate &qd, int XBegin, int XEnd,
|
|
int YTop, int YBottom )
|
|
{
|
|
if ( mAllDayMode ) {
|
|
kdDebug(5850) << "KOAgenda: calling insertMultiItem in all-day mode is illegal." << endl;
|
|
return;
|
|
}
|
|
mActionType = NOP;
|
|
|
|
int cellX,cellYTop,cellYBottom;
|
|
TQString newtext;
|
|
int width = XEnd - XBegin + 1;
|
|
int count = 0;
|
|
KOAgendaItem *current = 0;
|
|
TQPtrList<KOAgendaItem> multiItems;
|
|
const int visibleCount = mSelectedDates.first().daysTo( mSelectedDates.last() );
|
|
for ( cellX = XBegin; cellX <= XEnd; ++cellX ) {
|
|
++count;
|
|
//Only add the items that are visible.
|
|
if( cellX >= 0 && cellX <= visibleCount ) {
|
|
if ( cellX == XBegin ) {
|
|
cellYTop = YTop;
|
|
} else {
|
|
cellYTop = 0;
|
|
}
|
|
|
|
if ( cellX == XEnd ) {
|
|
cellYBottom = YBottom;
|
|
} else {
|
|
cellYBottom = rows() - 1;
|
|
}
|
|
|
|
newtext = TQString("(%1/%2): ").arg( count ).arg( width );
|
|
newtext.append( event->summary() );
|
|
|
|
current = insertItem( event, qd, cellX, cellYTop, cellYBottom, count, width );
|
|
current->setText( newtext );
|
|
multiItems.append( current );
|
|
}
|
|
}
|
|
TQPtrList<KOAgendaItem>::iterator it = multiItems.begin();
|
|
TQPtrList<KOAgendaItem>::iterator e = multiItems.end();
|
|
|
|
if ( it != e ) { // .first asserts if the list is empty
|
|
KOAgendaItem *first = multiItems.first();
|
|
KOAgendaItem *last = multiItems.last();
|
|
KOAgendaItem *prev = 0, *next = 0;
|
|
|
|
while ( it != e ) {
|
|
KOAgendaItem *item = *it;
|
|
++it;
|
|
next = ( it == e ) ? 0 : (*it);
|
|
if ( item ) {
|
|
item->setMultiItem( ( item == first ) ? 0 : first,
|
|
prev, next,
|
|
( item == last ) ? 0 : last );
|
|
}
|
|
prev = item;
|
|
}
|
|
}
|
|
|
|
marcus_bains();
|
|
}
|
|
|
|
void KOAgenda::removeIncidence( Incidence *incidence )
|
|
{
|
|
// First find all items to be deleted and store them
|
|
// in its own list. Otherwise removeAgendaItem will reset
|
|
// the current position and mess this up.
|
|
TQPtrList<KOAgendaItem> itemsToRemove;
|
|
|
|
KOAgendaItem *item = mItems.first();
|
|
while ( item ) {
|
|
if ( item->incidence() == incidence ) {
|
|
itemsToRemove.append( item );
|
|
}
|
|
item = mItems.next();
|
|
}
|
|
item = itemsToRemove.first();
|
|
while ( item ) {
|
|
removeAgendaItem( item );
|
|
item = itemsToRemove.next();
|
|
}
|
|
}
|
|
|
|
void KOAgenda::showAgendaItem( KOAgendaItem *agendaItem )
|
|
{
|
|
if ( !agendaItem ) {
|
|
return;
|
|
}
|
|
|
|
agendaItem->hide();
|
|
addChild( agendaItem );
|
|
if ( !mItems.containsRef( agendaItem ) ) {
|
|
mItems.append( agendaItem );
|
|
}
|
|
placeSubCells( agendaItem );
|
|
|
|
agendaItem->show();
|
|
}
|
|
|
|
bool KOAgenda::removeAgendaItem( KOAgendaItem *item )
|
|
{
|
|
// we found the item. Let's remove it and update the conflicts
|
|
bool taken = false;
|
|
KOAgendaItem *thisItem = item;
|
|
TQPtrList<KOAgendaItem> conflictItems = thisItem->conflictItems();
|
|
removeChild( thisItem );
|
|
|
|
int pos = mItems.find( thisItem );
|
|
if ( pos >= 0 ) {
|
|
mItems.take( pos );
|
|
taken = true;
|
|
}
|
|
|
|
KOAgendaItem *confitem;
|
|
for ( confitem = conflictItems.first(); confitem != 0;
|
|
confitem = conflictItems.next() ) {
|
|
// the item itself is also in its own conflictItems list!
|
|
if ( confitem != thisItem ) placeSubCells(confitem);
|
|
|
|
}
|
|
mItemsToDelete.append( thisItem );
|
|
TQTimer::singleShot( 0, this, TQ_SLOT( deleteItemsToDelete() ) );
|
|
return taken;
|
|
}
|
|
|
|
void KOAgenda::deleteItemsToDelete()
|
|
{
|
|
mItemsToDelete.clear();
|
|
}
|
|
|
|
/*TQSizePolicy KOAgenda::sizePolicy() const
|
|
{
|
|
// Thought this would make the all-day event agenda minimum size and the
|
|
// normal agenda take the remaining space. But it doesnt work. The TQSplitter
|
|
// dont seem to think that an Expanding widget needs more space than a
|
|
// Preferred one.
|
|
// But it doesnt hurt, so it stays.
|
|
if (mAllDayMode) {
|
|
return TQSizePolicy(TQSizePolicy::Expanding,TQSizePolicy::Preferred);
|
|
} else {
|
|
return TQSizePolicy(TQSizePolicy::Expanding,TQSizePolicy::Expanding);
|
|
}
|
|
}
|
|
*/
|
|
|
|
/*
|
|
Overridden from TQScrollView to provide proper resizing of KOAgendaItems.
|
|
*/
|
|
void KOAgenda::resizeEvent ( TQResizeEvent *ev )
|
|
{
|
|
// kdDebug(5850) << "KOAgenda::resizeEvent" << endl;
|
|
|
|
TQSize newSize( ev->size() );
|
|
if (mAllDayMode) {
|
|
mGridSpacingX = double( newSize.width() - 2 * frameWidth() ) / (double)mColumns;
|
|
mGridSpacingY = newSize.height() - 2 * frameWidth();
|
|
} else {
|
|
int scrollbarWidth = vScrollBarMode() != AlwaysOff ? verticalScrollBar()->width() : 0;
|
|
mGridSpacingX = double( newSize.width() - scrollbarWidth - 2 * frameWidth()) / double(mColumns);
|
|
// make sure that there are not more than 24 per day
|
|
mGridSpacingY = double(newSize.height() - 2 * frameWidth()) / double(mRows);
|
|
if ( mGridSpacingY < mDesiredGridSpacingY )
|
|
mGridSpacingY = mDesiredGridSpacingY;
|
|
}
|
|
calculateWorkingHours();
|
|
TQTimer::singleShot( 0, this, TQ_SLOT( resizeAllContents() ) );
|
|
emit gridSpacingYChanged( mGridSpacingY * 4 );
|
|
TQScrollView::resizeEvent(ev);
|
|
}
|
|
|
|
void KOAgenda::resizeAllContents()
|
|
{
|
|
double subCellWidth;
|
|
if ( mItems.count() > 0 ) {
|
|
KOAgendaItem *item;
|
|
if (mAllDayMode) {
|
|
for ( item=mItems.first(); item != 0; item=mItems.next() ) {
|
|
subCellWidth = calcSubCellWidth( item );
|
|
placeAgendaItem( item, subCellWidth );
|
|
}
|
|
} else {
|
|
for ( item=mItems.first(); item != 0; item=mItems.next() ) {
|
|
subCellWidth = calcSubCellWidth( item );
|
|
placeAgendaItem( item, subCellWidth );
|
|
}
|
|
}
|
|
}
|
|
checkScrollBoundaries();
|
|
marcus_bains();
|
|
}
|
|
|
|
void KOAgenda::scrollUp()
|
|
{
|
|
scrollBy(0,-mScrollOffset);
|
|
}
|
|
|
|
|
|
void KOAgenda::scrollDown()
|
|
{
|
|
scrollBy(0,mScrollOffset);
|
|
}
|
|
|
|
|
|
/*
|
|
Calculates the minimum width
|
|
*/
|
|
int KOAgenda::minimumWidth() const
|
|
{
|
|
// FIXME:: develop a way to dynamically determine the minimum width
|
|
int min = 100;
|
|
|
|
return min;
|
|
}
|
|
|
|
void KOAgenda::updateConfig()
|
|
{
|
|
double oldGridSpacingY = mGridSpacingY;
|
|
|
|
mDesiredGridSpacingY = KOPrefs::instance()->mHourSize;
|
|
if ( mDesiredGridSpacingY < 4 || mDesiredGridSpacingY > 30 ) {
|
|
mDesiredGridSpacingY = 10;
|
|
}
|
|
|
|
// make sure that there are not more than 24 per day
|
|
mGridSpacingY = (double)height() / (double)mRows;
|
|
if ( mGridSpacingY < mDesiredGridSpacingY ) {
|
|
mGridSpacingY = mDesiredGridSpacingY;
|
|
}
|
|
|
|
//can be two doubles equal?, it's better to compare them with an epsilon
|
|
if ( fabs( oldGridSpacingY - mGridSpacingY ) > 0.1 ) {
|
|
resizeContents( int( mGridSpacingX * mColumns ),
|
|
int( mGridSpacingY * mRows ) );
|
|
}
|
|
|
|
calculateWorkingHours();
|
|
|
|
marcus_bains();
|
|
}
|
|
|
|
void KOAgenda::checkScrollBoundaries()
|
|
{
|
|
// Invalidate old values to force update
|
|
mOldLowerScrollValue = -1;
|
|
mOldUpperScrollValue = -1;
|
|
|
|
checkScrollBoundaries(verticalScrollBar()->value());
|
|
}
|
|
|
|
void KOAgenda::checkScrollBoundaries( int v )
|
|
{
|
|
int yMin = int( (v) / mGridSpacingY );
|
|
int yMax = int( ( v + visibleHeight() ) / mGridSpacingY );
|
|
|
|
// kdDebug(5850) << "--- yMin: " << yMin << " yMax: " << yMax << endl;
|
|
|
|
if ( yMin != mOldLowerScrollValue ) {
|
|
mOldLowerScrollValue = yMin;
|
|
emit lowerYChanged(yMin);
|
|
}
|
|
if ( yMax != mOldUpperScrollValue ) {
|
|
mOldUpperScrollValue = yMax;
|
|
emit upperYChanged(yMax);
|
|
}
|
|
}
|
|
|
|
int KOAgenda::visibleContentsYMin()
|
|
{
|
|
int v = verticalScrollBar()->value();
|
|
return int( v / mGridSpacingY );
|
|
}
|
|
|
|
int KOAgenda::visibleContentsYMax()
|
|
{
|
|
int v = verticalScrollBar()->value();
|
|
return int( ( v + visibleHeight() ) / mGridSpacingY );
|
|
}
|
|
|
|
void KOAgenda::deselectItem()
|
|
{
|
|
if ( mSelectedItem.isNull() ) {
|
|
return;
|
|
}
|
|
mSelectedItem->select(false);
|
|
mSelectedItem = 0;
|
|
}
|
|
|
|
void KOAgenda::selectItem(KOAgendaItem *item)
|
|
{
|
|
if ((KOAgendaItem *)mSelectedItem == item) return;
|
|
deselectItem();
|
|
if (item == 0) {
|
|
emit incidenceSelected( 0, TQDate() );
|
|
return;
|
|
}
|
|
mSelectedItem = item;
|
|
mSelectedItem->select();
|
|
assert( mSelectedItem->incidence() );
|
|
mSelectedUid = mSelectedItem->incidence()->uid();
|
|
emit incidenceSelected( mSelectedItem->incidence(), mSelectedItem->itemDate() );
|
|
}
|
|
|
|
void KOAgenda::selectItemByUID( const TQString& uid )
|
|
{
|
|
KOAgendaItem *item;
|
|
for ( item = mItems.first(); item != 0; item = mItems.next() ) {
|
|
if( item->incidence() && item->incidence()->uid() == uid ) {
|
|
selectItem( item );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// This function seems never be called.
|
|
void KOAgenda::keyPressEvent( TQKeyEvent *kev )
|
|
{
|
|
switch(kev->key()) {
|
|
case Key_PageDown:
|
|
verticalScrollBar()->addPage();
|
|
break;
|
|
case Key_PageUp:
|
|
verticalScrollBar()->subtractPage();
|
|
break;
|
|
case Key_Down:
|
|
verticalScrollBar()->addLine();
|
|
break;
|
|
case Key_Up:
|
|
verticalScrollBar()->subtractLine();
|
|
break;
|
|
default:
|
|
;
|
|
}
|
|
}
|
|
|
|
void KOAgenda::calculateWorkingHours()
|
|
{
|
|
mWorkingHoursEnable = !mAllDayMode;
|
|
|
|
TQTime tmp = KOPrefs::instance()->mWorkingHoursStart.time();
|
|
mWorkingHoursYTop = int( 4 * mGridSpacingY *
|
|
( tmp.hour() + tmp.minute() / 60. +
|
|
tmp.second() / 3600. ) );
|
|
tmp = KOPrefs::instance()->mWorkingHoursEnd.time();
|
|
mWorkingHoursYBottom = int( 4 * mGridSpacingY *
|
|
( tmp.hour() + tmp.minute() / 60. +
|
|
tmp.second() / 3600. ) - 1 );
|
|
}
|
|
|
|
|
|
DateList KOAgenda::dateList() const
|
|
{
|
|
return mSelectedDates;
|
|
}
|
|
|
|
void KOAgenda::setDateList(const DateList &selectedDates)
|
|
{
|
|
mSelectedDates = selectedDates;
|
|
marcus_bains();
|
|
}
|
|
|
|
void KOAgenda::setHolidayMask(TQMemArray<bool> *mask)
|
|
{
|
|
mHolidayMask = mask;
|
|
|
|
}
|
|
|
|
void KOAgenda::contentsMousePressEvent ( TQMouseEvent *event )
|
|
{
|
|
kdDebug(5850) << "KOagenda::contentsMousePressEvent(): type: " << event->type() << endl;
|
|
TQScrollView::contentsMousePressEvent(event);
|
|
}
|
|
|
|
void KOAgenda::setTypeAheadReceiver( TQObject *o )
|
|
{
|
|
mTypeAheadReceiver = o;
|
|
}
|
|
|
|
TQObject *KOAgenda::typeAheadReceiver() const
|
|
{
|
|
return mTypeAheadReceiver;
|
|
}
|