/* This file is part of the KDE libraries Copyright (C) 2000 Reginald Stadlbauer (reggie@kde.org) (C) 1997, 1998 Stephan Kulow (coolo@kde.org) (C) 1997, 1998 Mark Donohoe (donohoe@kde.org) (C) 1997, 1998 Sven Radej (radej@kde.org) (C) 1997, 1998 Matthias Ettrich (ettrich@kde.org) (C) 1999 Chris Schlaeger (cs@kde.org) (C) 1999 Kurt Granroth (granroth@kde.org) This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License version 2 as published by the Free Software Foundation. This library 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 Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include <config.h> #ifdef KDE_USE_FINAL #undef Always #include <tqdockwindow.h> #endif #include <string.h> #include <tqpainter.h> #include <tqtooltip.h> #include <tqdrawutil.h> #include <tqstring.h> #include <tqrect.h> #include <tqobjectlist.h> #include <tqtimer.h> #include <tqstyle.h> #include <tqlayout.h> #include <ktoolbar.h> #include <kmainwindow.h> #include <klineedit.h> #include <kseparator.h> #include <klocale.h> #include <kapplication.h> #include <kaction.h> #include <kstdaction.h> #include <kglobal.h> #include <kconfig.h> #include <kiconloader.h> #include <kcombobox.h> #include <kpopupmenu.h> #include <kanimwidget.h> #include <kedittoolbar.h> #include <kipc.h> #include <kwin.h> #include <kdebug.h> #include <ktoolbarbutton.h> enum { CONTEXT_TOP = 0, CONTEXT_LEFT = 1, CONTEXT_RIGHT = 2, CONTEXT_BOTTOM = 3, CONTEXT_FLOAT = 4, CONTEXT_FLAT = 5, CONTEXT_ICONS = 6, CONTEXT_TEXT = 7, CONTEXT_TEXTRIGHT = 8, CONTEXT_TEXTUNDER = 9, CONTEXT_ICONSIZES = 50 // starting point for the icon size list, put everything else before }; class KToolBarPrivate { public: KToolBarPrivate() { m_iconSize = 0; m_iconText = KToolBar::IconOnly; m_highlight = true; m_transparent = true; m_honorStyle = false; m_enableContext = true; m_xmlguiClient = 0; oldPos = Qt::DockUnmanaged; modified = m_isHorizontal = positioned = false; IconSizeDefault = 0; IconTextDefault = "IconOnly"; NewLineDefault = false; OffsetDefault = 0; PositionDefault = "Top"; HiddenDefault = false; idleButtons.setAutoDelete(true); } int m_iconSize; KToolBar::IconText m_iconText; bool m_highlight : 1; bool m_transparent : 1; bool m_honorStyle : 1; bool m_isHorizontal : 1; bool m_enableContext : 1; bool modified : 1; bool positioned : 1; TQWidget *m_parent; TQMainWindow::ToolBarDock oldPos; KXMLGUIClient *m_xmlguiClient; struct ToolBarInfo { ToolBarInfo() : index( -1 ), offset( -1 ), newline( false ), dock( Qt::DockTop ) {} ToolBarInfo( Qt::Dock d, int i, bool n, int o ) : index( i ), offset( o ), newline( n ), dock( d ) {} int index, offset; bool newline; Qt::Dock dock; }; ToolBarInfo toolBarInfo; TQValueList<int> iconSizes; TQTimer repaintTimer; // Default Values. bool HiddenDefault; int IconSizeDefault; TQString IconTextDefault; bool NewLineDefault; int OffsetDefault; TQString PositionDefault; TQPtrList<TQWidget> idleButtons; }; KToolBarSeparator::KToolBarSeparator(Orientation o , bool l, TQToolBar *parent, const char* name ) :TQFrame( parent, name ), line( l ) { connect( parent, TQT_SIGNAL(orientationChanged(Orientation)), this, TQT_SLOT(setOrientation(Orientation)) ); setOrientation( o ); setBackgroundMode( parent->backgroundMode() ); setBackgroundOrigin( ParentOrigin ); } void KToolBarSeparator::setOrientation( Orientation o ) { orient = o; setFrameStyle( NoFrame ); } void KToolBarSeparator::drawContents( TQPainter* p ) { if ( line ) { TQStyle::SFlags flags = TQStyle::Style_Default; if ( orientation() == Horizontal ) flags = flags | TQStyle::Style_Horizontal; style().drawPrimitive(TQStyle::PE_DockWindowSeparator, p, contentsRect(), colorGroup(), flags); } else { TQFrame::drawContents(p); } } void KToolBarSeparator::styleChange( TQStyle& ) { setOrientation( orient ); } TQSize KToolBarSeparator::sizeHint() const { int dim = style().pixelMetric( TQStyle::PM_DockWindowSeparatorExtent, this ); return orientation() == Vertical ? TQSize( 0, dim ) : TQSize( dim, 0 ); } TQSizePolicy KToolBarSeparator::sizePolicy() const { return TQSizePolicy( TQSizePolicy::Minimum, TQSizePolicy::Minimum ); } KToolBar::KToolBar( TQWidget *parent, const char *name, bool honorStyle, bool readConfig ) : TQToolBar( TQString::fromLatin1( name ), dynamic_cast<TQMainWindow*>(parent), parent, false, name ? name : "mainToolBar") { init( readConfig, honorStyle ); } KToolBar::KToolBar( TQMainWindow *parentWindow, TQMainWindow::ToolBarDock dock, bool newLine, const char *name, bool honorStyle, bool readConfig ) : TQToolBar( TQString::fromLatin1( name ), parentWindow, dock, newLine, name ? name : "mainToolBar") { init( readConfig, honorStyle ); } KToolBar::KToolBar( TQMainWindow *parentWindow, TQWidget *dock, bool newLine, const char *name, bool honorStyle, bool readConfig ) : TQToolBar( TQString::fromLatin1( name ), parentWindow, dock, newLine, name ? name : "mainToolBar") { init( readConfig, honorStyle ); } KToolBar::~KToolBar() { emit toolbarDestroyed(); delete d; } void KToolBar::init( bool readConfig, bool honorStyle ) { d = new KToolBarPrivate; setFullSize( true ); d->m_honorStyle = honorStyle; context = 0; layoutTimer = new TQTimer( this ); connect( layoutTimer, TQT_SIGNAL( timeout() ), this, TQT_SLOT( rebuildLayout() ) ); connect( &(d->repaintTimer), TQT_SIGNAL( timeout() ), this, TQT_SLOT( slotRepaint() ) ); if ( kapp ) { // may be null when started inside designer connect(kapp, TQT_SIGNAL(toolbarAppearanceChanged(int)), this, TQT_SLOT(slotAppearanceChanged())); // request notification of changes in icon style kapp->addKipcEventMask(KIPC::IconChanged); connect(kapp, TQT_SIGNAL(iconChanged(int)), this, TQT_SLOT(slotIconChanged(int))); } // finally, read in our configurable settings if ( readConfig ) slotReadConfig(); if ( mainWindow() ) connect( mainWindow(), TQT_SIGNAL( toolBarPositionChanged( TQToolBar * ) ), this, TQT_SLOT( toolBarPosChanged( TQToolBar * ) ) ); // Hack to make sure we recalculate our size when we dock. connect( this, TQT_SIGNAL(placeChanged(TQDockWindow::Place)), TQT_SLOT(rebuildLayout()) ); } int KToolBar::insertButton(const TQString& icon, int id, bool enabled, const TQString& text, int index, KInstance *_instance ) { KToolBarButton *button = new KToolBarButton( icon, id, this, 0, text, _instance ); insertWidgetInternal( button, index, id ); button->setEnabled( enabled ); doConnections( button ); return index; } int KToolBar::insertButton(const TQString& icon, int id, const char *signal, const TQObject *receiver, const char *slot, bool enabled, const TQString& text, int index, KInstance *_instance ) { KToolBarButton *button = new KToolBarButton( icon, id, this, 0, text, _instance); insertWidgetInternal( button, index, id ); button->setEnabled( enabled ); connect( button, signal, receiver, slot ); doConnections( button ); return index; } int KToolBar::insertButton(const TQPixmap& pixmap, int id, bool enabled, const TQString& text, int index ) { KToolBarButton *button = new KToolBarButton( pixmap, id, this, 0, text); insertWidgetInternal( button, index, id ); button->setEnabled( enabled ); doConnections( button ); return index; } int KToolBar::insertButton(const TQPixmap& pixmap, int id, const char *signal, const TQObject *receiver, const char *slot, bool enabled, const TQString& text, int index ) { KToolBarButton *button = new KToolBarButton( pixmap, id, this, 0, text); insertWidgetInternal( button, index, id ); button->setEnabled( enabled ); connect( button, signal, receiver, slot ); doConnections( button ); return index; } int KToolBar::insertButton(const TQString& icon, int id, TQPopupMenu *popup, bool enabled, const TQString &text, int index ) { KToolBarButton *button = new KToolBarButton( icon, id, this, 0, text ); insertWidgetInternal( button, index, id ); button->setEnabled( enabled ); button->setPopup( popup ); doConnections( button ); return index; } int KToolBar::insertButton(const TQPixmap& pixmap, int id, TQPopupMenu *popup, bool enabled, const TQString &text, int index ) { KToolBarButton *button = new KToolBarButton( pixmap, id, this, 0, text ); insertWidgetInternal( button, index, id ); button->setEnabled( enabled ); button->setPopup( popup ); doConnections( button ); return index; } int KToolBar::insertLined (const TQString& text, int id, const char *signal, const TQObject *receiver, const char *slot, bool enabled , const TQString& toolTipText, int size, int index ) { KLineEdit *lined = new KLineEdit ( this, 0 ); if ( !toolTipText.isEmpty() ) TQToolTip::add( lined, toolTipText ); if ( size > 0 ) lined->setMinimumWidth( size ); insertWidgetInternal( lined, index, id ); connect( lined, signal, receiver, slot ); lined->setText(text); lined->setEnabled( enabled ); return index; } int KToolBar::insertCombo (const TQStringList &list, int id, bool writable, const char *signal, const TQObject *receiver, const char *slot, bool enabled, const TQString& tooltiptext, int size, int index, TQComboBox::Policy policy ) { KComboBox *combo = new KComboBox ( writable, this ); insertWidgetInternal( combo, index, id ); combo->insertStringList (list); combo->setInsertionPolicy(policy); combo->setEnabled( enabled ); if ( size > 0 ) combo->setMinimumWidth( size ); if (!tooltiptext.isNull()) TQToolTip::add( combo, tooltiptext ); if ( signal && receiver && slot ) connect ( combo, signal, receiver, slot ); return index; } int KToolBar::insertCombo (const TQString& text, int id, bool writable, const char *signal, TQObject *receiver, const char *slot, bool enabled, const TQString& tooltiptext, int size, int index, TQComboBox::Policy policy ) { KComboBox *combo = new KComboBox ( writable, this ); insertWidgetInternal( combo, index, id ); combo->insertItem (text); combo->setInsertionPolicy(policy); combo->setEnabled( enabled ); if ( size > 0 ) combo->setMinimumWidth( size ); if (!tooltiptext.isNull()) TQToolTip::add( combo, tooltiptext ); connect (combo, signal, receiver, slot); return index; } int KToolBar::insertSeparator(int index, int id) { TQWidget *w = new KToolBarSeparator( orientation(), false, this, "tool bar separator" ); insertWidgetInternal( w, index, id ); return index; } int KToolBar::insertLineSeparator(int index, int id) { TQWidget *w = new KToolBarSeparator( orientation(), true, this, "tool bar separator" ); insertWidgetInternal( w, index, id ); return index; } int KToolBar::insertWidget(int id, int /*width*/, TQWidget *widget, int index) { removeWidgetInternal( widget ); // in case we already have it ? insertWidgetInternal( widget, index, id ); return index; } int KToolBar::insertAnimatedWidget(int id, TQObject *receiver, const char *slot, const TQString& icons, int index ) { KAnimWidget *anim = new KAnimWidget( icons, d->m_iconSize, this ); insertWidgetInternal( anim, index, id ); if ( receiver ) connect( anim, TQT_SIGNAL(clicked()), receiver, slot); return index; } KAnimWidget *KToolBar::animatedWidget( int id ) { Id2WidgetMap::Iterator it = id2widget.find( id ); if ( it == id2widget.end() ) return 0; KAnimWidget *aw = dynamic_cast<KAnimWidget *>(*it); if ( aw ) return aw; TQObjectList *l = queryList( "KAnimWidget" ); if ( !l || !l->first() ) { delete l; return 0; } for ( TQObject *o = l->first(); o; o = l->next() ) { KAnimWidget *aw = dynamic_cast<KAnimWidget *>(o); if ( aw ) { delete l; return aw; } } delete l; return 0; } void KToolBar::addConnection (int id, const char *signal, const TQObject *receiver, const char *slot) { TQWidget* w = getWidget( id ); if ( w ) connect( w, signal, receiver, slot ); } void KToolBar::setItemEnabled( int id, bool enabled ) { TQWidget* w = getWidget( id ); if ( w ) w->setEnabled( enabled ); } void KToolBar::setButtonPixmap( int id, const TQPixmap& _pixmap ) { KToolBarButton * button = getButton( id ); if ( button ) button->setPixmap( _pixmap ); } void KToolBar::setButtonIcon( int id, const TQString& _icon ) { KToolBarButton * button = getButton( id ); if ( button ) button->setIcon( _icon ); } void KToolBar::setButtonIconSet( int id, const TQIconSet& iconset ) { KToolBarButton * button = getButton( id ); if ( button ) button->setIconSet( iconset ); } void KToolBar::setDelayedPopup (int id , TQPopupMenu *_popup, bool toggle ) { KToolBarButton * button = getButton( id ); if ( button ) button->setDelayedPopup( _popup, toggle ); } void KToolBar::setAutoRepeat (int id, bool flag) { KToolBarButton * button = getButton( id ); if ( button ) button->setAutoRepeat( flag ); } void KToolBar::setToggle (int id, bool flag ) { KToolBarButton * button = getButton( id ); if ( button ) button->setToggle( flag ); } void KToolBar::toggleButton (int id) { KToolBarButton * button = getButton( id ); if ( button ) button->toggle(); } void KToolBar::setButton (int id, bool flag) { KToolBarButton * button = getButton( id ); if ( button ) button->on( flag ); } bool KToolBar::isButtonOn (int id) const { KToolBarButton * button = const_cast<KToolBar*>( this )->getButton( id ); return button ? button->isOn() : false; } void KToolBar::setLinedText (int id, const TQString& text) { KLineEdit * lineEdit = getLined( id ); if ( lineEdit ) lineEdit->setText( text ); } TQString KToolBar::getLinedText (int id) const { KLineEdit * lineEdit = const_cast<KToolBar*>( this )->getLined( id ); return lineEdit ? lineEdit->text() : TQString::null; } void KToolBar::insertComboItem (int id, const TQString& text, int index) { KComboBox * comboBox = getCombo( id ); if (comboBox) comboBox->insertItem( text, index ); } void KToolBar::insertComboList (int id, const TQStringList &list, int index) { KComboBox * comboBox = getCombo( id ); if (comboBox) comboBox->insertStringList( list, index ); } void KToolBar::removeComboItem (int id, int index) { KComboBox * comboBox = getCombo( id ); if (comboBox) comboBox->removeItem( index ); } void KToolBar::setCurrentComboItem (int id, int index) { KComboBox * comboBox = getCombo( id ); if (comboBox) comboBox->setCurrentItem( index ); } void KToolBar::changeComboItem (int id, const TQString& text, int index) { KComboBox * comboBox = getCombo( id ); if (comboBox) comboBox->changeItem( text, index ); } void KToolBar::clearCombo (int id) { KComboBox * comboBox = getCombo( id ); if (comboBox) comboBox->clear(); } TQString KToolBar::getComboItem (int id, int index) const { KComboBox * comboBox = const_cast<KToolBar*>( this )->getCombo( id ); return comboBox ? comboBox->text( index ) : TQString::null; } KComboBox * KToolBar::getCombo(int id) { Id2WidgetMap::Iterator it = id2widget.find( id ); if ( it == id2widget.end() ) return 0; return dynamic_cast<KComboBox *>( *it ); } KLineEdit * KToolBar::getLined (int id) { Id2WidgetMap::Iterator it = id2widget.find( id ); if ( it == id2widget.end() ) return 0; return dynamic_cast<KLineEdit *>( *it ); } KToolBarButton * KToolBar::getButton (int id) { Id2WidgetMap::Iterator it = id2widget.find( id ); if ( it == id2widget.end() ) return 0; return dynamic_cast<KToolBarButton *>( *it ); } void KToolBar::alignItemRight (int id, bool right ) { Id2WidgetMap::Iterator it = id2widget.find( id ); if ( it == id2widget.end() ) return; if ( rightAligned && !right && (*it) == rightAligned ) rightAligned = 0; if ( (*it) && right ) rightAligned = (*it); } TQWidget *KToolBar::getWidget (int id) { Id2WidgetMap::Iterator it = id2widget.find( id ); return ( it == id2widget.end() ) ? 0 : (*it); } void KToolBar::setItemAutoSized (int id, bool yes ) { TQWidget *w = getWidget(id); if ( w && yes ) setStretchableWidget( w ); } void KToolBar::clear () { /* Delete any idle buttons, so TQToolBar doesn't delete them itself, making a mess */ for(TQWidget *w=d->idleButtons.first(); w; w=d->idleButtons.next()) w->blockSignals(false); d->idleButtons.clear(); TQToolBar::clear(); widget2id.clear(); id2widget.clear(); } void KToolBar::removeItem(int id) { Id2WidgetMap::Iterator it = id2widget.find( id ); if ( it == id2widget.end() ) { kdDebug(220) << name() << " KToolBar::removeItem item " << id << " not found" << endl; return; } TQWidget * w = (*it); id2widget.remove( id ); widget2id.remove( w ); widgets.removeRef( w ); delete w; } void KToolBar::removeItemDelayed(int id) { Id2WidgetMap::Iterator it = id2widget.find( id ); if ( it == id2widget.end() ) { kdDebug(220) << name() << " KToolBar::removeItem item " << id << " not found" << endl; return; } TQWidget * w = (*it); id2widget.remove( id ); widget2id.remove( w ); widgets.removeRef( w ); w->blockSignals(true); d->idleButtons.append(w); layoutTimer->start( 50, true ); } void KToolBar::hideItem (int id) { TQWidget *w = getWidget(id); if ( w ) w->hide(); } void KToolBar::showItem (int id) { TQWidget *w = getWidget(id); if ( w ) w->show(); } int KToolBar::itemIndex (int id) { TQWidget *w = getWidget(id); return w ? widgets.findRef(w) : -1; } int KToolBar::idAt (int index) { TQWidget *w = widgets.at(index); return widget2id[w]; } void KToolBar::setFullSize(bool flag ) { setHorizontalStretchable( flag ); setVerticalStretchable( flag ); } bool KToolBar::fullSize() const { return isHorizontalStretchable() || isVerticalStretchable(); } void KToolBar::enableMoving(bool flag ) { setMovingEnabled(flag); } void KToolBar::setBarPos (BarPosition bpos) { if ( !mainWindow() ) return; mainWindow()->moveDockWindow( this, (Dock)bpos ); //kdDebug(220) << name() << " setBarPos dockWindowIndex=" << dockWindowIndex() << endl; } KToolBar::BarPosition KToolBar::barPos() const { if ( !this->mainWindow() ) return place() == TQDockWindow::InDock ? KToolBar::Top : KToolBar::Floating; Dock dock; int dm1, dm2; bool dm3; this->mainWindow()->getLocation( (TQToolBar*)this, dock, dm1, dm3, dm2 ); if ( dock == DockUnmanaged ) { return (KToolBar::BarPosition)DockTop; } return (BarPosition)dock; } bool KToolBar::enable(BarStatus stat) { bool mystat = isVisible(); if ( (stat == Toggle && mystat) || stat == Hide ) hide(); else show(); return isVisible() == mystat; } void KToolBar::setMaxHeight ( int h ) { setMaximumHeight( h ); } int KToolBar::maxHeight() { return maximumHeight(); } void KToolBar::setMaxWidth (int dw) { setMaximumWidth( dw ); } int KToolBar::maxWidth() { return maximumWidth(); } void KToolBar::setTitle (const TQString& _title) { setLabel( _title ); } void KToolBar::enableFloating (bool ) { } void KToolBar::setIconText(IconText it) { setIconText( it, true ); } void KToolBar::setIconText(IconText icontext, bool update) { bool doUpdate=false; if (icontext != d->m_iconText) { d->m_iconText = icontext; doUpdate=true; //kdDebug(220) << name() << " icontext has changed, doUpdate=true" << endl; } else { //kdDebug(220) << name() << " icontext hasn't changed, doUpdate=false" << endl; } if (!update) return; if (doUpdate) doModeChange(); // tell buttons what happened // ugly hack to force a TQMainWindow::triggerLayout( true ) TQMainWindow *mw = mainWindow(); if ( mw ) { mw->setUpdatesEnabled( false ); mw->setToolBarsMovable( !mw->toolBarsMovable() ); mw->setToolBarsMovable( !mw->toolBarsMovable() ); mw->setUpdatesEnabled( true ); } } KToolBar::IconText KToolBar::iconText() const { return d->m_iconText; } void KToolBar::setIconSize(int size) { setIconSize( size, true ); } void KToolBar::setIconSize(int size, bool update) { bool doUpdate=false; if ( size != d->m_iconSize ) { d->m_iconSize = size; doUpdate=true; } if (!update) return; if (doUpdate) doModeChange(); // tell buttons what happened // ugly hack to force a TQMainWindow::triggerLayout( true ) if ( mainWindow() ) { TQMainWindow *mw = mainWindow(); mw->setUpdatesEnabled( false ); mw->setToolBarsMovable( !mw->toolBarsMovable() ); mw->setToolBarsMovable( !mw->toolBarsMovable() ); mw->setUpdatesEnabled( true ); } } int KToolBar::iconSize() const { if ( !d->m_iconSize ) // default value? return iconSizeDefault(); return d->m_iconSize; } int KToolBar::iconSizeDefault() const { if (!::qstrcmp(TQObject::name(), "mainToolBar")) return KGlobal::iconLoader()->currentSize(KIcon::MainToolbar); return KGlobal::iconLoader()->currentSize(KIcon::Toolbar); } void KToolBar::setEnableContextMenu(bool enable ) { d->m_enableContext = enable; } bool KToolBar::contextMenuEnabled() const { return d->m_enableContext; } void KToolBar::setItemNoStyle(int id, bool no_style ) { KToolBarButton * button = getButton( id ); if (button) button->setNoStyle( no_style ); } void KToolBar::setFlat (bool flag) { if ( !mainWindow() ) return; if ( flag ) mainWindow()->moveDockWindow( this, DockMinimized ); else mainWindow()->moveDockWindow( this, DockTop ); // And remember to save the new look later KMainWindow *kmw = dynamic_cast<KMainWindow *>(mainWindow()); if ( kmw ) kmw->setSettingsDirty(); } int KToolBar::count() const { return id2widget.count(); } void KToolBar::saveState() { // first, try to save to the xml file if ( d->m_xmlguiClient && !d->m_xmlguiClient->xmlFile().isEmpty() ) { //kdDebug(220) << name() << " saveState: saving to " << d->m_xmlguiClient->xmlFile() << endl; TQString barname(!::qstrcmp(name(), "unnamed") ? "mainToolBar" : name()); // try to find our toolbar d->modified = false; // go down one level to get to the right tags TQDomElement current; for( TQDomNode n = d->m_xmlguiClient->domDocument().documentElement().firstChild(); !n.isNull(); n = n.nextSibling()) { current = n.toElement(); if ( current.tagName().lower() != "toolbar" ) continue; TQString curname(current.attribute( "name" )); if ( curname == barname ) { saveState( current ); break; } } // if we didn't make changes, then just return if ( !d->modified ) return; // now we load in the (non-merged) local file TQString local_xml(KXMLGUIFactory::readConfigFile(d->m_xmlguiClient->xmlFile(), true, d->m_xmlguiClient->instance())); TQDomDocument local; local.setContent(local_xml); // make sure we don't append if this toolbar already exists locally bool just_append = true; for( TQDomNode n = local.documentElement().firstChild(); !n.isNull(); n = n.nextSibling()) { TQDomElement elem = n.toElement(); if ( elem.tagName().lower() != "toolbar" ) continue; TQString curname(elem.attribute( "name" )); if ( curname == barname ) { just_append = false; local.documentElement().replaceChild( current, elem ); break; } } if (just_append) local.documentElement().appendChild( current ); KXMLGUIFactory::saveConfigFile(local, d->m_xmlguiClient->localXMLFile(), d->m_xmlguiClient->instance() ); return; } // if that didn't work, we save to the config file KConfig *config = KGlobal::config(); saveSettings(config, TQString::null); config->sync(); } TQString KToolBar::settingsGroup() const { TQString configGroup; if (!::qstrcmp(name(), "unnamed") || !::qstrcmp(name(), "mainToolBar")) configGroup = "Toolbar style"; else configGroup = TQString(name()) + " Toolbar style"; if ( this->mainWindow() ) { configGroup.prepend(" "); configGroup.prepend( this->mainWindow()->name() ); } return configGroup; } void KToolBar::saveSettings(KConfig *config, const TQString &_configGroup) { TQString configGroup = _configGroup; if (configGroup.isEmpty()) configGroup = settingsGroup(); //kdDebug(220) << name() << " saveSettings() group=" << _configGroup << " -> " << configGroup << endl; TQString position, icontext; int index; getAttributes( position, icontext, index ); //kdDebug(220) << name() << " position=" << position << " index=" << index << " offset=" << offset() << " newLine=" << newLine() << endl; KConfigGroupSaver saver(config, configGroup); if(!config->hasDefault("Position") && position == d->PositionDefault ) config->revertToDefault("Position"); else config->writeEntry("Position", position); //kdDebug(220) << name() << " icontext=" << icontext << " hasDefault:" << config->hasDefault( "IconText" ) << " d->IconTextDefault=" << d->IconTextDefault << endl; if(d->m_honorStyle && icontext == d->IconTextDefault && !config->hasDefault("IconText") ) { //kdDebug(220) << name() << " reverting icontext to default" << endl; config->revertToDefault("IconText"); } else { //kdDebug(220) << name() << " writing icontext " << icontext << endl; config->writeEntry("IconText", icontext); } if(!config->hasDefault("IconSize") && iconSize() == iconSizeDefault() ) config->revertToDefault("IconSize"); else config->writeEntry("IconSize", iconSize()); if(!config->hasDefault("Hidden") && isHidden() == d->HiddenDefault ) config->revertToDefault("Hidden"); else config->writeEntry("Hidden", isHidden()); // Note that index, unlike the other settings, depends on the other toolbars // So on the first run with a clean local config file, even the usual // hasDefault/==IndexDefault test would save the toolbar indexes // (IndexDefault was 0, whereas index is the real index in the GUI) // // Saving the whole set of indexes is necessary though. When moving only // one toolbar, if we only saved the changed indexes, the toolbars wouldn't // reappear at the same position the next time. // The whole set of indexes has to be saved. //kdDebug(220) << name() << " writing index " << index << endl; KMainWindow *kmw = dynamic_cast<KMainWindow *>(mainWindow()); // don't save if there's only one toolbar // Don't use kmw->toolBarIterator() because you might // mess up someone else's iterator. Make the list on your own TQPtrList<KToolBar> toolbarList; TQPtrList<TQToolBar> lst; for ( int i = (int)TQMainWindow::DockUnmanaged; i <= (int)DockMinimized; ++i ) { lst = kmw->toolBars( (ToolBarDock)i ); for ( TQToolBar *tb = lst.first(); tb; tb = lst.next() ) { if ( !tb->inherits( "KToolBar" ) ) continue; toolbarList.append( (KToolBar*)tb ); } } TQPtrListIterator<KToolBar> toolbarIterator( toolbarList ); if ( !kmw || toolbarIterator.count() > 1 ) config->writeEntry("Index", index); else config->revertToDefault("Index"); if(!config->hasDefault("Offset") && offset() == d->OffsetDefault ) config->revertToDefault("Offset"); else config->writeEntry("Offset", offset()); if(!config->hasDefault("NewLine") && newLine() == d->NewLineDefault ) config->revertToDefault("NewLine"); else config->writeEntry("NewLine", newLine()); } void KToolBar::setXMLGUIClient( KXMLGUIClient *client ) { d->m_xmlguiClient = client; } void KToolBar::setText( const TQString & txt ) { setLabel( txt + " (" + kapp->caption() + ") " ); } TQString KToolBar::text() const { return label(); } void KToolBar::doConnections( KToolBarButton *button ) { connect(button, TQT_SIGNAL(clicked(int)), this, TQT_SIGNAL( clicked( int ) ) ); connect(button, TQT_SIGNAL(doubleClicked(int)), this, TQT_SIGNAL( doubleClicked( int ) ) ); connect(button, TQT_SIGNAL(released(int)), this, TQT_SIGNAL( released( int ) ) ); connect(button, TQT_SIGNAL(pressed(int)), this, TQT_SIGNAL( pressed( int ) ) ); connect(button, TQT_SIGNAL(toggled(int)), this, TQT_SIGNAL( toggled( int ) ) ); connect(button, TQT_SIGNAL(highlighted(int, bool)), this, TQT_SIGNAL( highlighted( int, bool ) ) ); } void KToolBar::mousePressEvent ( TQMouseEvent *m ) { if ( !mainWindow() ) return; TQMainWindow *mw = mainWindow(); if ( mw->toolBarsMovable() && d->m_enableContext ) { if ( m->button() == RightButton ) { TQGuardedPtr<KToolBar> guard( this ); int i = contextMenu()->exec( m->globalPos(), 0 ); // "Configure Toolbars" recreates toolbars, so we might not exist anymore. if ( guard ) slotContextAboutToHide(); switch ( i ) { case -1: return; // popup canceled case CONTEXT_LEFT: mw->moveDockWindow( this, DockLeft ); break; case CONTEXT_RIGHT: mw->moveDockWindow( this, DockRight ); break; case CONTEXT_TOP: mw->moveDockWindow( this, DockTop ); break; case CONTEXT_BOTTOM: mw->moveDockWindow( this, DockBottom ); break; case CONTEXT_FLOAT: mw->moveDockWindow( this, DockTornOff ); break; case CONTEXT_FLAT: mw->moveDockWindow( this, DockMinimized ); break; case CONTEXT_ICONS: setIconText( IconOnly ); break; case CONTEXT_TEXTRIGHT: setIconText( IconTextRight ); break; case CONTEXT_TEXT: setIconText( TextOnly ); break; case CONTEXT_TEXTUNDER: setIconText( IconTextBottom ); break; default: if ( i >= CONTEXT_ICONSIZES ) setIconSize( i - CONTEXT_ICONSIZES ); else return; // assume this was an action handled elsewhere, no need for setSettingsDirty() } KMainWindow *kmw = dynamic_cast<KMainWindow *>(mw); if ( kmw ) kmw->setSettingsDirty(); } } } void KToolBar::doModeChange() { for(TQWidget *w=d->idleButtons.first(); w; w=d->idleButtons.next()) w->blockSignals(false); d->idleButtons.clear(); emit modechange(); } void KToolBar::rebuildLayout() { for(TQWidget *w=d->idleButtons.first(); w; w=d->idleButtons.next()) w->blockSignals(false); d->idleButtons.clear(); layoutTimer->stop(); TQApplication::sendPostedEvents( this, TQEvent::ChildInserted ); TQBoxLayout *l = boxLayout(); // clear the old layout TQLayoutIterator it = l->iterator(); while ( it.current() ) it.deleteCurrent(); for ( TQWidget *w = widgets.first(); w; w = widgets.next() ) { if ( w == rightAligned ) continue; KToolBarSeparator *ktbs = dynamic_cast<KToolBarSeparator *>(w); if ( ktbs && !ktbs->showLine() ) { l->addSpacing( orientation() == Vertical ? w->sizeHint().height() : w->sizeHint().width() ); w->hide(); continue; } if ( dynamic_cast<TQPopupMenu *>(w) ) // w is a QPopupMenu? continue; l->addWidget( w ); w->show(); if ((orientation() == Horizontal) && dynamic_cast<TQLineEdit *>(w)) // w is TQLineEdit ? l->addSpacing(2); // A little bit extra spacing behind it. } if ( rightAligned ) { l->addStretch(); l->addWidget( rightAligned ); rightAligned->show(); } if ( fullSize() ) { if ( !rightAligned ) l->addStretch(); if ( stretchableWidget ) l->setStretchFactor( stretchableWidget, 10 ); } l->invalidate(); TQApplication::postEvent( this, new TQEvent( TQEvent::LayoutHint ) ); } void KToolBar::childEvent( TQChildEvent *e ) { if ( e->child()->isWidgetType() ) { TQWidget * w = dynamic_cast<TQWidget *>(e->child()); if (!w || !(::qstrcmp( "qt_dockwidget_internal", w->name()))) { TQToolBar::childEvent( e ); return; } if ( e->type() == TQEvent::ChildInserted ) { if ( !dynamic_cast<TQPopupMenu *>(w)) { // e->child() is not a QPopupMenu // prevent items that have been explicitly inserted by insert*() from // being inserted again if ( !widget2id.contains( w ) ) { int dummy = -1; insertWidgetInternal( w, dummy, -1 ); } } } else { removeWidgetInternal( w ); } if ( isVisibleTo( 0 ) ) { layoutTimer->start( 50, true ); TQBoxLayout *l = boxLayout(); // clear the old layout so that we don't get unnecassery layout // changes till we have rebuild the thing TQLayoutIterator it = l->iterator(); while ( it.current() ) it.deleteCurrent(); } } TQToolBar::childEvent( e ); } void KToolBar::insertWidgetInternal( TQWidget *w, int &index, int id ) { // we can't have it in widgets, or something is really wrong //widgets.removeRef( w ); connect( w, TQT_SIGNAL( destroyed() ), this, TQT_SLOT( widgetDestroyed() ) ); if ( index == -1 || index > (int)widgets.count() ) { index = (int)widgets.count(); widgets.append( w ); } else widgets.insert( index, w ); if ( id == -1 ) id = id2widget.count(); id2widget.insert( id, w ); widget2id.insert( w, id ); } void KToolBar::showEvent( TQShowEvent *e ) { TQToolBar::showEvent( e ); rebuildLayout(); } void KToolBar::setStretchableWidget( TQWidget *w ) { TQToolBar::setStretchableWidget( w ); stretchableWidget = w; } TQSizePolicy KToolBar::sizePolicy() const { if ( orientation() == Horizontal ) return TQSizePolicy( TQSizePolicy::Expanding, TQSizePolicy::Fixed ); else return TQSizePolicy( TQSizePolicy::Fixed, TQSizePolicy::Expanding ); } TQSize KToolBar::sizeHint() const { TQSize minSize(0,0); KToolBar *ncThis = const_cast<KToolBar *>(this); ncThis->polish(); int margin = static_cast<TQWidget*>(ncThis)->layout()->margin() + frameWidth(); switch( barPos() ) { case KToolBar::Top: case KToolBar::Bottom: for ( TQWidget *w = ncThis->widgets.first(); w; w = ncThis->widgets.next() ) { TQSize sh = w->sizeHint(); if ( w->sizePolicy().horData() == TQSizePolicy::Ignored ) sh.setWidth( 1 ); if ( w->sizePolicy().verData() == TQSizePolicy::Ignored ) sh.setHeight( 1 ); sh = sh.boundedTo( w->maximumSize() ) .expandedTo( w->minimumSize() ).expandedTo( TQSize(1, 1) ); minSize = minSize.expandedTo(TQSize(0, sh.height())); minSize += TQSize(sh.width()+1, 0); if (dynamic_cast<TQLineEdit *>(w)) // w is a TQLineEdit ? minSize += TQSize(2, 0); // A little bit extra spacing behind it. } minSize += TQSize(TQApplication::style().pixelMetric( TQStyle::PM_DockWindowHandleExtent ), 0); minSize += TQSize(margin*2, margin*2); break; case KToolBar::Left: case KToolBar::Right: for ( TQWidget *w = ncThis->widgets.first(); w; w = ncThis->widgets.next() ) { TQSize sh = w->sizeHint(); if ( w->sizePolicy().horData() == TQSizePolicy::Ignored ) sh.setWidth( 1 ); if ( w->sizePolicy().verData() == TQSizePolicy::Ignored ) sh.setHeight( 1 ); sh = sh.boundedTo( w->maximumSize() ) .expandedTo( w->minimumSize() ).expandedTo( TQSize(1, 1) ); minSize = minSize.expandedTo(TQSize(sh.width(), 0)); minSize += TQSize(0, sh.height()+1); } minSize += TQSize(0, TQApplication::style().pixelMetric( TQStyle::PM_DockWindowHandleExtent )); minSize += TQSize(margin*2, margin*2); break; default: minSize = TQToolBar::sizeHint(); break; } return minSize; } TQSize KToolBar::minimumSize() const { return minimumSizeHint(); } TQSize KToolBar::minimumSizeHint() const { return sizeHint(); } bool KToolBar::highlight() const { return d->m_highlight; } void KToolBar::hide() { TQToolBar::hide(); } void KToolBar::show() { TQToolBar::show(); } void KToolBar::resizeEvent( TQResizeEvent *e ) { bool b = isUpdatesEnabled(); setUpdatesEnabled( false ); TQToolBar::resizeEvent( e ); if (b) { if (layoutTimer->isActive()) { // Wait with repainting till layout is complete. d->repaintTimer.start( 100, true ); } else { // Repaint now slotRepaint(); } } } void KToolBar::slotIconChanged(int group) { if ((group != KIcon::Toolbar) && (group != KIcon::MainToolbar)) return; if ((group == KIcon::MainToolbar) != !::qstrcmp(name(), "mainToolBar")) return; doModeChange(); if (isVisible()) updateGeometry(); } void KToolBar::slotReadConfig() { //kdDebug(220) << name() << " slotReadConfig" << endl; // Read appearance settings (hmm, we used to do both here, // but a well behaved application will call applyMainWindowSettings // anyway, right ?) applyAppearanceSettings(KGlobal::config(), TQString::null ); } void KToolBar::slotAppearanceChanged() { // Read appearance settings from global file. applyAppearanceSettings(KGlobal::config(), TQString::null, true /* lose local settings */ ); // And remember to save the new look later KMainWindow *kmw = dynamic_cast<KMainWindow *>(mainWindow()); if ( kmw ) kmw->setSettingsDirty(); } //static bool KToolBar::highlightSetting() { TQString grpToolbar(TQString::fromLatin1("Toolbar style")); KConfigGroupSaver saver(KGlobal::config(), grpToolbar); return KGlobal::config()->readBoolEntry(TQString::fromLatin1("Highlighting"),true); } //static bool KToolBar::transparentSetting() { TQString grpToolbar(TQString::fromLatin1("Toolbar style")); KConfigGroupSaver saver(KGlobal::config(), grpToolbar); return KGlobal::config()->readBoolEntry(TQString::fromLatin1("TransparentMoving"),true); } //static KToolBar::IconText KToolBar::iconTextSetting() { TQString grpToolbar(TQString::fromLatin1("Toolbar style")); KConfigGroupSaver saver(KGlobal::config(), grpToolbar); TQString icontext = KGlobal::config()->readEntry(TQString::fromLatin1("IconText"),TQString::fromLatin1("IconOnly")); if ( icontext == "IconTextRight" ) return IconTextRight; else if ( icontext == "IconTextBottom" ) return IconTextBottom; else if ( icontext == "TextOnly" ) return TextOnly; else return IconOnly; } void KToolBar::applyAppearanceSettings(KConfig *config, const TQString &_configGroup, bool forceGlobal) { TQString configGroup = _configGroup.isEmpty() ? settingsGroup() : _configGroup; //kdDebug(220) << name() << " applyAppearanceSettings: configGroup=" << configGroup << " forceGlobal=" << forceGlobal << endl; // If we have application-specific settings in the XML file, // and nothing in the application's config file, then // we don't apply the global defaults, the XML ones are preferred // (see applySettings for a full explanation) // This is the reason for the xmlgui tests below. bool xmlgui = d->m_xmlguiClient && !d->m_xmlguiClient->xmlFile().isEmpty(); KConfig *gconfig = KGlobal::config(); static const TQString &attrIconText = KGlobal::staticQString("IconText"); static const TQString &attrHighlight = KGlobal::staticQString("Highlighting"); static const TQString &attrTrans = KGlobal::staticQString("TransparentMoving"); static const TQString &attrIconSize = KGlobal::staticQString("IconSize"); // we actually do this in two steps. // First, we read in the global styles [Toolbar style] (from the KControl module). // Then, if the toolbar is NOT 'mainToolBar', we will also try to read in [barname Toolbar style] bool highlight; int transparent; bool applyIconText = !xmlgui; // if xmlgui is used, global defaults won't apply bool applyIconSize = !xmlgui; int iconSize = d->IconSizeDefault; TQString iconText = d->IconTextDefault; // this is the first iteration TQString grpToolbar(TQString::fromLatin1("Toolbar style")); { // start block for KConfigGroupSaver KConfigGroupSaver saver(gconfig, grpToolbar); // first, get the generic settings highlight = gconfig->readBoolEntry(attrHighlight, true); transparent = gconfig->readBoolEntry(attrTrans, true); // we read in the IconText property *only* if we intend on actually // honoring it if (d->m_honorStyle) d->IconTextDefault = gconfig->readEntry(attrIconText, d->IconTextDefault); else d->IconTextDefault = "IconOnly"; // Use the default icon size for toolbar icons. d->IconSizeDefault = gconfig->readNumEntry(attrIconSize, d->IconSizeDefault); iconSize = d->IconSizeDefault; iconText = d->IconTextDefault; if ( !forceGlobal && config->hasGroup(configGroup) ) { config->setGroup(configGroup); // first, get the generic settings highlight = config->readBoolEntry(attrHighlight, highlight); transparent = config->readBoolEntry(attrTrans, transparent); // read in the IconText property if ( config->hasKey( attrIconText ) ) { iconText = config->readEntry(attrIconText); applyIconText = true; //kdDebug(220) << name() << " read icontext=" << d->IconTextDefault << ", that will be the default" << endl; } // now get the size if ( config->hasKey( attrIconSize ) ) { iconSize = config->readNumEntry(attrIconSize); applyIconSize = true; } } // revert back to the old group } // end block for KConfigGroupSaver bool doUpdate = false; IconText icon_text; if ( iconText == "IconTextRight" ) icon_text = IconTextRight; else if ( iconText == "IconTextBottom" ) icon_text = IconTextBottom; else if ( iconText == "TextOnly" ) icon_text = TextOnly; else icon_text = IconOnly; // check if the icon/text has changed if (icon_text != d->m_iconText && applyIconText) { //kdDebug(220) << name() << " applyAppearanceSettings setIconText " << icon_text << endl; setIconText(icon_text, false); doUpdate = true; } // ...and check if the icon size has changed if (iconSize != d->m_iconSize && applyIconSize) { setIconSize(iconSize, false); doUpdate = true; } TQMainWindow *mw = mainWindow(); // ...and if we should highlight if ( highlight != d->m_highlight ) { d->m_highlight = highlight; doUpdate = true; } // ...and if we should move transparently if ( mw && transparent != (!mw->opaqueMoving()) ) { mw->setOpaqueMoving( !transparent ); } if (doUpdate) doModeChange(); // tell buttons what happened if (isVisible ()) updateGeometry(); } void KToolBar::applySettings(KConfig *config, const TQString &_configGroup) { return applySettings(config,_configGroup,false); } void KToolBar::applySettings(KConfig *config, const TQString &_configGroup, bool force) { //kdDebug(220) << name() << " applySettings group=" << _configGroup << endl; TQString configGroup = _configGroup.isEmpty() ? settingsGroup() : _configGroup; /* Let's explain this a bit more in details. The order in which we apply settings is : Global config / <appnamerc> user settings if no XMLGUI is used Global config / App-XML attributes / <appnamerc> user settings if XMLGUI is used So in the first case, we simply read everything from KConfig as below, but in the second case we don't do anything here if there is no app-specific config, and the XMLGUI-related code (loadState()) uses the static methods of this class to get the global defaults. Global config doesn't include position (index, offset, newline and hidden/shown). */ // First the appearance stuff - the one which has a global config applyAppearanceSettings( config, configGroup ); // ...and now the position stuff if ( config->hasGroup(configGroup) || force ) { KConfigGroupSaver cgs(config, configGroup); static const TQString &attrPosition = KGlobal::staticQString("Position"); static const TQString &attrIndex = KGlobal::staticQString("Index"); static const TQString &attrOffset = KGlobal::staticQString("Offset"); static const TQString &attrNewLine = KGlobal::staticQString("NewLine"); static const TQString &attrHidden = KGlobal::staticQString("Hidden"); TQString position = config->readEntry(attrPosition, d->PositionDefault); int index = config->readNumEntry(attrIndex, -1); int offset = config->readNumEntry(attrOffset, d->OffsetDefault); bool newLine = config->readBoolEntry(attrNewLine, d->NewLineDefault); bool hidden = config->readBoolEntry(attrHidden, d->HiddenDefault); Dock pos(DockTop); if ( position == "Top" ) pos = DockTop; else if ( position == "Bottom" ) pos = DockBottom; else if ( position == "Left" ) pos = DockLeft; else if ( position == "Right" ) pos = DockRight; else if ( position == "Floating" ) pos = DockTornOff; else if ( position == "Flat" ) pos = DockMinimized; //kdDebug(220) << name() << " applySettings hidden=" << hidden << endl; if (hidden) hide(); else show(); if ( mainWindow() ) { //kdDebug(220) << name() << " applySettings updating ToolbarInfo" << endl; d->toolBarInfo = KToolBarPrivate::ToolBarInfo( pos, index, newLine, offset ); positionYourself( true ); } if (isVisible ()) updateGeometry(); } } bool KToolBar::event( TQEvent *e ) { if ( (e->type() == TQEvent::LayoutHint) && isUpdatesEnabled() ) d->repaintTimer.start( 100, true ); if (e->type() == TQEvent::ChildInserted ) { // Bypass TQToolBar::event, // it will show() the inserted child and we don't want to // do that until we have rebuilt the layout. childEvent((TQChildEvent *)e); return true; } return TQToolBar::event( e ); } void KToolBar::slotRepaint() { setUpdatesEnabled( false ); // Send a resizeEvent to update the "toolbar extension arrow" // (The button you get when your toolbar-items don't fit in // the available space) TQResizeEvent ev(size(), size()); resizeEvent(&ev); TQApplication::sendPostedEvents( this, TQEvent::LayoutHint ); setUpdatesEnabled( true ); repaint( true ); } void KToolBar::toolBarPosChanged( TQToolBar *tb ) { if ( tb != this ) return; if ( d->oldPos == DockMinimized ) rebuildLayout(); d->oldPos = (TQMainWindow::ToolBarDock)barPos(); KMainWindow *kmw = dynamic_cast<KMainWindow *>(mainWindow()); if ( kmw ) kmw->setSettingsDirty(); } static KToolBar::Dock stringToDock( const TQString& attrPosition ) { KToolBar::Dock dock = KToolBar::DockTop; if ( !attrPosition.isEmpty() ) { if ( attrPosition == "top" ) dock = KToolBar::DockTop; else if ( attrPosition == "left" ) dock = KToolBar::DockLeft; else if ( attrPosition == "right" ) dock = KToolBar::DockRight; else if ( attrPosition == "bottom" ) dock = KToolBar::DockBottom; else if ( attrPosition == "floating" ) dock = KToolBar::DockTornOff; else if ( attrPosition == "flat" ) dock = KToolBar::DockMinimized; } return dock; } void KToolBar::loadState( const TQDomElement &element ) { TQMainWindow *mw = mainWindow(); if ( !mw ) return; { TQCString text = element.namedItem( "text" ).toElement().text().utf8(); if ( text.isEmpty() ) text = element.namedItem( "Text" ).toElement().text().utf8(); if ( !text.isEmpty() ) setText( i18n( text ) ); } { TQCString attrFullWidth = element.attribute( "fullWidth" ).lower().latin1(); if ( !attrFullWidth.isEmpty() ) setFullSize( attrFullWidth == "true" ); } /* This method is called in order to load toolbar settings from XML. However this can be used in two rather different cases: - for the initial loading of the app's XML. In that case the settings are only the defaults, the user's KConfig settings will override them (KDE4 TODO: how about saving those user settings into the local XML file instead? Then this whole thing would be simpler, no KConfig settings to apply afterwards. OTOH we'd have to migrate those settings when the .rc version increases, like we do for shortcuts) - for later re-loading when switching between parts in KXMLGUIFactory. In that case the XML contains the final settings, not the defaults. We do need the defaults, and the toolbar might have been completely deleted and recreated meanwhile. So we store the app-default settings into the XML. */ bool loadingAppDefaults = true; if ( element.hasAttribute( "offsetDefault" ) ) { // this isn't the first time, so the defaults have been saved into the (in-memory) XML loadingAppDefaults = false; d->OffsetDefault = element.attribute( "offsetDefault" ).toInt(); d->NewLineDefault = element.attribute( "newlineDefault" ) == "true"; d->HiddenDefault = element.attribute( "hiddenDefault" ) == "true"; d->IconSizeDefault = element.attribute( "iconSizeDefault" ).toInt(); d->PositionDefault = element.attribute( "positionDefault" ); d->IconTextDefault = element.attribute( "iconTextDefault" ); } //kdDebug(220) << name() << " loadState loadingAppDefaults=" << loadingAppDefaults << endl; Dock dock = stringToDock( element.attribute( "position" ).lower() ); { TQCString attrIconText = element.attribute( "iconText" ).lower().latin1(); if ( !attrIconText.isEmpty() ) { //kdDebug(220) << name() << " loadState attrIconText=" << attrIconText << endl; if ( attrIconText == "icontextright" ) setIconText( KToolBar::IconTextRight ); else if ( attrIconText == "textonly" ) setIconText( KToolBar::TextOnly ); else if ( attrIconText == "icontextbottom" ) setIconText( KToolBar::IconTextBottom ); else if ( attrIconText == "icononly" ) setIconText( KToolBar::IconOnly ); } else { //kdDebug(220) << name() << " loadState no iconText attribute in XML, using iconTextSetting=" << iconTextSetting() << endl; // Use global setting if (d->m_honorStyle) setIconText( iconTextSetting() ); else setIconText( d->IconTextDefault ); } } TQString attrIconSize = element.attribute( "iconSize" ).lower(); int iconSize = d->IconSizeDefault; if ( !attrIconSize.isEmpty() ) iconSize = attrIconSize.toInt(); setIconSize( iconSize ); int index = -1; // append by default. This is very important, otherwise // with all 0 indexes, we keep reversing the toolbars. { TQString attrIndex = element.attribute( "index" ).lower(); if ( !attrIndex.isEmpty() ) index = attrIndex.toInt(); } int offset = d->OffsetDefault; bool newLine = d->NewLineDefault; bool hidden = d->HiddenDefault; { TQString attrOffset = element.attribute( "offset" ); if ( !attrOffset.isEmpty() ) offset = attrOffset.toInt(); } { TQString attrNewLine = element.attribute( "newline" ).lower(); if ( !attrNewLine.isEmpty() ) newLine = attrNewLine == "true"; } { TQString attrHidden = element.attribute( "hidden" ).lower(); if ( !attrHidden.isEmpty() ) { hidden = attrHidden == "true"; } } d->toolBarInfo = KToolBarPrivate::ToolBarInfo( dock, index, newLine, offset ); mw->addDockWindow( this, dock, newLine ); mw->moveDockWindow( this, dock, newLine, index, offset ); // Apply the highlight button setting d->m_highlight = highlightSetting(); if ( hidden ) hide(); else show(); if ( loadingAppDefaults ) { getAttributes( d->PositionDefault, d->IconTextDefault, index ); //kdDebug(220) << name() << " loadState IconTextDefault=" << d->IconTextDefault << endl; d->OffsetDefault = offset; d->NewLineDefault = newLine; d->HiddenDefault = hidden; d->IconSizeDefault = iconSize; } //kdDebug(220) << name() << " loadState hidden=" << hidden << endl; // Apply transparent-toolbar-moving setting (ok, this is global to the mainwindow, // but we do it only if there are toolbars...) // KDE4: move to KMainWindow if ( transparentSetting() != !mw->opaqueMoving() ) mw->setOpaqueMoving( !transparentSetting() ); } int KToolBar::dockWindowIndex() { int index = 0; Q_ASSERT( mainWindow() ); if ( mainWindow() ) { TQMainWindow::ToolBarDock dock; bool newLine; int offset; mainWindow()->getLocation( this, dock, index, newLine, offset ); } return index; } void KToolBar::getAttributes( TQString &position, TQString &icontext, int &index ) { // get all of the stuff to save switch ( barPos() ) { case KToolBar::Flat: position = "Flat"; break; case KToolBar::Bottom: position = "Bottom"; break; case KToolBar::Left: position = "Left"; break; case KToolBar::Right: position = "Right"; break; case KToolBar::Floating: position = "Floating"; break; case KToolBar::Top: default: position = "Top"; break; } index = dockWindowIndex(); switch (d->m_iconText) { case KToolBar::IconTextRight: icontext = "IconTextRight"; break; case KToolBar::IconTextBottom: icontext = "IconTextBottom"; break; case KToolBar::TextOnly: icontext = "TextOnly"; break; case KToolBar::IconOnly: default: icontext = "IconOnly"; break; } //kdDebug(220) << name() << " getAttributes: icontext=" << icontext << endl; } void KToolBar::saveState( TQDomElement ¤t ) { Q_ASSERT( !current.isNull() ); TQString position, icontext; int index = -1; getAttributes( position, icontext, index ); current.setAttribute( "noMerge", "1" ); current.setAttribute( "position", position ); current.setAttribute( "iconText", icontext ); current.setAttribute( "index", index ); current.setAttribute( "offset", offset() ); current.setAttribute( "newline", newLine() ); if ( isHidden() ) current.setAttribute( "hidden", "true" ); d->modified = true; // TODO if this method is used by more than KXMLGUIBuilder, e.g. to save XML settings to *disk*, // then the stuff below shouldn't always be done. current.setAttribute( "offsetDefault", d->OffsetDefault ); current.setAttribute( "newlineDefault", d->NewLineDefault ); current.setAttribute( "hiddenDefault", d->HiddenDefault ? "true" : "false" ); current.setAttribute( "iconSizeDefault", d->IconSizeDefault ); current.setAttribute( "positionDefault", d->PositionDefault ); current.setAttribute( "iconTextDefault", d->IconTextDefault ); //kdDebug(220) << name() << " saveState: saving index=" << index << " iconText=" << icontext << " hidden=" << isHidden() << endl; } // Called by KMainWindow::finalizeGUI void KToolBar::positionYourself( bool force ) { if (force) d->positioned = false; if ( d->positioned || !mainWindow() ) { //kdDebug(220) << name() << " positionYourself d->positioned=true ALREADY DONE" << endl; return; } // we can't test for ForceHide after moveDockWindow because QDockArea // does a reparent() with showIt == true bool hidden = isHidden(); //kdDebug(220) << name() << " positionYourself dock=" << d->toolBarInfo.dock << " newLine=" << d->toolBarInfo.newline << " index=" << d->toolBarInfo.index << " offset=" << d->toolBarInfo.offset << endl; mainWindow()->moveDockWindow( this, d->toolBarInfo.dock, d->toolBarInfo.newline, d->toolBarInfo.index, d->toolBarInfo.offset ); //kdDebug(220) << name() << " positionYourself dockWindowIndex=" << dockWindowIndex() << endl; if ( hidden ) hide(); else show(); // This method can only have an effect once - unless force is set d->positioned = true; } KPopupMenu *KToolBar::contextMenu() { if ( context ) return context; // Construct our context popup menu. Name it qt_dockwidget_internal so it // won't be deleted by TQToolBar::clear(). context = new KPopupMenu( this, "qt_dockwidget_internal" ); context->insertTitle(i18n("Toolbar Menu")); KPopupMenu *orient = new KPopupMenu( context, "orient" ); orient->insertItem( i18n("toolbar position string","Top"), CONTEXT_TOP ); orient->insertItem( i18n("toolbar position string","Left"), CONTEXT_LEFT ); orient->insertItem( i18n("toolbar position string","Right"), CONTEXT_RIGHT ); orient->insertItem( i18n("toolbar position string","Bottom"), CONTEXT_BOTTOM ); orient->insertSeparator(-1); orient->insertItem( i18n("toolbar position string","Floating"), CONTEXT_FLOAT ); orient->insertItem( i18n("min toolbar", "Flat"), CONTEXT_FLAT ); KPopupMenu *mode = new KPopupMenu( context, "mode" ); mode->insertItem( i18n("Icons Only"), CONTEXT_ICONS ); mode->insertItem( i18n("Text Only"), CONTEXT_TEXT ); mode->insertItem( i18n("Text Alongside Icons"), CONTEXT_TEXTRIGHT ); mode->insertItem( i18n("Text Under Icons"), CONTEXT_TEXTUNDER ); KPopupMenu *size = new KPopupMenu( context, "size" ); size->insertItem( i18n("Default"), CONTEXT_ICONSIZES ); // Query the current theme for available sizes KIconTheme *theme = KGlobal::instance()->iconLoader()->theme(); TQValueList<int> avSizes; if (theme) { if (!::qstrcmp(TQObject::name(), "mainToolBar")) avSizes = theme->querySizes( KIcon::MainToolbar); else avSizes = theme->querySizes( KIcon::Toolbar); } d->iconSizes = avSizes; qHeapSort(avSizes); TQValueList<int>::Iterator it; if (avSizes.count() < 10) { // Fixed or threshold type icons TQValueList<int>::Iterator end(avSizes.end()); for (it=avSizes.begin(); it!=end; ++it) { TQString text; if ( *it < 19 ) text = i18n("Small (%1x%2)").arg(*it).arg(*it); else if (*it < 25) text = i18n("Medium (%1x%2)").arg(*it).arg(*it); else if (*it < 35) text = i18n("Large (%1x%2)").arg(*it).arg(*it); else text = i18n("Huge (%1x%2)").arg(*it).arg(*it); //we use the size as an id, with an offset size->insertItem( text, CONTEXT_ICONSIZES + *it ); } } else { // Scalable icons. const int progression[] = {16, 22, 32, 48, 64, 96, 128, 192, 256}; it = avSizes.begin(); for (uint i = 0; i < 9; i++) { while (it++ != avSizes.end()) { if (*it >= progression[i]) { TQString text; if ( *it < 19 ) text = i18n("Small (%1x%2)").arg(*it).arg(*it); else if (*it < 25) text = i18n("Medium (%1x%2)").arg(*it).arg(*it); else if (*it < 35) text = i18n("Large (%1x%2)").arg(*it).arg(*it); else text = i18n("Huge (%1x%2)").arg(*it).arg(*it); //we use the size as an id, with an offset size->insertItem( text, CONTEXT_ICONSIZES + *it ); break; } } } } context->insertItem( i18n("Orientation"), orient ); orient->setItemChecked(CONTEXT_TOP, true); context->insertItem( i18n("Text Position"), mode ); context->setItemChecked(CONTEXT_ICONS, true); context->insertItem( i18n("Icon Size"), size ); connect( context, TQT_SIGNAL( aboutToShow() ), this, TQT_SLOT( slotContextAboutToShow() ) ); // Unplugging a submenu from abouttohide leads to the popupmenu floating around // So better simply call that code from after exec() returns (DF) //connect( context, TQT_SIGNAL( aboutToHide() ), this, TQT_SLOT( slotContextAboutToHide() ) ); return context; } void KToolBar::slotContextAboutToShow() { // The idea here is to reuse the "static" part of the menu to save time. // But the "Toolbars" action is dynamic (can be a single action or a submenu) // and ToolBarHandler::setupActions() deletes it, so better not keep it around. // So we currently plug/unplug the last two actions of the menu. // Another way would be to keep around the actions and plug them all into a (new each time) popupmenu. KMainWindow *kmw = dynamic_cast<KMainWindow *>(mainWindow()); if ( kmw ) { kmw->setupToolbarMenuActions(); // Only allow hiding a toolbar if the action is also plugged somewhere else (e.g. menubar) KAction *tbAction = kmw->toolBarMenuAction(); if ( tbAction && tbAction->containerCount() > 0 ) tbAction->plug(context); } // try to find "configure toolbars" action KAction *configureAction = 0; const char* actionName = KStdAction::name(KStdAction::ConfigureToolbars); if ( d->m_xmlguiClient ) configureAction = d->m_xmlguiClient->actionCollection()->action(actionName); if ( !configureAction && kmw ) configureAction = kmw->actionCollection()->action(actionName); if ( configureAction ) configureAction->plug(context); KEditToolbar::setDefaultToolbar(TQObject::name()); for(int i = CONTEXT_ICONS; i <= CONTEXT_TEXTUNDER; ++i) context->setItemChecked(i, false); switch( d->m_iconText ) { case IconOnly: default: context->setItemChecked(CONTEXT_ICONS, true); break; case IconTextRight: context->setItemChecked(CONTEXT_TEXTRIGHT, true); break; case TextOnly: context->setItemChecked(CONTEXT_TEXT, true); break; case IconTextBottom: context->setItemChecked(CONTEXT_TEXTUNDER, true); break; } TQValueList<int>::ConstIterator iIt = d->iconSizes.begin(); TQValueList<int>::ConstIterator iEnd = d->iconSizes.end(); for (; iIt != iEnd; ++iIt ) context->setItemChecked( CONTEXT_ICONSIZES + *iIt, false ); context->setItemChecked( CONTEXT_ICONSIZES, false ); context->setItemChecked( CONTEXT_ICONSIZES + d->m_iconSize, true ); for ( int i = CONTEXT_TOP; i <= CONTEXT_FLAT; ++i ) context->setItemChecked( i, false ); switch ( barPos() ) { case KToolBar::Flat: context->setItemChecked( CONTEXT_FLAT, true ); break; case KToolBar::Bottom: context->setItemChecked( CONTEXT_BOTTOM, true ); break; case KToolBar::Left: context->setItemChecked( CONTEXT_LEFT, true ); break; case KToolBar::Right: context->setItemChecked( CONTEXT_RIGHT, true ); break; case KToolBar::Floating: context->setItemChecked( CONTEXT_FLOAT, true ); break; case KToolBar::Top: context->setItemChecked( CONTEXT_TOP, true ); break; default: break; } } void KToolBar::slotContextAboutToHide() { // We have to unplug whatever slotContextAboutToShow plugged into the menu. // Unplug the toolbar menu action KMainWindow *kmw = dynamic_cast<KMainWindow *>(mainWindow()); if ( kmw && kmw->toolBarMenuAction() ) if ( kmw->toolBarMenuAction()->containerCount() > 1 ) kmw->toolBarMenuAction()->unplug(context); // Unplug the configure toolbars action too, since it's afterwards anyway KAction *configureAction = 0; const char* actionName = KStdAction::name(KStdAction::ConfigureToolbars); if ( d->m_xmlguiClient ) configureAction = d->m_xmlguiClient->actionCollection()->action(actionName); if ( !configureAction && kmw ) configureAction = kmw->actionCollection()->action(actionName); if ( configureAction ) configureAction->unplug(context); TQPtrListIterator<TQWidget> it( widgets ); TQWidget *wdg; while ( ( wdg = it.current() ) != 0 ) { if ( wdg->inherits( "QToolButton" ) ) static_cast<TQToolButton*>( wdg )->setDown( false ); ++it; } } void KToolBar::widgetDestroyed() { removeWidgetInternal( (TQWidget*)sender() ); } void KToolBar::removeWidgetInternal( TQWidget * w ) { widgets.removeRef( w ); TQMap< TQWidget*, int >::Iterator it = widget2id.find( w ); if ( it == widget2id.end() ) return; id2widget.remove( *it ); widget2id.remove( it ); } void KToolBar::virtual_hook( int, void* ) { /*BASE::virtual_hook( id, data );*/ } #include "ktoolbar.moc"