ViewManager.cpp 29.5 KB
Newer Older
1
/*
2
    Copyright 2006-2008 by Robert Knight <robertknight@gmail.com>
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19

    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.
*/

20 21 22
// Own
#include "ViewManager.h"

23 24 25
// System
#include <assert.h>

26
// Qt
27
#include <QtCore/QDateTime>
Dirk Mueller's avatar
Dirk Mueller committed
28
#include <QtCore/QSignalMapper>
29
#include <QtGui/QMenu>
30

31 32
// KDE
#include <kdebug.h>
Robert Knight's avatar
Robert Knight committed
33
#include <KAcceleratorManager>
34
#include <KGlobal>
35 36 37 38 39
#include <KLocale>
#include <KToggleAction>
#include <KXMLGUIFactory>

// Konsole
40
#include "ColorScheme.h"
41
#include "ProfileList.h"
42
#include "Session.h"
43
#include "TerminalDisplay.h"
44 45 46 47 48
#include "SessionController.h"
#include "SessionManager.h"
#include "ViewContainer.h"
#include "ViewSplitter.h"

49 50
using namespace Konsole;

51 52
ViewManager::ViewManager(QObject* parent , KActionCollection* collection)
    : QObject(parent)
53
    , _viewSplitter(0)
54 55
    , _actionCollection(collection)
    , _containerSignalMapper(new QSignalMapper(this))
56
    , _navigationMethod(TabbedNavigation)
57
    , _newViewMenu(0)
58
{
59
    // create main view area
Robert Knight's avatar
Robert Knight committed
60 61 62
    _viewSplitter = new ViewSplitter(0);  
	KAcceleratorManager::setNoAccel(_viewSplitter);

63 64 65 66 67 68 69
    // the ViewSplitter class supports both recursive and non-recursive splitting,
    // in non-recursive mode, all containers are inserted into the same top-level splitter
    // widget, and all the divider lines between the containers have the same orientation
    //
    // the ViewManager class is not currently able to handle a ViewSplitter in recursive-splitting
    // mode 
    _viewSplitter->setRecursiveSplitting(false);
70
    _viewSplitter->setFocusPolicy(Qt::NoFocus);
71

72 73 74 75 76 77
    // setup actions which relating to the view
    setupActions();

    // emit a signal when all of the views held by this view manager are destroyed
    connect( _viewSplitter , SIGNAL(allContainersEmpty()) , this , SIGNAL(empty()) );
    connect( _viewSplitter , SIGNAL(empty(ViewSplitter*)) , this , SIGNAL(empty()) );
78 79 80 81

    // listen for addition or removal of views from associated containers
    connect( _containerSignalMapper , SIGNAL(mapped(QObject*)) , this , 
            SLOT(containerViewsChanged(QObject*)) ); 
82 83

    // listen for profile changes
84 85
    connect( SessionManager::instance() , SIGNAL(profileChanged(Profile::Ptr)) , this,
            SLOT(profileChanged(Profile::Ptr)) );
86 87
    connect( SessionManager::instance() , SIGNAL(sessionUpdated(Session*)) , this,
            SLOT(updateViewsForSession(Session*)) );
88 89 90 91
}

ViewManager::~ViewManager()
{
92 93 94 95 96 97 98 99 100 101 102 103 104 105
    delete _newViewMenu;
}
QMenu* ViewManager::createNewViewMenu() 
{
    if (_newViewMenu)
        return _newViewMenu;

    _newViewMenu = new QMenu(0);
    ProfileList* newViewProfiles = new ProfileList(false,_newViewMenu);
    newViewProfiles->syncWidgetActions(_newViewMenu,true);
    connect(newViewProfiles,SIGNAL(profileSelected(Profile::Ptr)),this,
        SIGNAL(newViewRequest(Profile::Ptr)));

    return _newViewMenu;
106
}
107 108 109 110 111 112 113 114 115 116 117 118
QWidget* ViewManager::activeView() const
{
    ViewContainer* container = _viewSplitter->activeContainer();
    if ( container )
    {
        return container->activeView();
    }
    else
    {
        return 0;
    }
}
119

120 121 122 123 124
QWidget* ViewManager::widget() const
{
    return _viewSplitter;
}

