* Sync debian foo with experimental * Bumped compat to 10 * Removed --parallel from rules, standard compat 10 * Bumped Standards to 3.9.8, no changes needed * Bumped minimum version debhelper (>= 10) * Bumped minimum version libqtermwidget5-0-dev (>= 0.7.0) * Added build dependency libkf5windowsystem-dev * Added build dependency liblxqt0-dev (>= 0.11.0) * Added build dependency libqt5svg5-dev * Added build dependency libqt5xdg-dev (>= 3.0.0) * Added Recommends qterminal-l10n * Fixed VCS fields, using https, plain /git/ and pointing to the right branch * Fixed copyright Format field, using https * Exported LC_ALL=C.UTF-8, make builds reproducible * Added hardening options * Added translation control to rules * Set CMAKE_BUILD_TYPE=RelWithDebInfo
342 lines
9.5 KiB
C++
342 lines
9.5 KiB
C++
/***************************************************************************
|
|
* Copyright (C) 2010 by Petr Vanek *
|
|
* petr@scribus.info *
|
|
* *
|
|
* 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, see <http://www.gnu.org/licenses/>. *
|
|
***************************************************************************/
|
|
|
|
#include <QGridLayout>
|
|
#include <QSplitter>
|
|
#include <QInputDialog>
|
|
|
|
#include "termwidgetholder.h"
|
|
#include "termwidget.h"
|
|
#include "properties.h"
|
|
#include <assert.h>
|
|
|
|
|
|
TermWidgetHolder::TermWidgetHolder(const QString & wdir, const QString & shell, QWidget * parent)
|
|
: QWidget(parent),
|
|
m_wdir(wdir),
|
|
m_shell(shell),
|
|
m_currentTerm(0)
|
|
{
|
|
setFocusPolicy(Qt::NoFocus);
|
|
QGridLayout * lay = new QGridLayout(this);
|
|
lay->setSpacing(0);
|
|
lay->setContentsMargins(0, 0, 0, 0);
|
|
|
|
QSplitter *s = new QSplitter(this);
|
|
s->setFocusPolicy(Qt::NoFocus);
|
|
TermWidget *w = newTerm();
|
|
s->addWidget(w);
|
|
lay->addWidget(s);
|
|
|
|
setLayout(lay);
|
|
}
|
|
|
|
TermWidgetHolder::~TermWidgetHolder()
|
|
{
|
|
}
|
|
|
|
void TermWidgetHolder::setInitialFocus()
|
|
{
|
|
QList<TermWidget*> list = findChildren<TermWidget*>();
|
|
TermWidget * w = list.count() == 0 ? 0 : list.at(0);
|
|
if (w)
|
|
w->setFocus(Qt::OtherFocusReason);
|
|
}
|
|
|
|
void TermWidgetHolder::loadSession()
|
|
{
|
|
bool ok;
|
|
QString name = QInputDialog::getItem(this, tr("Load Session"),
|
|
tr("List of saved sessions:"),
|
|
Properties::Instance()->sessions.keys(),
|
|
0, false, &ok);
|
|
if (!ok || name.isEmpty())
|
|
return;
|
|
#if 0
|
|
foreach (QWidget * w, findChildren<QWidget*>())
|
|
{
|
|
if (w)
|
|
{
|
|
delete w;
|
|
w = 0;
|
|
}
|
|
}
|
|
|
|
qDebug() << "load" << name << QString(Properties::Instance()->sessions[name]);
|
|
QStringList splitters = QString(Properties::Instance()->sessions[name]).split("|", QString::SkipEmptyParts);
|
|
foreach (QString splitter, splitters)
|
|
{
|
|
QStringList components = splitter.split(",");
|
|
qDebug() << "comp" << components;
|
|
// orientation
|
|
Qt::Orientation orientation;
|
|
if (components.size() > 0)
|
|
orientation = components.takeAt(0).toInt();
|
|
// sizes
|
|
QList<int> sizes;
|
|
QList<TermWidget*> widgets;
|
|
foreach (QString s, components)
|
|
{
|
|
sizes << s.toInt();
|
|
widgets << newTerm();
|
|
}
|
|
// new terms
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void TermWidgetHolder::saveSession(const QString & name)
|
|
{
|
|
Session dump;
|
|
QString num("%1");
|
|
foreach(QSplitter *w, findChildren<QSplitter*>())
|
|
{
|
|
dump += '|' + num.arg(w->orientation());
|
|
foreach (int i, w->sizes())
|
|
dump += ',' + num.arg(i);
|
|
}
|
|
Properties::Instance()->sessions[name] = dump;
|
|
qDebug() << "dump" << dump;
|
|
}
|
|
|
|
TermWidget* TermWidgetHolder::currentTerminal()
|
|
{
|
|
return m_currentTerm;
|
|
}
|
|
|
|
void TermWidgetHolder::setWDir(const QString & wdir)
|
|
{
|
|
m_wdir = wdir;
|
|
}
|
|
|
|
void TermWidgetHolder::switchNextSubterminal()
|
|
{
|
|
// TODO/FIXME: merge switchPrevSubterminal with switchNextSubterminal
|
|
QList<TermWidget*> l = findChildren<TermWidget*>();
|
|
int ix = -1;
|
|
foreach (TermWidget * w, l)
|
|
{
|
|
++ix;
|
|
if (w->impl()->hasFocus())
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (ix < l.count()-1)
|
|
{
|
|
l.at(ix+1)->impl()->setFocus(Qt::OtherFocusReason);
|
|
}
|
|
else if (ix == l.count()-1)
|
|
{
|
|
l.at(0)->impl()->setFocus(Qt::OtherFocusReason);
|
|
}
|
|
}
|
|
|
|
void TermWidgetHolder::switchPrevSubterminal()
|
|
{
|
|
// TODO/FIXME: merge switchPrevSubterminal with switchNextSubterminal
|
|
QList<TermWidget*> l = findChildren<TermWidget*>();
|
|
int ix = -1;
|
|
foreach (TermWidget * w, l)
|
|
{
|
|
++ix;
|
|
if (w->impl()->hasFocus())
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (ix > 0)
|
|
{
|
|
l.at(ix-1)->impl()->setFocus(Qt::OtherFocusReason);
|
|
}
|
|
else if (ix == 0)
|
|
{
|
|
l.at(l.count()-1)->impl()->setFocus(Qt::OtherFocusReason);
|
|
}
|
|
}
|
|
|
|
void TermWidgetHolder::clearActiveTerminal()
|
|
{
|
|
currentTerminal()->impl()->clear();
|
|
}
|
|
|
|
void TermWidgetHolder::propertiesChanged()
|
|
{
|
|
foreach(TermWidget *w, findChildren<TermWidget*>())
|
|
w->propertiesChanged();
|
|
}
|
|
|
|
void TermWidgetHolder::splitHorizontal(TermWidget * term)
|
|
{
|
|
split(term, Qt::Vertical);
|
|
}
|
|
|
|
void TermWidgetHolder::splitVertical(TermWidget * term)
|
|
{
|
|
split(term, Qt::Horizontal);
|
|
}
|
|
|
|
void TermWidgetHolder::splitCollapse(TermWidget * term)
|
|
{
|
|
QSplitter * parent = qobject_cast<QSplitter*>(term->parent());
|
|
assert(parent);
|
|
term->setParent(0);
|
|
delete term;
|
|
|
|
QWidget *nextFocus = Q_NULLPTR;
|
|
|
|
// Collapse splitters containing a single element, excluding the top one.
|
|
if (parent->count() == 1)
|
|
{
|
|
QSplitter *uselessSplitterParent = qobject_cast<QSplitter*>(parent->parent());
|
|
if (uselessSplitterParent != Q_NULLPTR) {
|
|
int idx = uselessSplitterParent->indexOf(parent);
|
|
assert(idx != -1);
|
|
QWidget *singleHeir = parent->widget(0);
|
|
uselessSplitterParent->insertWidget(idx, singleHeir);
|
|
if (qobject_cast<TermWidget*>(singleHeir))
|
|
{
|
|
nextFocus = singleHeir;
|
|
}
|
|
else
|
|
{
|
|
nextFocus = singleHeir->findChild<TermWidget*>();
|
|
}
|
|
parent->setParent(0);
|
|
delete parent;
|
|
// Make sure there's no access to the removed parent
|
|
parent = uselessSplitterParent;
|
|
}
|
|
}
|
|
|
|
if (parent->count() > 0)
|
|
{
|
|
if (nextFocus)
|
|
{
|
|
nextFocus->setFocus(Qt::OtherFocusReason);
|
|
}
|
|
else
|
|
{
|
|
parent->widget(0)->setFocus(Qt::OtherFocusReason);
|
|
}
|
|
parent->update();
|
|
}
|
|
else
|
|
emit finished();
|
|
}
|
|
|
|
void TermWidgetHolder::split(TermWidget *term, Qt::Orientation orientation)
|
|
{
|
|
QSplitter *parent = qobject_cast<QSplitter *>(term->parent());
|
|
assert(parent);
|
|
|
|
int ix = parent->indexOf(term);
|
|
QList<int> parentSizes = parent->sizes();
|
|
|
|
QList<int> sizes;
|
|
sizes << 1 << 1;
|
|
|
|
QSplitter *s = new QSplitter(orientation, this);
|
|
s->setFocusPolicy(Qt::NoFocus);
|
|
s->insertWidget(0, term);
|
|
|
|
// wdir settings
|
|
QString wd(m_wdir);
|
|
if (Properties::Instance()->useCWD)
|
|
{
|
|
wd = term->impl()->workingDirectory();
|
|
if (wd.isEmpty())
|
|
wd = m_wdir;
|
|
}
|
|
|
|
TermWidget * w = newTerm(wd);
|
|
s->insertWidget(1, w);
|
|
s->setSizes(sizes);
|
|
|
|
parent->insertWidget(ix, s);
|
|
parent->setSizes(parentSizes);
|
|
|
|
w->setFocus(Qt::OtherFocusReason);
|
|
}
|
|
|
|
TermWidget *TermWidgetHolder::newTerm(const QString & wdir, const QString & shell)
|
|
{
|
|
QString wd(wdir);
|
|
if (wd.isEmpty())
|
|
wd = m_wdir;
|
|
|
|
QString sh(shell);
|
|
if (shell.isEmpty())
|
|
sh = m_shell;
|
|
|
|
TermWidget *w = new TermWidget(wd, sh, this);
|
|
// proxy signals
|
|
connect(w, SIGNAL(renameSession()), this, SIGNAL(renameSession()));
|
|
connect(w, SIGNAL(removeCurrentSession()), this, SIGNAL(lastTerminalClosed()));
|
|
connect(w, SIGNAL(finished()), this, SLOT(handle_finished()));
|
|
// consume signals
|
|
|
|
connect(w, SIGNAL(splitHorizontal(TermWidget *)),
|
|
this, SLOT(splitHorizontal(TermWidget *)));
|
|
connect(w, SIGNAL(splitVertical(TermWidget *)),
|
|
this, SLOT(splitVertical(TermWidget *)));
|
|
connect(w, SIGNAL(splitCollapse(TermWidget *)),
|
|
this, SLOT(splitCollapse(TermWidget *)));
|
|
connect(w, SIGNAL(termGetFocus(TermWidget *)),
|
|
this, SLOT(setCurrentTerminal(TermWidget *)));
|
|
connect(w, &TermWidget::termTitleChanged, this, &TermWidgetHolder::onTermTitleChanged);
|
|
|
|
return w;
|
|
}
|
|
|
|
void TermWidgetHolder::setCurrentTerminal(TermWidget* term)
|
|
{
|
|
TermWidget * old_current = m_currentTerm;
|
|
m_currentTerm = term;
|
|
if (old_current != m_currentTerm)
|
|
{
|
|
if (m_currentTerm->impl()->isTitleChanged())
|
|
{
|
|
emit termTitleChanged(m_currentTerm->impl()->title(), m_currentTerm->impl()->icon());
|
|
} else
|
|
{
|
|
emit termTitleChanged(windowTitle(), QString{});
|
|
}
|
|
}
|
|
}
|
|
|
|
void TermWidgetHolder::handle_finished()
|
|
{
|
|
TermWidget * w = qobject_cast<TermWidget*>(sender());
|
|
if (!w)
|
|
{
|
|
qDebug() << "TermWidgetHolder::handle_finished: Unknown object to handle" << w;
|
|
assert(0);
|
|
}
|
|
splitCollapse(w);
|
|
}
|
|
|
|
void TermWidgetHolder::onTermTitleChanged(QString title, QString icon) const
|
|
{
|
|
TermWidget * term = qobject_cast<TermWidget *>(sender());
|
|
if (m_currentTerm == term)
|
|
emit termTitleChanged(title, icon);
|
|
}
|