/********************************************************************** ** Copyright (C) 2002 Trolltech AS. All rights reserved. ** ** This file is part of TQt Designer. ** ** This file may be distributed and/or modified under the terms of the ** GNU General Public License version 2 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. ** ** Licensees holding valid TQt Enterprise Edition or TQt Professional Edition ** licenses may use this file in accordance with the TQt Commercial License ** Agreement provided with the Software. ** ** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE ** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. ** ** See http://www.trolltech.com/gpl/ for GPL licensing information. ** See http://www.trolltech.com/pricing.html or email sales@trolltech.com for ** information about TQt Commercial License Agreements. ** ** Contact info@trolltech.com if any conditions of this licensing are ** not clear to you. ** **********************************************************************/ #include "listboxdnd.h" #include #include #include #include #include // The Dragobject Declaration --------------------------------------- class ListBoxItemDrag : public TQStoredDrag { public: ListBoxItemDrag( ListBoxItemList & items, bool sendPtr = FALSE, TQListBox * parent = 0, const char * name = 0 ); ~ListBoxItemDrag() {}; static bool canDecode( TQDragMoveEvent * event ); static bool decode( TQDropEvent * event, TQListBox * parent, TQListBoxItem * insertPoint ); enum ItemType { ListBoxText = 1, ListBoxPixmap = 2 }; }; // ------------------------------------------------------------------ ListBoxDnd::ListBoxDnd( TQListBox * eventSource, const char * name ) : ListDnd( eventSource, name ) { } void ListBoxDnd::confirmDrop( TQListBoxItem * ) { dropConfirmed = TRUE; } bool ListBoxDnd::dropEvent( TQDropEvent * event ) { if ( dragInside ) { if ( dMode & NullDrop ) { // combined with Move, a NullDrop will delete an item event->accept(); emit dropped( 0 ); // a NullDrop return TRUE; } TQPoint pos = event->pos(); TQListBoxItem * after = itemAt( pos ); if ( ListBoxItemDrag::decode( event, (TQListBox *) src, after ) ) { event->accept(); TQListBox * src = (TQListBox *) this->src; TQListBoxItem * item = ( after ? after->next() : src->firstItem() ); src->setCurrentItem( item ); emit dropped( item ); // ###FIX: Supports only one item! } } line->hide(); dragInside = FALSE; return TRUE; } bool ListBoxDnd::mouseMoveEvent( TQMouseEvent * event ) { if ( event->state() & Qt::LeftButton ) { if ( ( event->pos() - mousePressPos ).manhattanLength() > 3 ) { ListBoxItemList list; buildList( list ); ListBoxItemDrag * dragobject = new ListBoxItemDrag( list, (dMode & Internal), (TQListBox *) src ); // Emit signal for all dragged items TQListBoxItem * i = list.first(); while ( i ) { emit dragged( i ); i = list.next(); } if ( dMode & Move ) { removeList( list ); // "hide" items } dragobject->dragCopy(); if ( dMode & Move ) { if ( dropConfirmed ) { // ###FIX: memleak ? in internal mode, only pointers are transfered... //list.setAutoDelete( TRUE ); list.clear(); dropConfirmed = FALSE; } insertList( list ); // "show" items } } } return FALSE; } int ListBoxDnd::buildList( ListBoxItemList & list ) { TQListBoxItem * i = ((TQListBox *)src)->firstItem(); while ( i ) { if ( i->isSelected() ) { ((TQListBox *)src)->setSelected( i, FALSE ); list.append( i ); } i = i->next(); } return list.count(); } void ListBoxDnd::insertList( ListBoxItemList & list ) { TQListBoxItem * i = list.first(); while ( i ) { ((TQListBox *)src)->insertItem( i, i->prev() ); i = list.next(); } } void ListBoxDnd::removeList( ListBoxItemList & list ) { TQListBoxItem * i = list.first(); while ( i ) { ((TQListBox *)src)->takeItem( i ); // remove item from TQListBox i = list.next(); } } void ListBoxDnd::updateLine( const TQPoint & dragPos ) { TQListBox * src = (TQListBox *) this->src; TQListBoxItem *item = itemAt( dragPos ); int ypos = item ? ( src->itemRect( item ).bottom() - ( line->height() / 2 ) ) : ( src->itemRect( ((TQListBox *)src)->firstItem() ).top() ); line->resize( src->viewport()->width(), line->height() ); line->move( 0, ypos ); } TQListBoxItem * ListBoxDnd::itemAt( TQPoint pos ) { TQListBox * src = (TQListBox *) this->src; TQListBoxItem * result = src->itemAt( pos ); TQListBoxItem * last = src->item( src->count() - 1 ); int i = src->index( result ); if ( result && ( pos.y() < (src->itemRect(result).top() + src->itemHeight(i)/2) ) ) result = result->prev(); else if ( !result && pos.y() > src->itemRect( last ).bottom() ) result = last; return result; } bool ListBoxDnd::canDecode( TQDragEnterEvent * event ) { return ListBoxItemDrag::canDecode( event ); } // ------------------------------------------------------------------ // The Dragobject Implementation ------------------------------------ // ------------------------------------------------------------------ ListBoxItemDrag::ListBoxItemDrag( ListBoxItemList & items, bool sendPtr, TQListBox * parent, const char * name ) : TQStoredDrag( "qt/listboxitem", parent, name ) { // ### FIX! TQByteArray data( sizeof( TQ_INT32 ) + sizeof( TQListBoxItem ) * items.count() ); TQDataStream stream( data, IO_WriteOnly ); stream << items.count(); stream << (TQ_UINT8) sendPtr; // just transfer item pointer; omit data TQListBoxItem * i = items.first(); if ( sendPtr ) { while ( i ) { stream << (TQ_ULONG) i; //###FIX: demands sizeof(ulong) >= sizeof(void*) i = items.next(); } } else { while ( i ) { TQ_UINT8 b = 0; b = (TQ_UINT8) ( i->text() != TQString() ); // does item have text ? stream << b; if ( b ) { stream << i->text(); } b = (TQ_UINT8) ( !!i->pixmap() ); // does item have a pixmap ? stream << b; if ( b ) { stream << ( *i->pixmap() ); } stream << (TQ_UINT8) i->isSelectable(); i = items.next(); } } setEncodedData( data ); } bool ListBoxItemDrag::canDecode( TQDragMoveEvent * event ) { return event->provides( "qt/listboxitem" ); } bool ListBoxItemDrag::decode( TQDropEvent * event, TQListBox * parent, TQListBoxItem * after ) { TQByteArray data = event->encodedData( "qt/listboxitem" ); if ( data.size() ) { event->accept(); TQDataStream stream( data, IO_ReadOnly ); int count = 0; stream >> count; TQ_UINT8 recievePtr = 0; // data contains just item pointers; no data stream >> recievePtr; TQListBoxItem * item = 0; if ( recievePtr ) { for( int i = 0; i < count; i++ ) { TQ_ULONG p = 0; //###FIX: demands sizeof(ulong) >= sizeof(void*) stream >> p; item = (TQListBoxItem *) p; parent->insertItem( item, after ); } } else { for ( int i = 0; i < count; i++ ) { TQ_UINT8 hasText = 0; TQString text; stream >> hasText; if ( hasText ) { stream >> text; } TQ_UINT8 hasPixmap = 0; TQPixmap pixmap; stream >> hasPixmap; if ( hasPixmap ) { stream >> pixmap; } TQ_UINT8 isSelectable = 0; stream >> isSelectable; if ( hasPixmap ) { item = new TQListBoxPixmap( parent, pixmap, text, after ); } else { item = new TQListBoxText( parent, text, after ); } item->setSelectable( isSelectable ); } } return TRUE; } return FALSE; }