125 126
void ViewManager::setupActions()
{
127
    KActionCollection* collection = _actionCollection;
128

129 130 131
    KAction* nextViewAction = new KAction( i18n("Next View") , this );
    KAction* previousViewAction = new KAction( i18n("Previous View") , this );
    QAction* nextContainerAction = new QAction( i18n("Next View Container") , this);
132 133 134 135
  
    QAction* moveViewLeftAction = new QAction( i18n("Move View Left") , this );
    QAction* moveViewRightAction = new QAction( i18n("Move View Right") , this );

136 137 138 139 140
    // list of actions that should only be enabled when there are multiple view
    // containers open
    QList<QAction*> multiViewOnlyActions;
    multiViewOnlyActions << nextContainerAction;

141 142
    if ( collection )
    {
143
        KAction* splitLeftRightAction = new KAction( KIcon("view-split-left-right"),
Thomas Reitelbach's avatar
Thomas Reitelbach committed
144
                                                      i18nc("@action:inmenu", "Split View Left/Right"),
145 146 147 148 149
                                                      this );
        splitLeftRightAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_L) );
        collection->addAction("split-view-left-right",splitLeftRightAction);
        connect( splitLeftRightAction , SIGNAL(triggered()) , this , SLOT(splitLeftRight()) );

150
        KAction* splitTopBottomAction = new KAction( KIcon("view-split-top-bottom") , 
Thomas Reitelbach's avatar
Thomas Reitelbach committed
151
                                             i18nc("@action:inmenu", "Split View Top/Bottom"),this);
152 153 154 155
        splitTopBottomAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_T) );
        collection->addAction("split-view-top-bottom",splitTopBottomAction);
        connect( splitTopBottomAction , SIGNAL(triggered()) , this , SLOT(splitTopBottom()));

Thomas Reitelbach's avatar
Thomas Reitelbach committed
156
        KAction* closeActiveAction = new KAction( i18nc("@action:inmenu Close Active View", "Close Active") , this );
Pino Toscano's avatar
Pino Toscano committed
157
        closeActiveAction->setIcon(KIcon("view-close"));
158
        closeActiveAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_S) );
159
        closeActiveAction->setEnabled(false);
160 161
        collection->addAction("close-active-view",closeActiveAction);
        connect( closeActiveAction , SIGNAL(triggered()) , this , SLOT(closeActiveView()) );
162 163 164
      
        multiViewOnlyActions << closeActiveAction; 

Thomas Reitelbach's avatar
Thomas Reitelbach committed
165
        KAction* closeOtherAction = new KAction( i18nc("@action:inmenu Close Other Views", "Close Others") , this );
166
        closeOtherAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_O) );
167
        closeOtherAction->setEnabled(false);
168 169
        collection->addAction("close-other-views",closeOtherAction);
        connect( closeOtherAction , SIGNAL(triggered()) , this , SLOT(closeOtherViews()) );
170 171

        multiViewOnlyActions << closeOtherAction;
172

173
        KAction* detachViewAction = collection->addAction("detach-view");
Pino Toscano's avatar
Pino Toscano committed
174
        detachViewAction->setIcon( KIcon("tab-detach") );
175 176 177 178
        detachViewAction->setText( i18n("&Detach View") );
        // Ctrl+Shift+D is not used as a shortcut by default because it is too close
        // to Ctrl+D - which will terminate the session in many cases
        detachViewAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_H) );
179 180
       
  		connect( this , SIGNAL(splitViewToggle(bool)) , this , SLOT(updateDetachViewState()) ); 
181
        connect( detachViewAction , SIGNAL(triggered()) , this , SLOT(detachActiveView()) );
182 183
   
        // Expand & Shrink Active View
Thomas Reitelbach's avatar
Thomas Reitelbach committed
184
        KAction* expandActiveAction = new KAction( i18nc("@action:inmenu", "Expand View") , this );
185 186 187 188
        expandActiveAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_BracketRight) );
        collection->addAction("expand-active-view",expandActiveAction);
        connect( expandActiveAction , SIGNAL(triggered()) , this , SLOT(expandActiveView()) );

189 190
        multiViewOnlyActions << expandActiveAction;

Thomas Reitelbach's avatar
Thomas Reitelbach committed
191
        KAction* shrinkActiveAction = new KAction( i18nc("@action:inmenu", "Shrink View") , this );
192 193 194 195
        shrinkActiveAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_BracketLeft) );
        collection->addAction("shrink-active-view",shrinkActiveAction);
        connect( shrinkActiveAction , SIGNAL(triggered()) , this , SLOT(shrinkActiveView()) );

