ViewManager.cpp 37.2 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
// Qt
24
#include <QtCore/QDateTime>
Dirk Mueller's avatar
Dirk Mueller committed
25
#include <QtCore/QSignalMapper>
Jekyll Wu's avatar
Jekyll Wu committed
26
#include <QtCore/QStringList>
27
#include <QtGui/QMenu>
28

29
// KDE
Robert Knight's avatar
Robert Knight committed
30
#include <KAcceleratorManager>
31
#include <KGlobal>
32
33
#include <KLocale>
#include <KToggleAction>
34
#include <KActionCollection>
35
#include <KXMLGUIFactory>
Laurent Montel's avatar
Laurent Montel committed
36
#include <KConfigGroup>
37
38

// Konsole
39
40
#include <konsoleadaptor.h>

41
#include "ColorScheme.h"
42
#include "ProfileList.h"
43
#include "Session.h"
44
#include "TerminalDisplay.h"
45
46
47
48
#include "SessionController.h"
#include "SessionManager.h"
#include "ViewContainer.h"
#include "ViewSplitter.h"
49
#include "Profile.h"
50

51
52
using namespace Konsole;

53
54
ViewManager::ViewManager(QObject* parent , KActionCollection* collection)
    : QObject(parent)
55
    , _viewSplitter(0)
56
57
    , _actionCollection(collection)
    , _containerSignalMapper(new QSignalMapper(this))
58
    , _newViewMenu(0)
59
60
61
62
63
    , _navigationMethod(TabbedNavigation)
    , _navigationVisibility(ViewContainer::AlwaysShowNavigation)
    , _navigationPosition(ViewContainer::NavigationPositionTop)
    , _showQuickButtons(false)
    , _newTabBehavior(PutNewTabAtTheEnd)
64
{
65
    // create main view area
Kurt Hindenburg's avatar
Kurt Hindenburg committed
66
    _viewSplitter = new ViewSplitter(0);
67
    KAcceleratorManager::setNoAccel(_viewSplitter);
Robert Knight's avatar
Robert Knight committed
68

69
70
71
72
73
    // 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
Kurt Hindenburg's avatar
Kurt Hindenburg committed
74
    // mode
75
    _viewSplitter->setRecursiveSplitting(false);
76
    _viewSplitter->setFocusPolicy(Qt::NoFocus);
77

78
    // setup actions which are related to the views
79
80
81
    setupActions();

    // emit a signal when all of the views held by this view manager are destroyed
Kurt Hindenburg's avatar
Kurt Hindenburg committed
82
83
    connect(_viewSplitter , SIGNAL(allContainersEmpty()) , this , SIGNAL(empty()));
    connect(_viewSplitter , SIGNAL(empty(ViewSplitter*)) , this , SIGNAL(empty()));
84
85

    // listen for addition or removal of views from associated containers
Kurt Hindenburg's avatar
Kurt Hindenburg committed
86
87
    connect(_containerSignalMapper , SIGNAL(mapped(QObject*)) , this ,
            SLOT(containerViewsChanged(QObject*)));
88
89

    // listen for profile changes
Kurt Hindenburg's avatar
Kurt Hindenburg committed
90
91
92
93
    connect(SessionManager::instance() , SIGNAL(profileChanged(Profile::Ptr)) , this,
            SLOT(profileChanged(Profile::Ptr)));
    connect(SessionManager::instance() , SIGNAL(sessionUpdated(Session*)) , this,
            SLOT(updateViewsForSession(Session*)));
94
95
96
97
98

    //prepare DBus communication
    new KonsoleAdaptor(this);
    QDBusConnection::sessionBus().registerObject(QLatin1String("/Konsole"), this);

99
100
101
102
}

ViewManager::~ViewManager()
{
103
104
    delete _newViewMenu;
}
Kurt Hindenburg's avatar
Kurt Hindenburg committed
105
QMenu* ViewManager::createNewViewMenu()
106
107
108
109
110
{
    if (_newViewMenu)
        return _newViewMenu;

    _newViewMenu = new QMenu(0);
Kurt Hindenburg's avatar
Kurt Hindenburg committed
111
112
113
114
    ProfileList* newViewProfiles = new ProfileList(false, _newViewMenu);
    newViewProfiles->syncWidgetActions(_newViewMenu, true);
    connect(newViewProfiles, SIGNAL(profileSelected(Profile::Ptr)), this,
            SIGNAL(newViewRequest(Profile::Ptr)));
115
116

    return _newViewMenu;
117
}
118
119
120
QWidget* ViewManager::activeView() const
{
    ViewContainer* container = _viewSplitter->activeContainer();
Kurt Hindenburg's avatar
Kurt Hindenburg committed
121
    if (container) {
122
        return container->activeView();
Kurt Hindenburg's avatar
Kurt Hindenburg committed
123
    } else {
124
125
126
        return 0;
    }
}
127

128
129
130
131
132
QWidget* ViewManager::widget() const
{
    return _viewSplitter;
}

