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

    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
Dirk Mueller's avatar
Dirk Mueller committed
27
#include <QtCore/QSignalMapper>
28

29 30 31 32 33 34 35
// KDE
#include <kdebug.h>
#include <KLocale>
#include <KToggleAction>
#include <KXMLGUIFactory>

// Konsole
36
#include "BookmarkHandler.h"
37
#include "ColorScheme.h"
38
#include "Session.h"
39
#include "TerminalDisplay.h"
40 41 42 43 44
#include "SessionController.h"
#include "SessionManager.h"
#include "ViewContainer.h"
#include "ViewSplitter.h"

45 46
using namespace Konsole;

47 48
ViewManager::ViewManager(QObject* parent , KActionCollection* collection)
    : QObject(parent)
49
    , _viewSplitter(0)
50 51
    , _actionCollection(collection)
    , _containerSignalMapper(new QSignalMapper(this))
52
{
53 54 55 56 57 58 59 60 61 62
    // create main view area
    _viewSplitter = new ViewSplitter(0);   
    // 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);

63 64 65 66 67 68
    // 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()) );
69 70 71 72

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

    // listen for profile changes
    connect( SessionManager::instance() , SIGNAL(profileChanged(const QString&)) , this,
            SLOT(profileChanged(const QString&)) );
77 78
    connect( SessionManager::instance() , SIGNAL(sessionUpdated(Session*)) , this,
            SLOT(updateViewsForSession(Session*)) );
79 80 81 82 83
}

ViewManager::~ViewManager()
{
}
84 85 86 87 88 89 90 91 92 93 94 95
QWidget* ViewManager::activeView() const
{
    ViewContainer* container = _viewSplitter->activeContainer();
    if ( container )
    {
        return container->activeView();
    }
    else
    {
        return 0;
    }
}
96

97 98 99 100 101
QWidget* ViewManager::widget() const
{
    return _viewSplitter;
}

102 103
void ViewManager::setupActions()
{
104
    KActionCollection* collection = _actionCollection;
105

106 107 108
    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);
109 110 111 112 113 114
   
    // list of actions that should only be enabled when there are multiple view
    // containers open
    QList<QAction*> multiViewOnlyActions;
    multiViewOnlyActions << nextContainerAction;

115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130
    if ( collection )
    {
        KAction* splitLeftRightAction = new KAction( KIcon("view-left-right"),
                                                      i18n("Split View Left/Right"),
                                                      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()) );

        KAction* splitTopBottomAction = new KAction( KIcon("view-top-bottom") , 
                                             i18n("Split View Top/Bottom"),this);
        splitTopBottomAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_T) );
        collection->addAction("split-view-top-bottom",splitTopBottomAction);
        connect( splitTopBottomAction , SIGNAL(triggered()) , this , SLOT(splitTopBottom()));

        KAction* closeActiveAction = new KAction( i18n("Close Active") , this );
131
        closeActiveAction->setIcon(KIcon("view-remove"));
132
        closeActiveAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_S) );
133
        closeActiveAction->setEnabled(false);
134 135
        collection->addAction("close-active-view",closeActiveAction);
        connect( closeActiveAction , SIGNAL(triggered()) , this , SLOT(closeActiveView()) );
136 137 138
      
        multiViewOnlyActions << closeActiveAction; 

139 140
        KAction* closeOtherAction = new KAction( i18n("Close Others") , this );
        closeOtherAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_O) );
141
        closeOtherAction->setEnabled(false);
142 143
        collection->addAction("close-other-views",closeOtherAction);
        connect( closeOtherAction , SIGNAL(triggered()) , this , SLOT(closeOtherViews()) );
144 145

        multiViewOnlyActions << closeOtherAction;
146 147 148 149 150 151 152

        QAction* detachViewAction = collection->addAction("detach-view");
        detachViewAction->setIcon( KIcon("tab-breakoff") );
        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) );
153 154
        
        multiViewOnlyActions << detachViewAction;
155 156

        connect( detachViewAction , SIGNAL(triggered()) , this , SLOT(detachActiveView()) );