196 197
        multiViewOnlyActions << shrinkActiveAction;

198
        // Next / Previous View , Next Container
199 200 201
        collection->addAction("next-view",nextViewAction);
        collection->addAction("previous-view",previousViewAction);
        collection->addAction("next-container",nextContainerAction);
202 203
        collection->addAction("move-view-left",moveViewLeftAction);
        collection->addAction("move-view-right",moveViewRightAction);
204
    }
Robert Knight's avatar
 
Robert Knight committed
205

206 207 208 209 210 211
    QListIterator<QAction*> iter(multiViewOnlyActions);
    while ( iter.hasNext() )
    {
        connect( this , SIGNAL(splitViewToggle(bool)) , iter.next() , SLOT(setEnabled(bool)) );
    }

212
    // keyboard shortcut only actions
213
    KShortcut nextViewShortcut = nextViewAction->shortcut();
214 215
    nextViewShortcut.setPrimary( QKeySequence(Qt::SHIFT+Qt::Key_Right) );
    nextViewShortcut.setAlternate( QKeySequence(Qt::CTRL+Qt::Key_PageUp) );
216
    nextViewAction->setShortcut(nextViewShortcut); 
Robert Knight's avatar
 
Robert Knight committed
217
    connect( nextViewAction, SIGNAL(triggered()) , this , SLOT(nextView()) );
218
    _viewSplitter->addAction(nextViewAction);
Robert Knight's avatar
 
Robert Knight committed
219

220 221 222 223
    KShortcut previousViewShortcut = previousViewAction->shortcut();
    previousViewShortcut.setPrimary( QKeySequence(Qt::SHIFT+Qt::Key_Left) );
    previousViewShortcut.setAlternate( QKeySequence(Qt::CTRL+Qt::Key_PageDown) );
    previousViewAction->setShortcut(previousViewShortcut);
Robert Knight's avatar
 
Robert Knight committed
224
    connect( previousViewAction, SIGNAL(triggered()) , this , SLOT(previousView()) );
225
    _viewSplitter->addAction(previousViewAction);
Robert Knight's avatar
 
Robert Knight committed
226 227 228

    nextContainerAction->setShortcut( QKeySequence(Qt::SHIFT+Qt::Key_Tab) );
    connect( nextContainerAction , SIGNAL(triggered()) , this , SLOT(nextContainer()) );
229
    _viewSplitter->addAction(nextContainerAction);
230 231 232 233 234 235 236

    moveViewLeftAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_Left) );
    connect( moveViewLeftAction , SIGNAL(triggered()) , this , SLOT(moveActiveViewLeft()) );
    _viewSplitter->addAction(moveViewLeftAction);
    moveViewRightAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_Right) );
    connect( moveViewRightAction , SIGNAL(triggered()) , this , SLOT(moveActiveViewRight()) );
    _viewSplitter->addAction(moveViewRightAction);
Robert Knight's avatar
 
Robert Knight committed
237 238
}

239 240 241 242 243
void ViewManager::updateDetachViewState()
{
	if (!_actionCollection)
		return;

244

245 246 247 248 249 250 251 252
	bool splitView = _viewSplitter->containers().count() >= 2;
	bool shouldEnable = splitView || _viewSplitter->activeContainer()->views().count() >= 2;

	QAction* detachAction = _actionCollection->action("detach-view");

	if ( detachAction && shouldEnable != detachAction->isEnabled() )
		detachAction->setEnabled(shouldEnable);
}
253 254 255 256 257 258 259 260 261 262 263 264
void ViewManager::moveActiveViewLeft()
{
    ViewContainer* container = _viewSplitter->activeContainer();
    Q_ASSERT( container );
    container->moveActiveView( ViewContainer::MoveViewLeft );
}
void ViewManager::moveActiveViewRight()
{
    ViewContainer* container = _viewSplitter->activeContainer();
    Q_ASSERT( container );
    container->moveActiveView( ViewContainer::MoveViewRight );
}
Robert Knight's avatar
 
Robert Knight committed
265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285
void ViewManager::nextContainer()
{
    _viewSplitter->activateNextContainer();
}

void ViewManager::nextView()
{
    ViewContainer* container = _viewSplitter->activeContainer();

    Q_ASSERT( container );

    container->activateNextView();
}

