/********************************************************************** ** Copyright (C) 2002 Trolltech AS. All rights reserved. ** ** This file is part of Qt 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 Qt Enterprise Edition or Qt Professional Edition ** licenses may use this file in accordance with the Qt 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 Qt Commercial License Agreements. ** ** Contact info@trolltech.com if any conditions of this licensing are ** not clear to you. ** **********************************************************************/ #include "connectionitems.h" #include "formwindow.h" #include "mainwindow.h" #include "metadatabase.h" #include "widgetfactory.h" #include "project.h" #include #include #include #include #include #include #include static const char* const ignore_slots[] = { "destroyed()", "setCaption(const QString&)", "setIcon(const QPixmap&)", "setIconText(const QString&)", "setMouseTracking(bool)", "clearFocus()", "setUpdatesEnabled(bool)", "update()", "update(int,int,int,int)", "update(const QRect&)", "repaint()", "repaint(bool)", "repaint(int,int,int,int,bool)", "repaint(const QRect&,bool)", "repaint(const QRegion&,bool)", "show()", "hide()", "iconify()", "showMinimized()", "showMaximized()", "showFullScreen()", "showNormal()", "polish()", "constPolish()", "raise()", "lower()", "stackUnder(QWidget*)", "move(int,int)", "move(const QPoint&)", "resize(int,int)", "resize(const QSize&)", "setGeometry(int,int,int,int)", "setGeometry(const QRect&)", "focusProxyDestroyed()", "showExtension(bool)", "setUpLayout()", "showDockMenu(const QPoint&)", "init()", "destroy()", "deleteLater()", 0 }; ConnectionItem::ConnectionItem( QTable *table, FormWindow *fw ) : QComboTableItem( table, QStringList(), FALSE ), formWindow( fw ), conn( 0 ) { setReplaceable( FALSE ); } void ConnectionItem::senderChanged( QObject * ) { emit changed(); QWidget *w = table()->cellWidget( row(), col() ); if ( w ) setContentFromEditor( w ); } void ConnectionItem::receiverChanged( QObject * ) { emit changed(); QWidget *w = table()->cellWidget( row(), col() ); if ( w ) setContentFromEditor( w ); } void ConnectionItem::signalChanged( const QString & ) { emit changed(); QWidget *w = table()->cellWidget( row(), col() ); if ( w ) setContentFromEditor( w ); } void ConnectionItem::slotChanged( const QString & ) { emit changed(); QWidget *w = table()->cellWidget( row(), col() ); if ( w ) setContentFromEditor( w ); } void ConnectionItem::setSender( SenderItem *i ) { connect( i, SIGNAL( currentSenderChanged( QObject * ) ), this, SLOT( senderChanged( QObject * ) ) ); } void ConnectionItem::setReceiver( ReceiverItem *i ) { connect( i, SIGNAL( currentReceiverChanged( QObject * ) ), this, SLOT( receiverChanged( QObject * ) ) ); } void ConnectionItem::setSignal( SignalItem *i ) { connect( i, SIGNAL( currentSignalChanged( const QString & ) ), this, SLOT( signalChanged( const QString & ) ) ); } void ConnectionItem::setSlot( SlotItem *i ) { connect( i, SIGNAL( currentSlotChanged( const QString & ) ), this, SLOT( slotChanged( const QString & ) ) ); } void ConnectionItem::paint( QPainter *p, const QColorGroup &cg, const QRect &cr, bool selected ) { p->fillRect( 0, 0, cr.width(), cr.height(), selected ? cg.brush( QColorGroup::Highlight ) : cg.brush( QColorGroup::Base ) ); int w = cr.width(); int h = cr.height(); int x = 0; if ( currentText()[0] == '<' ) p->setPen( QObject::red ); else if ( selected ) p->setPen( cg.highlightedText() ); else p->setPen( cg.text() ); QFont f( p->font() ); QFont oldf( p->font() ); if ( conn && conn->isModified() ) { f.setBold( TRUE ); p->setFont( f ); } p->drawText( x + 2, 0, w - x - 4, h, alignment(), currentText() ); p->setFont( oldf ); } void ConnectionItem::setConnection( ConnectionContainer *c ) { conn = c; } // ------------------------------------------------------------------ static void appendChildActions( QAction *action, QStringList &lst ) { QObjectListIt it( *action->children() ); while ( it.current() ) { QObject *o = it.current(); ++it; if ( !::qt_cast(o) ) continue; lst << o->name(); if ( o->children() && ::qt_cast(o) ) appendChildActions( (QAction*)o, lst ); } } static QStringList flatActions( const QPtrList &l ) { QStringList lst; QPtrListIterator it( l ); while ( it.current() ) { QAction *action = it.current(); lst << action->name(); if ( action->children() && ::qt_cast(action) ) appendChildActions( action, lst ); ++it; } return lst; } // ------------------------------------------------------------------ SenderItem::SenderItem( QTable *table, FormWindow *fw ) : ConnectionItem( table, fw ) { QStringList lst; QPtrDictIterator it( *formWindow->widgets() ); while ( it.current() ) { if ( lst.find( it.current()->name() ) != lst.end() ) { ++it; continue; } if ( !QString( it.current()->name() ).startsWith( "qt_dead_widget_" ) && !::qt_cast(it.current()) && !::qt_cast(it.current()) && !::qt_cast(it.current()) && qstrcmp( it.current()->name(), "central widget" ) != 0 ) { lst << it.current()->name(); } ++it; } lst += flatActions( formWindow->actionList() ); lst.prepend( "" ); lst.sort(); setStringList( lst ); } QWidget *SenderItem::createEditor() const { QComboBox *cb = (QComboBox*)ConnectionItem::createEditor(); cb->listBox()->setMinimumWidth( cb->fontMetrics().width( "01234567890123456789012345678901234567890123456789" ) ); connect( cb, SIGNAL( activated( const QString & ) ), this, SLOT( senderChanged( const QString & ) ) ); return cb; } void SenderItem::setSenderEx( QObject *sender ) { setCurrentItem( sender->name() ); emit currentSenderChanged( sender ); } void SenderItem::senderChanged( const QString &sender ) { QObject *o = formWindow->child( sender, "QObject" ); if ( !o ) o = formWindow->findAction( sender ); if ( !o ) return; emit currentSenderChanged( o ); } // ------------------------------------------------------------------ ReceiverItem::ReceiverItem( QTable *table, FormWindow *fw ) : ConnectionItem( table, fw ) { QStringList lst; QPtrDictIterator it( *formWindow->widgets() ); while ( it.current() ) { if ( lst.find( it.current()->name() ) != lst.end() ) { ++it; continue; } if ( !QString( it.current()->name() ).startsWith( "qt_dead_widget_" ) && !::qt_cast(it.current()) && !::qt_cast(it.current()) && !::qt_cast(it.current()) && qstrcmp( it.current()->name(), "central widget" ) != 0 ) { lst << it.current()->name(); } ++it; } lst += flatActions( formWindow->actionList() ); lst.prepend( "" ); lst.sort(); setStringList( lst ); } QWidget *ReceiverItem::createEditor() const { QComboBox *cb = (QComboBox*)ConnectionItem::createEditor(); cb->listBox()->setMinimumWidth( cb->fontMetrics().width( "01234567890123456789012345678901234567890123456789" ) ); connect( cb, SIGNAL( activated( const QString & ) ), this, SLOT( receiverChanged( const QString & ) ) ); return cb; } void ReceiverItem::setReceiverEx( QObject *receiver ) { setCurrentItem( receiver->name() ); emit currentReceiverChanged( receiver ); } void ReceiverItem::receiverChanged( const QString &receiver ) { QObject *o = formWindow->child( receiver, "QObject" ); if ( !o ) o = formWindow->findAction( receiver ); if ( !o ) return; emit currentReceiverChanged( o ); } // ------------------------------------------------------------------ SignalItem::SignalItem( QTable *table, FormWindow *fw ) : ConnectionItem( table, fw ) { QStringList lst; lst << ""; lst.sort(); setStringList( lst ); } void SignalItem::senderChanged( QObject *sender ) { QStrList sigs = sender->metaObject()->signalNames( TRUE ); sigs.remove( "destroyed()" ); sigs.remove( "destroyed(QObject*)" ); sigs.remove( "accessibilityChanged(int)" ); sigs.remove( "accessibilityChanged(int,int)" ); QStringList lst = QStringList::fromStrList( sigs ); if ( ::qt_cast(sender) ) { MetaDataBase::CustomWidget *w = ( (CustomWidget*)sender )->customWidget(); for ( QValueList::Iterator it = w->lstSignals.begin(); it != w->lstSignals.end(); ++it ) lst << MetaDataBase::normalizeFunction( *it ); } if ( sender == formWindow->mainContainer() ) { QStringList extra = MetaDataBase::signalList( formWindow ); if ( !extra.isEmpty() ) lst += extra; } lst.prepend( "" ); lst.sort(); setStringList( lst ); ConnectionItem::senderChanged( sender ); } QWidget *SignalItem::createEditor() const { QComboBox *cb = (QComboBox*)ConnectionItem::createEditor(); cb->listBox()->setMinimumWidth( cb->fontMetrics().width( "01234567890123456789012345678901234567890123456789" ) ); connect( cb, SIGNAL( activated( const QString & ) ), this, SIGNAL( currentSignalChanged( const QString & ) ) ); return cb; } // ------------------------------------------------------------------ SlotItem::SlotItem( QTable *table, FormWindow *fw ) : ConnectionItem( table, fw ) { QStringList lst; lst << ""; lst.sort(); setStringList( lst ); lastReceiver = 0; lastSignal = ""; } void SlotItem::receiverChanged( QObject *receiver ) { lastReceiver = receiver; updateSlotList(); ConnectionItem::receiverChanged( receiver ); } void SlotItem::signalChanged( const QString &signal ) { lastSignal = signal; updateSlotList(); ConnectionItem::signalChanged( signal ); } bool SlotItem::ignoreSlot( const char* slot ) const { #ifndef QT_NO_SQL if ( qstrcmp( slot, "update()" ) == 0 && ::qt_cast(lastReceiver) ) return FALSE; #endif for ( int i = 0; ignore_slots[i]; i++ ) { if ( qstrcmp( slot, ignore_slots[i] ) == 0 ) return TRUE; } if ( !formWindow->isMainContainer( (QWidget*)lastReceiver ) ) { if ( qstrcmp( slot, "close()" ) == 0 ) return TRUE; } if ( qstrcmp( slot, "setFocus()" ) == 0 ) if ( lastReceiver->isWidgetType() && ( (QWidget*)lastReceiver )->focusPolicy() == QWidget::NoFocus ) return TRUE; return FALSE; } void SlotItem::updateSlotList() { QStringList lst; if ( !lastReceiver || lastSignal == "" ) { lst << ""; lst.sort(); setStringList( lst ); return; } QString signal = MetaDataBase::normalizeFunction( lastSignal ); int n = lastReceiver->metaObject()->numSlots( TRUE ); QStringList slts; for( int i = 0; i < n; ++i ) { // accept only public slots. For the form window, also accept protected slots const QMetaData* md = lastReceiver->metaObject()->slot( i, TRUE ); if ( ( (lastReceiver->metaObject()->slot( i, TRUE )->access == QMetaData::Public) || (formWindow->isMainContainer( (QWidget*)lastReceiver ) && lastReceiver->metaObject()->slot(i, TRUE)->access == QMetaData::Protected) ) && !ignoreSlot( md->name ) && checkConnectArgs( signal.latin1(), lastReceiver, md->name ) ) if ( lst.find( md->name ) == lst.end() ) lst << MetaDataBase::normalizeFunction( md->name ); } LanguageInterface *iface = MetaDataBase::languageInterface( formWindow->project()->language() ); if ( !iface || iface->supports( LanguageInterface::ConnectionsToCustomSlots ) ) { if ( formWindow->isMainContainer( (QWidget*)lastReceiver ) ) { QValueList moreSlots = MetaDataBase::slotList( formWindow ); if ( !moreSlots.isEmpty() ) { for ( QValueList::Iterator it = moreSlots.begin(); it != moreSlots.end(); ++it ) { QCString s = (*it).function; if ( !s.data() ) continue; s = MetaDataBase::normalizeFunction( s ); if ( checkConnectArgs( signal.latin1(), lastReceiver, s ) ) { if ( lst.find( (*it).function ) == lst.end() ) lst << s; } } } } } if ( ::qt_cast(lastReceiver) ) { MetaDataBase::CustomWidget *w = ( (CustomWidget*)lastReceiver )->customWidget(); for ( QValueList::Iterator it = w->lstSlots.begin(); it != w->lstSlots.end(); ++it ) { QCString s = (*it).function; if ( !s.data() ) continue; s = MetaDataBase::normalizeFunction( s ); if ( checkConnectArgs( signal.latin1(), lastReceiver, s ) ) { if ( lst.find( (*it).function ) == lst.end() ) lst << s; } } } lst.prepend( "" ); lst.sort(); setStringList( lst ); } QWidget *SlotItem::createEditor() const { QComboBox *cb = (QComboBox*)ConnectionItem::createEditor(); cb->listBox()->setMinimumWidth( cb->fontMetrics().width( "01234567890123456789012345678901234567890123456789" ) ); connect( cb, SIGNAL( activated( const QString & ) ), this, SIGNAL( currentSlotChanged( const QString & ) ) ); return cb; } void SlotItem::customSlotsChanged() { QString currSlot = currentText(); updateSlotList(); setCurrentItem( "" ); setCurrentItem( currSlot ); emit currentSlotChanged( currentText() ); }