157 158 159 160 161 162 163
   
        // Expand & Shrink Active View
        KAction* expandActiveAction = new KAction( i18n("Expand View") , this );
        expandActiveAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_BracketRight) );
        collection->addAction("expand-active-view",expandActiveAction);
        connect( expandActiveAction , SIGNAL(triggered()) , this , SLOT(expandActiveView()) );

164 165
        multiViewOnlyActions << expandActiveAction;

166 167 168 169 170
        KAction* shrinkActiveAction = new KAction( i18n("Shrink View") , this );
        shrinkActiveAction->setShortcut( QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_BracketLeft) );
        collection->addAction("shrink-active-view",shrinkActiveAction);
        connect( shrinkActiveAction , SIGNAL(triggered()) , this , SLOT(shrinkActiveView()) );

171 172
        multiViewOnlyActions << shrinkActiveAction;

173
        // Next / Previous View , Next Container
174 175 176
        collection->addAction("next-view",nextViewAction);
        collection->addAction("previous-view",previousViewAction);
        collection->addAction("next-container",nextContainerAction);
Robert Knight's avatar
 
Robert Knight committed
177

178
    }
Robert Knight's avatar
 
Robert Knight committed
179

180 181 182 183 184 185
    QListIterator<QAction*> iter(multiViewOnlyActions);
    while ( iter.hasNext() )
    {
        connect( this , SIGNAL(splitViewToggle(bool)) , iter.next() , SLOT(setEnabled(bool)) );
    }

186
    KShortcut nextViewShortcut = nextViewAction->shortcut();
187 188
    nextViewShortcut.setPrimary( QKeySequence(Qt::SHIFT+Qt::Key_Right) );
    nextViewShortcut.setAlternate( QKeySequence(Qt::CTRL+Qt::Key_PageUp) );
189
    nextViewAction->setShortcut(nextViewShortcut); 
Robert Knight's avatar
 
Robert Knight committed
190
    connect( nextViewAction, SIGNAL(triggered()) , this , SLOT(nextView()) );
191
    _viewSplitter->addAction(nextViewAction);
Robert Knight's avatar
 
Robert Knight committed
192

193 194 195 196
    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
197
    connect( previousViewAction, SIGNAL(triggered()) , this , SLOT(previousView()) );
198
    _viewSplitter->addAction(previousViewAction);
Robert Knight's avatar
 
Robert Knight committed
199 200 201

    nextContainerAction->setShortcut( QKeySequence(Qt::SHIFT+Qt::Key_Tab) );
    connect( nextContainerAction , SIGNAL(triggered()) , this , SLOT(nextContainer()) );
202
    _viewSplitter->addAction(nextContainerAction);
Robert Knight's avatar
 
Robert Knight committed
203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225
}

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();
226 227 228 229 230
}
void ViewManager::detachActiveView()
{
    // find the currently active view and remove it from its container 
    ViewContainer* container = _viewSplitter->activeContainer();
231
    TerminalDisplay* activeView = dynamic_cast<TerminalDisplay*>(container->activeView());
232 233 234 235 236 237 238 239 240 241

    if (!activeView)
        return;

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

    // remove the view from this window
    container->removeView(activeView);
242
    activeView->deleteLater();
243 244 245 246 247 248 249

    // 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 )
    {
250
        removeContainer(container);
251 252 253 254
    }

}

255
void ViewManager::sessionFinished()
256
{
257 258
    Session* session = qobject_cast<Session*>(sender());

259 260 261 262 263 264 265 266
    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();
    }

267 268
    Q_ASSERT(session);

269
    QList<TerminalDisplay*> children = _viewSplitter->findChildren<TerminalDisplay*>();
270

271
    foreach ( TerminalDisplay* view , children )
272 273 274 275
    {
        if ( _sessionMap[view] == session )
        {
            _sessionMap.remove(view);
276
            view->deleteLater();
277
        }
Robert Knight's avatar
 
Robert Knight committed
278 279 280 281 282
    }
}

void ViewManager::focusActiveView()
{
Robert Knight's avatar
Robert Knight committed
283 284 285 286 287 288
    // 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
289 290 291 292 293 294 295 296 297
    ViewContainer* container = _viewSplitter->activeContainer(); 
    if ( container )
    {
        QWidget* activeView = container->activeView();
        if ( activeView )
        {
            activeView->setFocus(Qt::MouseFocusReason);
        }
    }
298 299
}