void ViewManager::previousView()
{
    ViewContainer* container = _viewSplitter->activeContainer();

    Q_ASSERT( container );

    container->activatePreviousView();
286 287 288 289 290
}
void ViewManager::detachActiveView()
{
    // find the currently active view and remove it from its container 
    ViewContainer* container = _viewSplitter->activeContainer();
291
    TerminalDisplay* activeView = dynamic_cast<TerminalDisplay*>(container->activeView());
292 293 294 295 296 297 298 299 300 301

    if (!activeView)
        return;

    emit viewDetached(_sessionMap[activeView]);
    
    _sessionMap.remove(activeView);

    // remove the view from this window
    container->removeView(activeView);
302
    activeView->deleteLater();
303 304 305 306 307 308 309

    // if the container from which the view was removed is now empty then it can be deleted,
    // unless it is the only container in the window, in which case it is left empty
    // so that there is always an active container
    if ( _viewSplitter->containers().count() > 1 && 
         container->views().count() == 0 )
    {
310
        removeContainer(container);
311 312 313 314
    }

}

315
void ViewManager::sessionFinished()
316
{
317 318 319 320 321
	// if this slot is called after the view manager's main widget
	// has been destroyed, do nothing
	if (!_viewSplitter)
		return;

322 323
    Session* session = qobject_cast<Session*>(sender());

324 325 326 327 328 329 330 331
    if ( _sessionMap[qobject_cast<TerminalDisplay*>(activeView())] == session )
    {
        // switch to the previous view before deleting the session views to prevent flicker 
        // occurring as a result of an interval between removing the active view and switching
        // to the previous view
        previousView();
    }

332 333
    Q_ASSERT(session);

334
    // close attached views
335
    QList<TerminalDisplay*> children = _viewSplitter->findChildren<TerminalDisplay*>();
336

337
    foreach ( TerminalDisplay* view , children )
338 339 340 341
    {
        if ( _sessionMap[view] == session )
        {
            _sessionMap.remove(view);
342
            view->deleteLater();
343
        }
Robert Knight's avatar
 
Robert Knight committed
344
    }
345

Robert Knight's avatar
 
Robert Knight committed
346 347 348 349
}

void ViewManager::focusActiveView()
{
Robert Knight's avatar
Robert Knight committed
350 351 352 353 354 355
    // give the active view in a container the focus.  this ensures 
    // that controller associated with that view is activated and the session-specific
    // menu items are replaced with the ones for the newly focused view

    // see the viewFocused() method

Robert Knight's avatar
 
Robert Knight committed
356 357 358 359 360 361 362 363 364
    ViewContainer* container = _viewSplitter->activeContainer(); 
    if ( container )
    {
        QWidget* activeView = container->activeView();
        if ( activeView )
        {
            activeView->setFocus(Qt::MouseFocusReason);
        }
    }
365 366
}

367

368 369
void ViewManager::viewActivated( QWidget* view )
{
370 371 372 373 374 375
    Q_ASSERT( view != 0 );

    // focus the activated view, this will cause the SessionController
    // to notify the world that the view has been focused and the appropriate UI
    // actions will be plugged in.
    view->setFocus(Qt::OtherFocusReason);
376 377
}

378
void ViewManager::splitLeftRight()
379
{
380
    splitView(Qt::Horizontal);
381
}
382
void ViewManager::splitTopBottom()
383
{
384
    splitView(Qt::Vertical);
385 386
}