133
134
void ViewManager::setupActions()
{
135
    KActionCollection* collection = _actionCollection;
136

Kurt Hindenburg's avatar
Kurt Hindenburg committed
137
138
139
140
    KAction* nextViewAction = new KAction(i18nc("@action Shortcut entry", "Next Tab") , this);
    KAction* previousViewAction = new KAction(i18nc("@action Shortcut entry", "Previous Tab") , this);
    KAction* lastViewAction = new KAction(i18nc("@action Shortcut entry", "Switch to Last Tab") , this);
    KAction* nextContainerAction = new KAction(i18nc("@action Shortcut entry", "Next View Container") , this);
141

Kurt Hindenburg's avatar
Kurt Hindenburg committed
142
143
    KAction* moveViewLeftAction = new KAction(i18nc("@action Shortcut entry", "Move Tab Left") , this);
    KAction* moveViewRightAction = new KAction(i18nc("@action Shortcut entry", "Move Tab Right") , this);
144

145
146
147
148
149
    // list of actions that should only be enabled when there are multiple view
    // containers open
    QList<QAction*> multiViewOnlyActions;
    multiViewOnlyActions << nextContainerAction;

Kurt Hindenburg's avatar
Kurt Hindenburg committed
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
    if (collection) {
        KAction* splitLeftRightAction = new KAction(KIcon("view-split-left-right"),
                i18nc("@action:inmenu", "Split View Left/Right"),
                this);
        splitLeftRightAction->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_ParenLeft));
        collection->addAction("split-view-left-right", splitLeftRightAction);
        connect(splitLeftRightAction , SIGNAL(triggered()) , this , SLOT(splitLeftRight()));

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

        KAction* closeActiveAction = new KAction(i18nc("@action:inmenu Close Active View", "Close Active") , this);
Pino Toscano's avatar
Pino Toscano committed
165
        closeActiveAction->setIcon(KIcon("view-close"));
Kurt Hindenburg's avatar
Kurt Hindenburg committed
166
        closeActiveAction->setShortcut(QKeySequence(Qt::CTRL + Qt::SHIFT + Qt::Key_S));
167
        closeActiveAction->setEnabled(false);
Kurt Hindenburg's avatar
Kurt Hindenburg committed
168
169
        collection->addAction("close-active-view", closeActiveAction);
        connect(closeActiveAction , SIGNAL(triggered()) , this , SLOT(closeActiveView()));
170
171

        multiViewOnlyActions << closeActiveAction;
172

Kurt Hindenburg's avatar
Kurt Hindenburg committed
173
174
        KAction* closeOtherAction = new KAction(i18nc("@action:inmenu Close Other Views", "Close Others") , this);
        closeOtherAction->setShortcut(QKeySequence(Qt::CTRL + Qt::SHIFT + Qt::Key_O));
175
        closeOtherAction->setEnabled(false);
Kurt Hindenburg's avatar
Kurt Hindenburg committed
176
177
        collection->addAction("close-other-views", closeOtherAction);
        connect(closeOtherAction , SIGNAL(triggered()) , this , SLOT(closeOtherViews()));
178
179

        multiViewOnlyActions << closeOtherAction;
180

181
        KAction* detachViewAction = collection->addAction("detach-view");
182
        detachViewAction->setIcon(KIcon("tab-detach"));
Kurt Hindenburg's avatar
Kurt Hindenburg committed
183
        detachViewAction->setText(i18nc("@action:inmenu", "D&etach Current Tab"));
184
185
        // 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
186
187
        detachViewAction->setShortcut(QKeySequence(Qt::CTRL + Qt::SHIFT + Qt::Key_H));

Kurt Hindenburg's avatar
Kurt Hindenburg committed
188
189
        connect(this , SIGNAL(splitViewToggle(bool)) , this , SLOT(updateDetachViewState()));
        connect(detachViewAction , SIGNAL(triggered()) , this , SLOT(detachActiveView()));
190

191
        // Expand & Shrink Active View
Kurt Hindenburg's avatar
Kurt Hindenburg committed
192
193
        KAction* expandActiveAction = new KAction(i18nc("@action:inmenu", "Expand View") , this);
        expandActiveAction->setShortcut(QKeySequence(Qt::CTRL + Qt::SHIFT + Qt::Key_BracketRight));
194
        expandActiveAction->setEnabled(false);
Kurt Hindenburg's avatar
Kurt Hindenburg committed
195
196
        collection->addAction("expand-active-view", expandActiveAction);
        connect(expandActiveAction , SIGNAL(triggered()) , this , SLOT(expandActiveView()));
197

198
199
        multiViewOnlyActions << expandActiveAction;

Kurt Hindenburg's avatar
Kurt Hindenburg committed
200
201
        KAction* shrinkActiveAction = new KAction(i18nc("@action:inmenu", "Shrink View") , this);
        shrinkActiveAction->setShortcut(QKeySequence(Qt::CTRL + Qt::SHIFT + Qt::Key_BracketLeft));
202
        shrinkActiveAction->setEnabled(false);
Kurt Hindenburg's avatar
Kurt Hindenburg committed
203
204
        collection->addAction("shrink-active-view", shrinkActiveAction);
        connect(shrinkActiveAction , SIGNAL(triggered()) , this , SLOT(shrinkActiveView()));
205

206
207
        multiViewOnlyActions << shrinkActiveAction;