300

301 302
void ViewManager::viewActivated( QWidget* view )
{
303 304 305 306 307 308
    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);
309 310
}

311
void ViewManager::splitLeftRight()
312
{
313
    splitView(Qt::Horizontal);
314
}
315
void ViewManager::splitTopBottom()
316
{
317
    splitView(Qt::Vertical);
318 319
}

320
void ViewManager::splitView(Qt::Orientation orientation)
321
{
322 323 324 325
    // 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());
    
326
    ViewContainer* container = 0; 
327 328

    while (existingViewIter.hasNext())
329
    {
330 331
        Session* session = _sessionMap[(TerminalDisplay*)existingViewIter.next()];
        TerminalDisplay* display = createTerminalDisplay();
332
        applyProfile(display,session->profileKey()); 
333
        ViewProperties* properties = createController(session,display);
334

335
        _sessionMap[display] = session;
336

337 338 339 340 341
        // create a container using settings from the first 
        // session in the previous container
        if ( !container )
            container = createContainer(session->profileKey());

342 343 344
        container->addView(display,properties);
        session->addView( display );
    }
345

346 347
    _viewSplitter->addContainer(container,orientation);
    emit splitViewToggle(_viewSplitter->containers().count() > 0);
348

349 350
    // focus the new container
    container->containerWidget()->setFocus();
351 352

    // ensure that the active view is focused after the split / unsplit
353 354 355 356 357
    ViewContainer* activeContainer = _viewSplitter->activeContainer();
    QWidget* activeView = activeContainer ? activeContainer->activeView() : 0;

    if ( activeView )
        activeView->setFocus(Qt::OtherFocusReason);
358
}
359
void ViewManager::removeContainer(ViewContainer* container)
360
{
361 362 363 364 365
    // note that the _viewSplitter->containers().count() will not be updated
    // until the container is deleted when Qt returns to the main event loop,
    // so we take the previous container count and work out what the new count
    // would be.
    int previousCount = _viewSplitter->containers().count();
366
    container->deleteLater();
367
    emit splitViewToggle( (previousCount-1) > 1);
368
}
369 370 371 372 373 374 375 376
void ViewManager::expandActiveView()
{
    _viewSplitter->adjustContainerSize(_viewSplitter->activeContainer(),10);
}
void ViewManager::shrinkActiveView()
{
    _viewSplitter->adjustContainerSize(_viewSplitter->activeContainer(),-10);
}
377 378 379 380 381 382 383 384
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);
385

386 387 388 389
        // focus next container so that user can continue typing 
        // without having to manually focus it themselves
        nextContainer();
    }
390 391 392 393 394 395 396 397 398 399 400 401 402 403
}
void ViewManager::closeOtherViews()
{
    ViewContainer* active = _viewSplitter->activeContainer();

    QListIterator<ViewContainer*> iter(_viewSplitter->containers());
    while ( iter.hasNext() )
    {
        ViewContainer* next = iter.next();
        if ( next != active )
            delete next;
    }
}

404
SessionController* ViewManager::createController(Session* session , TerminalDisplay* view)
405
{
Robert Knight's avatar
Robert Knight committed
406 407
    // create a new controller for the session, and ensure that this view manager
    // is notified when the view gains the focus
408
    SessionController* controller = new SessionController(session,view,this);
409
    connect( controller , SIGNAL(focused(SessionController*)) , this , SIGNAL(activeViewChanged(SessionController*)) );
410 411
    connect( session , SIGNAL(destroyed()) , controller , SLOT(deleteLater()) );
    connect( view , SIGNAL(destroyed()) , controller , SLOT(deleteLater()) );
412 413
    connect( controller , SIGNAL(sendInputToAll(bool)) , this , SLOT(sendInputToAll()) );

414 415 416
    return controller;
}