387
void ViewManager::splitView(Qt::Orientation orientation)
388
{
389 390 391 392
    // iterate over each session which has a view in the current active
    // container and create a new view for that session in a new container 
    QListIterator<QWidget*> existingViewIter(_viewSplitter->activeContainer()->views());
    
393
    ViewContainer* container = 0; 
394 395

    while (existingViewIter.hasNext())
396
    {
397
        Session* session = _sessionMap[(TerminalDisplay*)existingViewIter.next()];
398
        TerminalDisplay* display = createTerminalDisplay(session);
399
        applyProfile(display,SessionManager::instance()->sessionProfile(session),false);
400
        ViewProperties* properties = createController(session,display);
401

402
        _sessionMap[display] = session;
403

404 405 406
        // create a container using settings from the first 
        // session in the previous container
        if ( !container )
407
            container = createContainer(SessionManager::instance()->sessionProfile(session));
408

409 410 411
        container->addView(display,properties);
        session->addView( display );
    }
412

413 414
    _viewSplitter->addContainer(container,orientation);
    emit splitViewToggle(_viewSplitter->containers().count() > 0);
415

416 417
    // focus the new container
    container->containerWidget()->setFocus();
418 419

    // ensure that the active view is focused after the split / unsplit
420 421 422 423 424
    ViewContainer* activeContainer = _viewSplitter->activeContainer();
    QWidget* activeView = activeContainer ? activeContainer->activeView() : 0;

    if ( activeView )
        activeView->setFocus(Qt::OtherFocusReason);
425
}
426
void ViewManager::removeContainer(ViewContainer* container)
427
{
428 429 430 431 432 433 434 435
    // remove session map entries for views in this container
    foreach( QWidget* view , container->views() )
    {
        TerminalDisplay* display = qobject_cast<TerminalDisplay*>(view);
        Q_ASSERT(display);
        _sessionMap.remove(display);
    } 

436
	_viewSplitter->removeContainer(container);
437
    container->deleteLater();
438 439

    emit splitViewToggle( _viewSplitter->containers().count() > 1 );
440
}
441 442 443 444 445 446 447 448
void ViewManager::expandActiveView()
{
    _viewSplitter->adjustContainerSize(_viewSplitter->activeContainer(),10);
}
void ViewManager::shrinkActiveView()
{
    _viewSplitter->adjustContainerSize(_viewSplitter->activeContainer(),-10);
}
449 450 451 452 453 454 455 456
void ViewManager::closeActiveView()
{
    // only do something if there is more than one container active
    if ( _viewSplitter->containers().count() > 1 )
    {
        ViewContainer* container = _viewSplitter->activeContainer();

        removeContainer(container);
457

458 459 460 461
        // focus next container so that user can continue typing 
        // without having to manually focus it themselves
        nextContainer();
    }
462 463 464 465 466 467 468 469 470 471
}
void ViewManager::closeOtherViews()
{
    ViewContainer* active = _viewSplitter->activeContainer();

    QListIterator<ViewContainer*> iter(_viewSplitter->containers());
    while ( iter.hasNext() )
    {
        ViewContainer* next = iter.next();
        if ( next != active )
472
            removeContainer(next);
473 474 475
    }
}

476
SessionController* ViewManager::createController(Session* session , TerminalDisplay* view)
477
{
Robert Knight's avatar
Robert Knight committed
478 479
    // create a new controller for the session, and ensure that this view manager
    // is notified when the view gains the focus
480
    SessionController* controller = new SessionController(session,view,this);
481
    connect( controller , SIGNAL(focused(SessionController*)) , this , SLOT(controllerChanged(SessionController*)) );
482 483
    connect( session , SIGNAL(destroyed()) , controller , SLOT(deleteLater()) );
    connect( view , SIGNAL(destroyed()) , controller , SLOT(deleteLater()) );
484

485 486
	// if this is the first controller created then set it as the active controller
	if (!_pluggedController)
487
		controllerChanged(controller);
488

489 490 491
    return controller;
}

492 493
void ViewManager::controllerChanged(SessionController* controller)
{
494 495 496
	if ( controller == _pluggedController )
		return;

497 498
	_viewSplitter->setFocusProxy(controller->view());

499 500 501 502 503 504 505 506
	_pluggedController = controller;
	emit activeViewChanged(controller);
}

SessionController* ViewManager::activeViewController() const
{
	return _pluggedController;
}
507 508 509 510 511 512 513 514 515
void ViewManager::createView(Session* session, ViewContainer* container, int index)
{
	// notify this view manager when the session finishes so that its view
    // can be deleted
	//
	// TODO - Find a more efficient a way to avoid multiple connections
	disconnect( session , SIGNAL(finished()) , this , SLOT(sessionFinished()) );
    connect( session , SIGNAL(finished()) , this , SLOT(sessionFinished()) );

516
     bool isFirst = _sessionMap.isEmpty();
517
     TerminalDisplay* display = createTerminalDisplay(session);
518
     applyProfile(display,SessionManager::instance()->sessionProfile(session),isFirst);
519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540
     
     // set initial size
     display->setSize(80,40);

     ViewProperties* properties = createController(session,display);

     _sessionMap[display] = session; 
     container->addView(display,properties,index);
     session->addView(display);

     // tell the session whether it has a light or dark background
 	const Profile::Ptr profile = SessionManager::instance()->sessionProfile(session);
     session->setDarkBackground( colorSchemeForProfile(profile)->hasDarkBackground() );

     if ( container == _viewSplitter->activeContainer() ) 
     {
         container->setActiveView(display);
         display->setFocus( Qt::OtherFocusReason );
     }
	
	 updateDetachViewState();
}
541