208
        // Next / Previous View , Next Container
Kurt Hindenburg's avatar
Kurt Hindenburg committed
209
210
211
212
213
214
        collection->addAction("next-view", nextViewAction);
        collection->addAction("previous-view", previousViewAction);
        collection->addAction("last-tab", lastViewAction);
        collection->addAction("next-container", nextContainerAction);
        collection->addAction("move-view-left", moveViewLeftAction);
        collection->addAction("move-view-right", moveViewRightAction);
215
216
217
218

        // Switch to tab N shortcuts
        const int SWITCH_TO_TAB_COUNT = 10;
        QSignalMapper* switchToTabMapper = new QSignalMapper(this);
Kurt Hindenburg's avatar
Kurt Hindenburg committed
219
220
221
222
223
        connect(switchToTabMapper, SIGNAL(mapped(int)), this, SLOT(switchToView(int)));
        for (int i = 0; i < SWITCH_TO_TAB_COUNT; i++) {
            KAction* switchToTabAction = new KAction(i18nc("@action Shortcut entry", "Switch to Tab %1", i + 1), this);
            switchToTabMapper->setMapping(switchToTabAction, i);
            connect(switchToTabAction, SIGNAL(triggered()), switchToTabMapper,
224
                    SLOT(map()));
Kurt Hindenburg's avatar
Kurt Hindenburg committed
225
            collection->addAction(QString("switch-to-tab-%1").arg(i), switchToTabAction);
226
        }
227
    }
Robert Knight's avatar
   
Robert Knight committed
228

229
    QListIterator<QAction*> iter(multiViewOnlyActions);
Kurt Hindenburg's avatar
Kurt Hindenburg committed
230
231
    while (iter.hasNext()) {
        connect(this , SIGNAL(splitViewToggle(bool)) , iter.next() , SLOT(setEnabled(bool)));
232
233
    }

234
    // keyboard shortcut only actions
Kurt Hindenburg's avatar
Kurt Hindenburg committed
235
236
    nextViewAction->setShortcut(QKeySequence(Qt::SHIFT + Qt::Key_Right));
    connect(nextViewAction, SIGNAL(triggered()) , this , SLOT(nextView()));
237
    _viewSplitter->addAction(nextViewAction);
Robert Knight's avatar
   
Robert Knight committed
238

Kurt Hindenburg's avatar
Kurt Hindenburg committed
239
240
    previousViewAction->setShortcut(QKeySequence(Qt::SHIFT + Qt::Key_Left));
    connect(previousViewAction, SIGNAL(triggered()) , this , SLOT(previousView()));
241
    _viewSplitter->addAction(previousViewAction);
Robert Knight's avatar
   
Robert Knight committed
242

Kurt Hindenburg's avatar
Kurt Hindenburg committed
243
244
    nextContainerAction->setShortcut(QKeySequence(Qt::SHIFT + Qt::Key_Tab));
    connect(nextContainerAction , SIGNAL(triggered()) , this , SLOT(nextContainer()));
245
    _viewSplitter->addAction(nextContainerAction);
246

Kurt Hindenburg's avatar
Kurt Hindenburg committed
247
248
    moveViewLeftAction->setShortcut(QKeySequence(Qt::CTRL + Qt::SHIFT + Qt::Key_Left));
    connect(moveViewLeftAction , SIGNAL(triggered()) , this , SLOT(moveActiveViewLeft()));
249
    _viewSplitter->addAction(moveViewLeftAction);
250

Kurt Hindenburg's avatar
Kurt Hindenburg committed
251
252
    moveViewRightAction->setShortcut(QKeySequence(Qt::CTRL + Qt::SHIFT + Qt::Key_Right));
    connect(moveViewRightAction , SIGNAL(triggered()) , this , SLOT(moveActiveViewRight()));
253
    _viewSplitter->addAction(moveViewRightAction);
254

Kurt Hindenburg's avatar
Kurt Hindenburg committed
255
    connect(lastViewAction, SIGNAL(triggered()) , this , SLOT(lastView()));
256
    _viewSplitter->addAction(lastViewAction);
Robert Knight's avatar
   
Robert Knight committed
257
}
258
259
260
261
void ViewManager::switchToView(int index)
{
    Q_ASSERT(index >= 0);
    ViewContainer* container = _viewSplitter->activeContainer();
Kurt Hindenburg's avatar
Kurt Hindenburg committed
262
    Q_ASSERT(container);
263
264
265
266
267
    QList<QWidget*> containerViews = container->views();
    if (index >= containerViews.count())
        return;
    container->setActiveView(containerViews.at(index));
}
268
269
void ViewManager::updateDetachViewState()
{
270
271
    if (!_actionCollection)
        return;
272

273

274
275
    bool splitView = _viewSplitter->containers().count() >= 2;
    bool shouldEnable = splitView || _viewSplitter->activeContainer()->views().count() >= 2;
276

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

Kurt Hindenburg's avatar
Kurt Hindenburg committed
279
    if (detachAction && shouldEnable != detachAction->isEnabled())
280
        detachAction->setEnabled(shouldEnable);
281
}
282
283
284
void ViewManager::moveActiveViewLeft()
{
    ViewContainer* container = _viewSplitter->activeContainer();
Kurt Hindenburg's avatar
Kurt Hindenburg committed
285
286
    Q_ASSERT(container);
    container->moveActiveView(ViewContainer::MoveViewLeft);
287
288
289
290
}
void ViewManager::moveActiveViewRight()
{
    ViewContainer* container = _viewSplitter->activeContainer();
Kurt Hindenburg's avatar
Kurt Hindenburg committed
291
292
    Q_ASSERT(container);
    container->moveActiveView(ViewContainer::MoveViewRight);
293
}
Robert Knight's avatar
   