417
void ViewManager::createView(Session* session)
418
{
419 420 421
    // create the default container
    if (_viewSplitter->containers().count() == 0)
    {
422 423
        _viewSplitter->addContainer( createContainer(session->profileKey()) , 
                                     Qt::Vertical );
424
        emit splitViewToggle(false);
425 426
    }

Robert Knight's avatar
Robert Knight committed
427 428
    // notify this view manager when the session finishes so that its view
    // can be deleted
429
    connect( session , SIGNAL(finished()) , this , SLOT(sessionFinished()) );
Robert Knight's avatar
Robert Knight committed
430 431 432 433
   
    // 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 
434 435 436 437 438 439
    ViewContainer* const activeContainer = _viewSplitter->activeContainer();
    QListIterator<ViewContainer*> containerIter(_viewSplitter->containers());

    while ( containerIter.hasNext() )
    {
        ViewContainer* container = containerIter.next();
440
        TerminalDisplay* display = createTerminalDisplay();
441
        applyProfile(display,session->profileKey());
442 443 444 445 446
        
        // set initial size
        // temporary default used for now
        display->setSize(80,40);
        
Robert Knight's avatar
 
Robert Knight committed
447
        ViewProperties* properties = createController(session,display);
448 449

        _sessionMap[display] = session; 
Robert Knight's avatar
 
Robert Knight committed
450
        container->addView(display,properties);
451 452
        session->addView(display);

453 454
        if ( container == activeContainer ) 
        {
455
            container->setActiveView(display);
456 457
            display->setFocus( Qt::OtherFocusReason );
        }
458 459 460
    }
}

461
ViewContainer* ViewManager::createContainer(const QString& profileKey)
462
{
463 464 465 466 467 468 469 470 471 472 473
    const Profile* info = SessionManager::instance()->profile(profileKey);

    Q_ASSERT( info );

    const int tabPosition = info->property(Profile::TabBarPosition).value<int>();

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

    ViewContainer* container = new TabbedViewContainerV2(position,_viewSplitter);
474

475
    // connect signals and slots
476 477 478 479 480 481
    connect( container , SIGNAL(viewAdded(QWidget*,ViewProperties*)) , _containerSignalMapper ,
           SLOT(map()) );
    connect( container , SIGNAL(viewRemoved(QWidget*)) , _containerSignalMapper ,
           SLOT(map()) ); 
    _containerSignalMapper->setMapping(container,container);

482
    connect( container , SIGNAL(viewRemoved(QWidget*)) , this , SLOT(viewCloseRequest(QWidget*)) );
483
    connect( container , SIGNAL(closeRequest(QWidget*)) , this , SLOT(viewCloseRequest(QWidget*)) );
484
    connect( container , SIGNAL(activeViewChanged(QWidget*)) , this , SLOT(viewActivated(QWidget*)));
485
    
486
    return container;
487 488
}

489 490 491 492 493 494 495 496 497 498
void ViewManager::containerViewsChanged(QObject* container)
{
    qDebug() << "Container views changed";

    if ( container == _viewSplitter->activeContainer() )
    {
        emit viewPropertiesChanged( viewProperties() );
    } 
}

499 500 501 502
void ViewManager::viewCloseRequest(QWidget* view)
{
    // 1. detach view from session
    // 2. if the session has no views left, close it
503 504 505
   
    qDebug() << "Removing view.";

506
    TerminalDisplay* display = (TerminalDisplay*)view;
507
    Session* session = _sessionMap[ display ];
508 509
    if ( session )
    {
510
        display->deleteLater();
511 512 513

        _sessionMap.remove(display);

514
        if ( session->views().count() == 0 )
515
            session->close();
516
    }
517
        
518
    focusActiveView();
519 520
}

521
TerminalDisplay* ViewManager::createTerminalDisplay()
522
{
523
   TerminalDisplay* display = new TerminalDisplay(0);
524 525 526 527 528 529

   //TODO Temporary settings used here
   display->setBellMode(0);
   display->setTerminalSizeHint(false);
   display->setCutToBeginningOfLine(true);
   display->setTerminalSizeStartup(false);
530
   display->setScrollBarLocation(TerminalDisplay::SCROLLBAR_RIGHT);
531
   
532 533 534
   return display;
}