542
void ViewManager::createView(Session* session)
543
{
544 545 546
    // create the default container
    if (_viewSplitter->containers().count() == 0)
    {
547
        _viewSplitter->addContainer( createContainer(SessionManager::instance()->sessionProfile(session)) , 
548
                                     Qt::Vertical );
549
        emit splitViewToggle(false);
550 551
    }

552
       
Robert Knight's avatar
Robert Knight committed
553 554 555
    // iterate over the view containers owned by this view manager
    // and create a new terminal display for the session in each of them, along with
    // a controller for the session/display pair 
556 557 558 559 560
    QListIterator<ViewContainer*> containerIter(_viewSplitter->containers());

    while ( containerIter.hasNext() )
    {
        ViewContainer* container = containerIter.next();
561
		createView(session,container,-1);
562
    }
563

564 565
}

566
ViewContainer* ViewManager::createContainer(const Profile::Ptr info)
567
{
568 569
    Q_ASSERT( info );

570
    const int tabPosition = info->property<int>(Profile::TabBarPosition);
571 572 573 574 575

    ViewContainer::NavigationPosition position = ( tabPosition == Profile::TabBarTop ) ?
                                                   ViewContainer::NavigationPositionTop :
                                                   ViewContainer::NavigationPositionBottom;

576 577 578 579 580 581 582 583 584 585 586
    ViewContainer* container = 0;

    switch ( _navigationMethod )
    {
        case TabbedNavigation:    
            container = new TabbedViewContainerV2(position,_viewSplitter);
            break;
        case NoNavigation:
        default:
            container = new StackedViewContainer(_viewSplitter);
    }
587

588
    // connect signals and slots
589 590 591 592 593 594
    connect( container , SIGNAL(viewAdded(QWidget*,ViewProperties*)) , _containerSignalMapper ,
           SLOT(map()) );
    connect( container , SIGNAL(viewRemoved(QWidget*)) , _containerSignalMapper ,
           SLOT(map()) ); 
    _containerSignalMapper->setMapping(container,container);

595
    connect( container, SIGNAL(newViewRequest()), this, SIGNAL(newViewRequest()) );
596 597
    connect( container, SIGNAL(moveViewRequest(int,int,bool&)), 
	this , SLOT(containerMoveViewRequest(int,int,bool&)) );
598
    connect( container , SIGNAL(viewRemoved(QWidget*)) , this , SLOT(viewCloseRequest(QWidget*)) );
599
    connect( container , SIGNAL(closeRequest(QWidget*)) , this , SLOT(viewCloseRequest(QWidget*)) );
600
    connect( container , SIGNAL(activeViewChanged(QWidget*)) , this , SLOT(viewActivated(QWidget*)));
601
    
602
    return container;
603
}
604 605 606 607
void ViewManager::containerMoveViewRequest(int index, int id, bool& moved)
{
	ViewContainer* container = qobject_cast<ViewContainer*>(sender());
	SessionController* controller = qobject_cast<SessionController*>(ViewProperties::propertiesById(id));
608

609 610 611 612 613 614
	if (!controller)
		return;

	createView(controller->session(),container,index);
	moved = true;
}
615 616 617 618 619 620 621 622 623 624 625 626 627 628 629
void ViewManager::setNavigationMethod(NavigationMethod method)
{
    _navigationMethod = method;

    KActionCollection* collection = _actionCollection;

    if ( collection )
    {
        QAction* action;

        action = collection->action( "next-view" );
        if ( action ) action->setEnabled( _navigationMethod != NoNavigation );

        action = collection->action( "previous-view" );
        if ( action ) action->setEnabled( _navigationMethod != NoNavigation );
630 631 632 633 634 635

        action = collection->action( "split-view-left-right" );
        if ( action ) action->setEnabled( _navigationMethod != NoNavigation );

        action = collection->action( "split-view-top-bottom" );
        if ( action ) action->setEnabled( _navigationMethod != NoNavigation );
636 637 638

        action = collection->action( "rename-session" );
        if ( action ) action->setEnabled( _navigationMethod != NoNavigation );
639 640 641
    }
}