Robert Knight committed
294
295
296
297
298
299
300
301
302
void ViewManager::nextContainer()
{
    _viewSplitter->activateNextContainer();
}

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

Kurt Hindenburg's avatar
Kurt Hindenburg committed
303
    Q_ASSERT(container);
Robert Knight's avatar
   
Robert Knight committed
304
305
306
307
308
309
310
311

    container->activateNextView();
}

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

Kurt Hindenburg's avatar
Kurt Hindenburg committed
312
    Q_ASSERT(container);
Robert Knight's avatar
   
Robert Knight committed
313
314

    container->activatePreviousView();
315
}
316

317
318
319
320
void ViewManager::lastView()
{
    ViewContainer* container = _viewSplitter->activeContainer();

Kurt Hindenburg's avatar
Kurt Hindenburg committed
321
    Q_ASSERT(container);
322
323
324
325

    container->activateLastView();
}

326
327
void ViewManager::detachActiveView()
{
Kurt Hindenburg's avatar
Kurt Hindenburg committed
328
    // find the currently active view and remove it from its container
329
330
    ViewContainer* container = _viewSplitter->activeContainer();

331
332
333
334
335
    detachView(container, container->activeView());
}

void ViewManager::detachView(ViewContainer* container, QWidget* widgetView)
{
Kurt Hindenburg's avatar
Kurt Hindenburg committed
336
    TerminalDisplay * viewToDetach =
337
338
339
        dynamic_cast<TerminalDisplay*>(widgetView);

    if (!viewToDetach)
340
341
        return;

342
    emit viewDetached(_sessionMap[viewToDetach]);
343

344
    _sessionMap.remove(viewToDetach);
345
346

    // remove the view from this window
347
348
    container->removeView(viewToDetach);
    viewToDetach->deleteLater();
349
350
351
352

    // 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
Kurt Hindenburg's avatar
Kurt Hindenburg committed
353
354
    if (_viewSplitter->containers().count() > 1 &&
            container->views().count() == 0) {
355
        removeContainer(container);
356
357
358
359
    }

}

360
void ViewManager::sessionFinished()
361
{
362
363
364
365
    // if this slot is called after the view manager's main widget
    // has been destroyed, do nothing
    if (!_viewSplitter)
        return;
366

367
368
    Session* session = qobject_cast<Session*>(sender());

369
370
    // We're using setSelectionBehaviorOnRemove(QTabBar::SelectPreviousTab)
    // so no need to manually select next tab.
371

372
373
    Q_ASSERT(session);

374
    // close attached views
375
    QList<TerminalDisplay*> children = _viewSplitter->findChildren<TerminalDisplay*>();
376

Kurt Hindenburg's avatar
Kurt Hindenburg committed
377
378
    foreach(TerminalDisplay * view , children) {
        if (_sessionMap[view] == session) {
379
            _sessionMap.remove(view);
380
            view->deleteLater();
381
        }
Robert Knight's avatar
   
Robert Knight committed
382
    }
383

Kurt Hindenburg's avatar
Kurt Hindenburg committed
384
    // This is needed to remove this controller from factory() in
385
386
387
    // order to prevent BUG: 185466 - disappearing menu popup
    if (_pluggedController)
        emit unplugController(_pluggedController);
Robert Knight's avatar
   
Robert Knight committed
388
389
390
391
}

void ViewManager::focusActiveView()
{
Kurt Hindenburg's avatar
Kurt Hindenburg committed
392
    // give the active view in a container the focus.  this ensures
Robert Knight's avatar
Robert Knight committed
393
394
395
396
397
    // 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

Kurt Hindenburg's avatar
Kurt Hindenburg committed
398
399
    ViewContainer* container = _viewSplitter->activeContainer();
    if (container) {
Robert Knight's avatar
   
Robert Knight committed
400
        QWidget* activeView = container->activeView();
Kurt Hindenburg's avatar
Kurt Hindenburg committed
401
        if (activeView) {
Robert Knight's avatar
   
Robert Knight committed
402
403
404
            activeView->setFocus(Qt::MouseFocusReason);
        }
    }
405
406
}

407

Kurt Hindenburg's avatar
Kurt Hindenburg committed
408
void ViewManager::viewActivated(QWidget* view)
409
{
Kurt Hindenburg's avatar
Kurt Hindenburg committed
410
    Q_ASSERT(view != 0);
411
412
413
414
415

    // 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);
416
417
}