535
void ViewManager::applyProfile(TerminalDisplay* view , const QString& profileKey)
536
{
537 538
    Profile* info = SessionManager::instance()->profile(profileKey);

539 540 541 542
    Q_ASSERT( info );

    const ColorScheme* colorScheme = ColorSchemeManager::instance()->
                                            findColorScheme(info->colorScheme());
543 544
    if ( !colorScheme )
       colorScheme = ColorSchemeManager::instance()->defaultColorScheme(); 
545 546
    Q_ASSERT( colorScheme );

547 548 549 550 551 552
    // menu bar visibility
    emit setMenuBarVisible( info->property(Profile::ShowMenuBar).value<bool>() );

    // tab bar visibility
    ViewContainer* container = _viewSplitter->activeContainer();
    int tabBarMode = info->property(Profile::TabBarMode).value<int>();
553 554
    int tabBarPosition = info->property(Profile::TabBarPosition).value<int>();

555 556 557 558 559 560 561
    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);

562 563 564 565 566
    if ( tabBarPosition == Profile::TabBarTop )
        container->setNavigationPosition(ViewContainer::NavigationPositionTop);
    else if ( tabBarPosition == Profile::TabBarBottom )
        container->setNavigationPosition(ViewContainer::NavigationPositionBottom);

567
    // load colour scheme
568
    view->setColorTable(colorScheme->colorTable());
569
    view->setOpacity(colorScheme->opacity());
570
    
571
    // load font 
572
    view->setVTFont(info->font());
573 574 575 576 577 578 579 580 581 582

    // set scroll-bar position
    int scrollBarPosition = info->property(Profile::ScrollBarPosition).value<int>();

    if ( scrollBarPosition == Profile::ScrollBarHidden )
       view->setScrollBarLocation(TerminalDisplay::SCROLLBAR_NONE);
    else if ( scrollBarPosition == Profile::ScrollBarLeft )
       view->setScrollBarLocation(TerminalDisplay::SCROLLBAR_LEFT);
    else if ( scrollBarPosition == Profile::ScrollBarRight )
       view->setScrollBarLocation(TerminalDisplay::SCROLLBAR_RIGHT);
583 584 585

    // terminal features
    bool blinkingCursor = info->property(Profile::BlinkingCursorEnabled).value<bool>();
586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605
    view->setBlinkingCursor(blinkingCursor);  

    // cursor shape
    int cursorShape = info->property(Profile::CursorShape).value<int>();

    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
    bool useCustomColor = info->property(Profile::UseCustomCursorColor).value<bool>();
    const QColor& cursorColor = info->property(Profile::CustomCursorColor).value<QColor>();
        
    view->setKeyboardCursorColor(!useCustomColor,cursorColor);

    // word characters
    view->setWordCharacters( info->property(Profile::WordCharacters).value<QString>() );
606 607
}

608 609 610 611 612 613 614 615 616
void ViewManager::updateViewsForSession(Session* session)
{
    QListIterator<QPointer<TerminalDisplay> > iter(_sessionMap.keys(session));
    while ( iter.hasNext() )
    {
        applyProfile(iter.next(),session->profileKey());
    }
}

617 618
void ViewManager::profileChanged(const QString& key)
{
619
    QHashIterator<QPointer<TerminalDisplay>,QPointer<Session> > iter(_sessionMap);
620 621 622 623 624 625

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

        // if session uses this profile, update the display
626
        if ( iter.key() != 0 && iter.value() != 0 && iter.value()->profileKey() == key )
627
        {
628
            applyProfile(iter.key(),key);
629 630
        }
    }
631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646
}

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

    ViewContainer* container = _viewSplitter->activeContainer();

    Q_ASSERT( container );

    QListIterator<QWidget*> viewIter(container->views());
    while ( viewIter.hasNext() )
    {
        ViewProperties* properties = container->viewProperties(viewIter.next());        Q_ASSERT( properties );
        list << properties; 
    } 
647

648
    return list;
649 650
}

651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666
void ViewManager::sendInputToAll()
{
    SessionGroup* group = new SessionGroup();
    group->setMasterMode( SessionGroup::CopyInputToAll );

    Session* activeSession = _sessionMap[qobject_cast<TerminalDisplay*>(activeView())];
    if ( activeSession != 0 )
    {
        QListIterator<Session*> iter( SessionManager::instance()->sessions() );
        while ( iter.hasNext() )
            group->addSession(iter.next());

        group->setMasterStatus(activeSession,true);
    }  
}

667 668 669 670 671
uint qHash(QPointer<TerminalDisplay> display)
{
    return qHash((TerminalDisplay*)display);
}

672
#include "ViewManager.moc"