642 643
ViewManager::NavigationMethod ViewManager::navigationMethod() const { return _navigationMethod; }

644 645 646 647 648 649 650 651
void ViewManager::containerViewsChanged(QObject* container)
{
    if ( container == _viewSplitter->activeContainer() )
    {
        emit viewPropertiesChanged( viewProperties() );
    } 
}

652 653
void ViewManager::viewCloseRequest(QWidget* view)
{
654 655 656 657 658
    //FIXME Check that this cast is actually legal
    TerminalDisplay* display = (TerminalDisplay*)view;
  
    Q_ASSERT(display);

659 660
    // 1. detach view from session
    // 2. if the session has no views left, close it
661
    Session* session = _sessionMap[ display ];
662
    _sessionMap.remove(display);
663 664
    if ( session )
    {
665
        display->deleteLater();
666

667
        if ( session->views().count() == 0 )
668
            session->close();
669
    }
670
        
671
    focusActiveView();
672
	updateDetachViewState();
673 674
}

675
TerminalDisplay* ViewManager::createTerminalDisplay(Session* session)
676
{
677
   TerminalDisplay* display = new TerminalDisplay(0);
678 679

   //TODO Temporary settings used here
680 681
   display->setBellMode(TerminalDisplay::NotifyBell);
   display->setTerminalSizeHint(true);
682
   display->setTripleClickMode(TerminalDisplay::SelectWholeLine);
683
   display->setTerminalSizeStartup(true);
684
   display->setScrollBarPosition(TerminalDisplay::ScrollBarRight);
685 686
   display->setRandomSeed(session->sessionId() * 31);

687 688 689
   return display;
}

690
const ColorScheme* ViewManager::colorSchemeForProfile(const Profile::Ptr info) const
691
{
692 693
    const ColorScheme* colorScheme = ColorSchemeManager::instance()->
                                            findColorScheme(info->colorScheme());
694 695
    if ( !colorScheme )
       colorScheme = ColorSchemeManager::instance()->defaultColorScheme(); 
696 697
    Q_ASSERT( colorScheme );

698 699 700
    return colorScheme;
}

701 702
void ViewManager::applyProfile(TerminalDisplay* view , const Profile::Ptr info, 
                               bool applyContainerSettings) 