418
void ViewManager::splitLeftRight()
419
{
420
    splitView(Qt::Horizontal);
421
}
422
void ViewManager::splitTopBottom()
423
{
424
    splitView(Qt::Vertical);
425
426
}

427
void ViewManager::splitView(Qt::Orientation orientation)
428
{
429
430
    ViewContainer* container = createContainer();

431
    // iterate over each session which has a view in the current active
Kurt Hindenburg's avatar
Kurt Hindenburg committed
432
    // container and create a new view for that session in a new container
433
    QListIterator<QWidget*> existingViewIter(_viewSplitter->activeContainer()->views());
434

Kurt Hindenburg's avatar
Kurt Hindenburg committed
435
    while (existingViewIter.hasNext()) {
436
        Session* session = _sessionMap[(TerminalDisplay*)existingViewIter.next()];
437
        TerminalDisplay* display = createTerminalDisplay(session);
Jekyll Wu's avatar
Jekyll Wu committed
438
439
        const Profile::Ptr profile = SessionManager::instance()->sessionProfile(session);
        applyProfileToView(display, profile);
Kurt Hindenburg's avatar
Kurt Hindenburg committed
440
        ViewProperties* properties = createController(session, display);
441

442
        _sessionMap[display] = session;
443

Kurt Hindenburg's avatar
Kurt Hindenburg committed
444
445
        container->addView(display, properties);
        session->addView(display);
446
    }
447

Kurt Hindenburg's avatar
Kurt Hindenburg committed
448
    _viewSplitter->addContainer(container, orientation);
449
    emit splitViewToggle(_viewSplitter->containers().count() > 0);
450

451
452
    // focus the new container
    container->containerWidget()->setFocus();
453
454

    // ensure that the active view is focused after the split / unsplit
455
456
457
    ViewContainer* activeContainer = _viewSplitter->activeContainer();
    QWidget* activeView = activeContainer ? activeContainer->activeView() : 0;

Kurt Hindenburg's avatar
Kurt Hindenburg committed
458
    if (activeView)
459
        activeView->setFocus(Qt::OtherFocusReason);
460
}
461
void ViewManager::removeContainer(ViewContainer* container)
462
{
463
    // remove session map entries for views in this container
Kurt Hindenburg's avatar
Kurt Hindenburg committed
464
    foreach(QWidget * view , container->views()) {
465
466
467
        TerminalDisplay* display = qobject_cast<TerminalDisplay*>(view);
        Q_ASSERT(display);
        _sessionMap.remove(display);
Kurt Hindenburg's avatar
Kurt Hindenburg committed
468
    }
469

470
    _viewSplitter->removeContainer(container);
471
    container->deleteLater();
472

Kurt Hindenburg's avatar
Kurt Hindenburg committed
473
    emit splitViewToggle(_viewSplitter->containers().count() > 1);
474
}
475
476
void ViewManager::expandActiveView()
{
Kurt Hindenburg's avatar
Kurt Hindenburg committed
477
    _viewSplitter->adjustContainerSize(_viewSplitter->activeContainer(), 10);
478
479
480
}
void ViewManager::shrinkActiveView()
{
Kurt Hindenburg's avatar
Kurt Hindenburg committed
481
    _viewSplitter->adjustContainerSize(_viewSplitter->activeContainer(), -10);
482
}
483
484
485
void ViewManager::closeActiveView()
{
    // only do something if there is more than one container active
Kurt Hindenburg's avatar
Kurt Hindenburg committed
486
    if (_viewSplitter->containers().count() > 1) {
487
488
489
        ViewContainer* container = _viewSplitter->activeContainer();

        removeContainer(container);
490

Kurt Hindenburg's avatar
Kurt Hindenburg committed
491
        // focus next container so that user can continue typing
492
493
494
        // without having to manually focus it themselves
        nextContainer();
    }
495
496
497
498
499
500
}
void ViewManager::closeOtherViews()
{
    ViewContainer* active = _viewSplitter->activeContainer();

    QListIterator<ViewContainer*> iter(_viewSplitter->containers());
Kurt Hindenburg's avatar
Kurt Hindenburg committed
501
    while (iter.hasNext()) {
502
        ViewContainer* next = iter.next();
Kurt Hindenburg's avatar
Kurt Hindenburg committed
503
        if (next != active)
504
            removeContainer(next);
505
506
507
    }
}

508
SessionController* ViewManager::createController(Session* session , TerminalDisplay* view)
509
{
Robert Knight's avatar
Robert Knight committed
510
511
    // create a new controller for the session, and ensure that this view manager
    // is notified when the view gains the focus
Kurt Hindenburg's avatar
Kurt Hindenburg committed
512
513
514
515
516
517
518
519
    SessionController* controller = new SessionController(session, view, this);
    connect(controller , SIGNAL(focused(SessionController*)) , this , SLOT(controllerChanged(SessionController*)));
    connect(session , SIGNAL(destroyed()) , controller , SLOT(deleteLater()));
    connect(session , SIGNAL(primaryScreenInUse(bool)) ,
            controller , SLOT(setupPrimaryScreenSpecificActions(bool)));
    connect(session , SIGNAL(selectedText(QString)) ,
            controller , SLOT(updateCopyAction(QString)));
    connect(view , SIGNAL(destroyed()) , controller , SLOT(deleteLater()));
520

521
522
523
    // if this is the first controller created then set it as the active controller
    if (!_pluggedController)
        controllerChanged(controller);
524

525
526
527
    return controller;
}

528
529
void ViewManager::controllerChanged(SessionController* controller)
{
Kurt Hindenburg's avatar
Kurt Hindenburg committed
530
    if (controller == _pluggedController)
531
        return;
532

533
    _viewSplitter->setFocusProxy(controller->view());
534

535
536
    _pluggedController = controller;
    emit activeViewChanged(controller);
537
538
539
540
}

SessionController* ViewManager::activeViewController() const
{
541
    return _pluggedController;
542
}
543
544
545
546
547
548

IncrementalSearchBar* ViewManager::searchBar() const
{
    return _viewSplitter->activeSplitter()->activeContainer()->searchBar();
}

549
550
void ViewManager::createView(Session* session, ViewContainer* container, int index)
{
551
    // notify this view manager when the session finishes so that its view
552
    // can be deleted
553
554
    //
    // TODO - Find a more efficient a way to avoid multiple connections
Kurt Hindenburg's avatar
Kurt Hindenburg committed
555
556
    disconnect(session , SIGNAL(finished()) , this , SLOT(sessionFinished()));
    connect(session , SIGNAL(finished()) , this , SLOT(sessionFinished()));
557

Kurt Hindenburg's avatar
Kurt Hindenburg committed
558
559
560
    TerminalDisplay* display = createTerminalDisplay(session);
    const Profile::Ptr profile = SessionManager::instance()->sessionProfile(session);
    applyProfileToView(display, profile);
561

Kurt Hindenburg's avatar
Kurt Hindenburg committed
562
563
    // set initial size
    display->setSize(80, 40);
564

Kurt Hindenburg's avatar
Kurt Hindenburg committed
565
    ViewProperties* properties = createController(session, display);
566

Kurt Hindenburg's avatar
Kurt Hindenburg committed
567
568
569
    _sessionMap[display] = session;
    container->addView(display, properties, index);
    session->addView(display);
570

Kurt Hindenburg's avatar
Kurt Hindenburg committed
571
572
    // tell the session whether it has a light or dark background
    session->setDarkBackground(colorSchemeForProfile(profile)->hasDarkBackground());
573

Kurt Hindenburg's avatar
Kurt Hindenburg committed
574
575
576
577
    if (container == _viewSplitter->activeContainer()) {
        container->setActiveView(display);
        display->setFocus(Qt::OtherFocusReason);
    }
578

Kurt Hindenburg's avatar
Kurt Hindenburg committed
579
    updateDetachViewState();
580
}
581

582
void ViewManager::createView(Session* session)
583
{
584
    // create the default container
Kurt Hindenburg's avatar
Kurt Hindenburg committed
585
    if (_viewSplitter->containers().count() == 0) {
586
587
        ViewContainer* container = createContainer();
        _viewSplitter->addContainer(container, Qt::Vertical);
588
        emit splitViewToggle(false);
589
590
    }

591
    // new tab will be put at the end by default.
592
593
    int index = -1;

594
    if (_newTabBehavior == PutNewTabAfterCurrentTab) {
595
        QWidget* view = activeView();
Kurt Hindenburg's avatar
Kurt Hindenburg committed
596
        if (view) {
597
598
599
            QList<QWidget*> views =  _viewSplitter->activeContainer()->views();
            index = views.indexOf(view) + 1;
        }
600
601
    }

Robert Knight's avatar
Robert Knight committed
602
603
    // 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
Kurt Hindenburg's avatar
Kurt Hindenburg committed
604
    // a controller for the session/display pair
605
    QListIterator<ViewContainer*> containerIter(_viewSplitter->containers());
Kurt Hindenburg's avatar
Kurt Hindenburg committed
606
    while (containerIter.hasNext()) {
607
        ViewContainer* container = containerIter.next();
608
        createView(session, container, index);
609
610
611
    }
}