703 704 705
{
    Q_ASSERT( info );
    
706
    const ColorScheme* colorScheme = colorSchemeForProfile(info);
707

708
    // menu bar visibility
709
    emit setMenuBarVisibleRequest( info->property<bool>(Profile::ShowMenuBar) );
710 711

    // tab bar visibility
712 713 714 715 716
    if (applyContainerSettings)
    {
        ViewContainer* container = _viewSplitter->activeContainer();
        int tabBarMode = info->property<int>(Profile::TabBarMode);
        int tabBarPosition = info->property<int>(Profile::TabBarPosition);
Robert Knight's avatar
 
Robert Knight committed
717
        bool showNewCloseButtons = info->property<bool>(Profile::ShowNewAndCloseTabButtons);
718

719 720 721 722 723 724
        if ( tabBarMode == Profile::AlwaysHideTabBar )
            container->setNavigationDisplayMode(ViewContainer::AlwaysHideNavigation);
        else if ( tabBarMode == Profile::AlwaysShowTabBar )
            container->setNavigationDisplayMode(ViewContainer::AlwaysShowNavigation);
        else if ( tabBarMode == Profile::ShowTabBarAsNeeded )
            container->setNavigationDisplayMode(ViewContainer::ShowNavigationAsNeeded);
725

726
        ViewContainer::NavigationPosition position = container->navigationPosition();
727

728 729 730 731
        if ( tabBarPosition == Profile::TabBarTop )
            position = ViewContainer::NavigationPositionTop;
        else if ( tabBarPosition == Profile::TabBarBottom )
            position = ViewContainer::NavigationPositionBottom; 
732

733 734
        if ( container->supportedNavigationPositions().contains(position) )
            container->setNavigationPosition(position);
735
       
Robert Knight's avatar
 
Robert Knight committed
736
        if (showNewCloseButtons)
737
        {
Robert Knight's avatar
 
Robert Knight committed
738 739
            container->setFeatures(container->features() 
                               | ViewContainer::QuickNewView | ViewContainer::QuickCloseView);
740 741 742
            container->setNewViewMenu(createNewViewMenu());
        }
        else
Robert Knight's avatar
 
Robert Knight committed
743 744
            container->setFeatures(container->features() 
                            & ~ViewContainer::QuickNewView & ~ViewContainer::QuickCloseView);
745
    }
746

747
    // load colour scheme
748 749 750 751
    ColorEntry table[TABLE_COLORS];
    
    colorScheme->getColorTable(table , view->randomSeed() );
    view->setColorTable(table);
752
    view->setOpacity(colorScheme->opacity());
753
  
754
    // load font 
755
    view->setAntialias(info->property<bool>(Profile::AntiAliasFonts));
756
    view->setVTFont(info->font());
757 758

    // set scroll-bar position
759
    int scrollBarPosition = info->property<int>(Profile::ScrollBarPosition);
760 761

    if ( scrollBarPosition == Profile::ScrollBarHidden )
762
       view->setScrollBarPosition(TerminalDisplay::NoScrollBar);
763
    else if ( scrollBarPosition == Profile::ScrollBarLeft )
764
       view->setScrollBarPosition(TerminalDisplay::ScrollBarLeft);
765
    else if ( scrollBarPosition == Profile::ScrollBarRight )
766
       view->setScrollBarPosition(TerminalDisplay::ScrollBarRight);
767 768

    // terminal features
769
    bool blinkingCursor = info->property<bool>(Profile::BlinkingCursorEnabled);
770 771
    view->setBlinkingCursor(blinkingCursor);  

772 773 774
    bool bidiEnabled = info->property<bool>(Profile::BidiRenderingEnabled);
    view->setBidiEnabled(bidiEnabled);

775
    // cursor shape
776
    int cursorShape = info->property<int>(Profile::CursorShape);
777 778 779 780 781 782 783 784 785

    if ( cursorShape == Profile::BlockCursor )
        view->setKeyboardCursorShape(TerminalDisplay::BlockCursor);  
    else if ( cursorShape == Profile::IBeamCursor )
        view->setKeyboardCursorShape(TerminalDisplay::IBeamCursor);
    else if ( cursorShape == Profile::UnderlineCursor )
        view->setKeyboardCursorShape(TerminalDisplay::UnderlineCursor);

    // cursor color
786 787
    bool useCustomColor = info->property<bool>(Profile::UseCustomCursorColor);
    const QColor& cursorColor = info->property<QColor>(Profile::CustomCursorColor);
788 789 790 791
        
    view->setKeyboardCursorColor(!useCustomColor,cursorColor);

    // word characters
792
    view->setWordCharacters( info->property<QString>(Profile::WordCharacters) );
793 794
}

795 796
void ViewManager::updateViewsForSession(Session* session)
{
797
    QListIterator<TerminalDisplay*> iter(_sessionMap.keys(session));
798 799
    while ( iter.hasNext() )
    {
800
        applyProfile(iter.next(),SessionManager::instance()->sessionProfile(session),false);
801 802 803
    }
}

804
void ViewManager::profileChanged(Profile::Ptr profile)
805
{
806
    QHashIterator<TerminalDisplay*,Session*> iter(_sessionMap);
807 808 809 810 811 812

    while ( iter.hasNext() )
    {
        iter.next();

        // if session uses this profile, update the display
813 814 815
        if ( iter.key() != 0 && 
			 iter.value() != 0 && 
			 SessionManager::instance()->sessionProfile(iter.value()) == profile ) 
816
        {
817
            applyProfile(iter.key(),profile,true);
818 819
        }
    }
820 821 822 823 824 825 826 827 828 829 830 831 832
}

QList<ViewProperties*> ViewManager::viewProperties() const
{
    QList<ViewProperties*> list;

    ViewContainer* container = _viewSplitter->activeContainer();

    Q_ASSERT( container );

    QListIterator<QWidget*> viewIter(container->views());
    while ( viewIter.hasNext() )
    {
Robert Knight's avatar
Robert Knight committed
833 834
        ViewProperties* properties = container->viewProperties(viewIter.next()); 
        Q_ASSERT( properties );
835 836
        list << properties; 
    } 
837

838
    return list;
839 840
}

841 842 843 844 845
uint qHash(QPointer<TerminalDisplay> display)
{
    return qHash((TerminalDisplay*)display);
}

846
#include "ViewManager.moc"