612
ViewContainer* ViewManager::createContainer()
613
{
614
615
    ViewContainer::NavigationPosition position =
        static_cast<ViewContainer::NavigationPosition>(_navigationPosition);
616

617
618
    ViewContainer* container = 0;

Kurt Hindenburg's avatar
Kurt Hindenburg committed
619
620
621
622
623
624
    switch (_navigationMethod) {
    case TabbedNavigation: {
        container =
            new TabbedViewContainer(position, _viewSplitter);

        connect(container,
625
                SIGNAL(detachTab(ViewContainer*,QWidget*)),
Kurt Hindenburg's avatar
Kurt Hindenburg committed
626
                this,
627
                SLOT(detachView(ViewContainer*,QWidget*))
Kurt Hindenburg's avatar
Kurt Hindenburg committed
628
629
               );
        connect(container,
630
                SIGNAL(closeTab(ViewContainer*,QWidget*)),
Kurt Hindenburg's avatar
Kurt Hindenburg committed
631
                this,
632
                SLOT(closeTabFromContainer(ViewContainer*,QWidget*)));
Kurt Hindenburg's avatar
Kurt Hindenburg committed
633
634
635
636
637
638

    }
    break;
    case NoNavigation:
    default:
        container = new StackedViewContainer(_viewSplitter);
639
    }
640

641
    applyNavigationOptions(container);
642

643
    // connect signals and slots
644
    connect(container , SIGNAL(viewAdded(QWidget*,ViewProperties*)) , _containerSignalMapper ,
Kurt Hindenburg's avatar
Kurt Hindenburg committed
645
646
647
648
649
650
            SLOT(map()));
    connect(container , SIGNAL(viewRemoved(QWidget*)) , _containerSignalMapper ,
            SLOT(map()));
    _containerSignalMapper->setMapping(container, container);

    connect(container, SIGNAL(newViewRequest()), this, SIGNAL(newViewRequest()));
651
652
    connect(container, SIGNAL(moveViewRequest(int,int,bool&)),
            this , SLOT(containerMoveViewRequest(int,int,bool&)));
Kurt Hindenburg's avatar
Kurt Hindenburg committed
653
654
655
    connect(container , SIGNAL(viewRemoved(QWidget*)) , this , SLOT(viewCloseRequest(QWidget*)));
    connect(container , SIGNAL(closeRequest(QWidget*)) , this , SLOT(viewCloseRequest(QWidget*)));
    connect(container , SIGNAL(activeViewChanged(QWidget*)) , this , SLOT(viewActivated(QWidget*)));
656

657
    return container;
658
}
659
660
void ViewManager::containerMoveViewRequest(int index, int id, bool& moved)
{
661
662
    ViewContainer* container = qobject_cast<ViewContainer*>(sender());
    SessionController* controller = qobject_cast<SessionController*>(ViewProperties::propertiesById(id));
663

664
665
    if (!controller)
        return;
666

Kurt Hindenburg's avatar
Kurt Hindenburg committed
667
    createView(controller->session(), container, index);
668
    controller->session()->refresh();
669
    moved = true;
670
}
671
672
673
674
675
676
void ViewManager::setNavigationMethod(NavigationMethod method)
{
    _navigationMethod = method;

    KActionCollection* collection = _actionCollection;

Kurt Hindenburg's avatar
Kurt Hindenburg committed
677
    if (collection) {
678
679
680
681
682
683
684
685
686
        // FIXME: The following disables certain actions for the KPart that it
        // doesn't actually have a use for, to avoid polluting the action/shortcut
        // namespace of an application using the KPart (otherwise, a shortcut may
        // be in use twice, and the user gets to see an "ambiguous shortcut over-
        // load" error dialog). However, this approach sucks - it's the inverse of
        // what it should be. Rather than disabling actions not used by the KPart,
        // a method should be devised to only enable those that are used, perhaps
        // by using a separate action collection.

687
        const bool enable = (_navigationMethod != NoNavigation) ;
688
689
        QAction* action;

Kurt Hindenburg's avatar
Kurt Hindenburg committed
690
691
        action = collection->action("next-view");
        if (action) action->setEnabled(enable);
692

Kurt Hindenburg's avatar
Kurt Hindenburg committed
693
694
        action = collection->action("previous-view");
        if (action) action->setEnabled(enable);
695

Kurt Hindenburg's avatar
Kurt Hindenburg committed
696
697
        action = collection->action("last-tab");
        if (action) action->setEnabled(enable);
698

Kurt Hindenburg's avatar
Kurt Hindenburg committed
699
700
        action = collection->action("split-view-left-right");
        if (action) action->setEnabled(enable);
701

Kurt Hindenburg's avatar
Kurt Hindenburg committed
702
703
        action = collection->action("split-view-top-bottom");
        if (action) action->setEnabled(enable);
704

Kurt Hindenburg's avatar
Kurt Hindenburg committed
705
706
        action = collection->action("rename-session");
        if (action) action->setEnabled(enable);
707

Kurt Hindenburg's avatar
Kurt Hindenburg committed
708
709
        action = collection->action("move-view-left");
        if (action) action->setEnabled(enable);
710

Kurt Hindenburg's avatar
Kurt Hindenburg committed
711
712
        action = collection->action("move-view-right");
        if (action) action->setEnabled(enable);
713
714
715
    }
}

Kurt Hindenburg's avatar
Kurt Hindenburg committed
716
717
718
719
ViewManager::NavigationMethod ViewManager::navigationMethod() const
{
    return _navigationMethod;
}
720

721
722
void ViewManager::containerViewsChanged(QObject* container)
{
Kurt Hindenburg's avatar
Kurt Hindenburg committed
723
724
725
    if (_viewSplitter && container == _viewSplitter->activeContainer()) {
        emit viewPropertiesChanged(viewProperties());
    }
726
727
}

728
729
void ViewManager::viewCloseRequest(QWidget* view)
{
730
731
    //FIXME Check that this cast is actually legal
    TerminalDisplay* display = (TerminalDisplay*)view;
732

733
734
    Q_ASSERT(display);

735
736
    // 1. detach view from session
    // 2. if the session has no views left, close it
737
    Session* session = _sessionMap[ display ];
738
    _sessionMap.remove(display);
Kurt Hindenburg's avatar
Kurt Hindenburg committed
739
    if (session) {
740
        display->deleteLater();
741

Kurt Hindenburg's avatar
Kurt Hindenburg committed
742
        if (session->views().count() == 0)
743
            session->close();
744
    }
745
746
747
748
749
    //we only update the focus if the splitter is still alive
    if (_viewSplitter) {
        focusActiveView();
        updateDetachViewState();
    }
750
751
752
753
    // The below causes the menus  to be messed up
    // Only happenss when using the tab bar close button
//    if (_pluggedController)
//        emit unplugController(_pluggedController);
754
755
}

756
TerminalDisplay* ViewManager::createTerminalDisplay(Session* session)
757
{
758
759
    TerminalDisplay* display = new TerminalDisplay(0);
    display->setRandomSeed(session->sessionId() * 31);
760

761
    return display;
762
763
}

Jekyll Wu's avatar
Jekyll Wu committed
764
const ColorScheme* ViewManager::colorSchemeForProfile(const Profile::Ptr profile) const
765
{
766
    const ColorScheme* colorScheme = ColorSchemeManager::instance()->
Kurt Hindenburg's avatar
Kurt Hindenburg committed
767
768
769
770
                                     findColorScheme(profile->colorScheme());
    if (!colorScheme)
        colorScheme = ColorSchemeManager::instance()->defaultColorScheme();
    Q_ASSERT(colorScheme);
771

772
773
774
    return colorScheme;
}

Jekyll Wu's avatar
Jekyll Wu committed
775
void ViewManager::applyProfileToView(TerminalDisplay* view , const Profile::Ptr profile)
776
{
Kurt Hindenburg's avatar
Kurt Hindenburg committed
777
    Q_ASSERT(profile);
778

Kurt Hindenburg's avatar
Kurt Hindenburg committed
779
    emit setSaveGeometryOnExitRequest(profile->property<bool>(Profile::SaveGeometryOnExit));
780

781
782
    emit updateWindowIcon();

783
    // load color scheme
784
    ColorEntry table[TABLE_COLORS];
785
    const ColorScheme* colorScheme = colorSchemeForProfile(profile);
Kurt Hindenburg's avatar
Kurt Hindenburg committed
786
    colorScheme->getColorTable(table , view->randomSeed());
787
    view->setColorTable(table);
788
    view->setOpacity(colorScheme->opacity());
789
    view->setWallpaper(colorScheme->wallpaper());
790

Kurt Hindenburg's avatar
Kurt Hindenburg committed
791
    // load font
Jekyll Wu's avatar
Jekyll Wu committed
792
793
794
    view->setAntialias(profile->property<bool>(Profile::AntiAliasFonts));
    view->setBoldIntense(profile->property<bool>(Profile::BoldIntense));
    view->setVTFont(profile->font());
795
796

    // set scroll-bar position
Jekyll Wu's avatar
Jekyll Wu committed
797
    int scrollBarPosition = profile->property<int>(Profile::ScrollBarPosition);
798

Kurt Hindenburg's avatar
Kurt Hindenburg committed
799
800
801
802
803
804
    if (scrollBarPosition == Profile::ScrollBarLeft)
        view->setScrollBarPosition(TerminalDisplay::ScrollBarLeft);
    else if (scrollBarPosition == Profile::ScrollBarRight)
        view->setScrollBarPosition(TerminalDisplay::ScrollBarRight);
    else if (scrollBarPosition == Profile::ScrollBarHidden)
        view->setScrollBarPosition(TerminalDisplay::ScrollBarHidden);
805

806
    // show hint about termianl size after resizing
Jekyll Wu's avatar
Jekyll Wu committed
807
    view->setShowTerminalSizeHint(profile->property<bool>(Profile::ShowTerminalSizeHint));
808

809
    // terminal features
Jekyll Wu's avatar
Jekyll Wu committed
810
    bool blinkingCursor = profile->property<bool>(Profile::BlinkingCursorEnabled);
Jekyll Wu's avatar
Jekyll Wu committed
811
    view->setBlinkingCursorEnabled(blinkingCursor);
812

Jekyll Wu's avatar
Jekyll Wu committed
813
    bool blinkingText = profile->property<bool>(Profile::BlinkingTextEnabled);
814
815
    view->setBlinkingTextEnabled(blinkingText);

Jekyll Wu's avatar
Jekyll Wu committed
816
    int tripleClickMode = profile->property<int>(Profile::TripleClickMode);
Kurt Hindenburg's avatar
Kurt Hindenburg committed
817
    view->setTripleClickMode(TerminalDisplay::TripleClickMode(tripleClickMode));
818

Jekyll Wu's avatar
Jekyll Wu committed
819
    view->setUnderlineLinks(profile->property<bool>(Profile::UnderlineLinksEnabled));
820

Jekyll Wu's avatar
Jekyll Wu committed
821
    bool bidiEnabled = profile->property<bool>(Profile::BidiRenderingEnabled);
822
823
    view->setBidiEnabled(bidiEnabled);

824
    // cursor shape
Jekyll Wu's avatar
Jekyll Wu committed
825
    int cursorShape = profile->property<int>(Profile::CursorShape);