You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
438 lines
12 KiB
438 lines
12 KiB
/* |
|
Copyright 2007-2008 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. |
|
*/ |
|
|
|
// Own |
|
#include "Part.h" |
|
|
|
// Qt |
|
#include <QStringList> |
|
#include <QDir> |
|
#include <QKeyEvent> |
|
#include <QMetaEnum> |
|
#include <QUrl> |
|
|
|
// KDE |
|
#include <QAction> |
|
#include <KActionCollection> |
|
#include <KPluginFactory> |
|
#include <KLocalizedString> |
|
#include <KConfigDialog> |
|
|
|
// Konsole |
|
#include "EditProfileDialog.h" |
|
#include "Emulation.h" |
|
#include "Session.h" |
|
#include "SessionController.h" |
|
#include "SessionManager.h" |
|
#include "ProfileManager.h" |
|
#include "TerminalDisplay.h" |
|
#include "ViewManager.h" |
|
#include "ViewContainer.h" |
|
#include "KonsoleSettings.h" |
|
#include "settings/PartInfo.h" |
|
#include "settings/ProfileSettings.h" |
|
|
|
using namespace Konsole; |
|
|
|
K_PLUGIN_FACTORY_WITH_JSON(KonsolePartFactory, |
|
"konsolepart.json", |
|
registerPlugin<Konsole::Part>();) |
|
|
|
Part::Part(QWidget *parentWidget, QObject *parent, const QVariantList &) : |
|
KParts::ReadOnlyPart(parent), |
|
_viewManager(nullptr), |
|
_pluggedController(nullptr) |
|
{ |
|
// create view widget |
|
_viewManager = new ViewManager(this, actionCollection()); |
|
_viewManager->setNavigationMethod(ViewManager::NoNavigation); |
|
|
|
connect(_viewManager, &Konsole::ViewManager::activeViewChanged, this, |
|
&Konsole::Part::activeViewChanged); |
|
connect(_viewManager, &Konsole::ViewManager::empty, this, &Konsole::Part::terminalExited); |
|
connect(_viewManager, &Konsole::ViewManager::newViewRequest, this, &Konsole::Part::newTab); |
|
|
|
_viewManager->widget()->setParent(parentWidget); |
|
|
|
setWidget(_viewManager->widget()); |
|
actionCollection()->addAssociatedWidget(_viewManager->widget()); |
|
const QList<QAction *> actionsList = actionCollection()->actions(); |
|
for (QAction *action : actionsList) { |
|
action->setShortcutContext(Qt::WidgetWithChildrenShortcut); |
|
} |
|
|
|
// Enable translucency support. |
|
_viewManager->widget()->setAttribute(Qt::WA_TranslucentBackground, true); |
|
|
|
// create basic session |
|
createSession(); |
|
} |
|
|
|
Part::~Part() |
|
{ |
|
ProfileManager::instance()->saveSettings(); |
|
delete _viewManager; |
|
} |
|
|
|
bool Part::openFile() |
|
{ |
|
return false; |
|
} |
|
|
|
void Part::terminalExited() |
|
{ |
|
deleteLater(); |
|
} |
|
|
|
void Part::newTab() |
|
{ |
|
createSession(); |
|
} |
|
|
|
Session *Part::activeSession() const |
|
{ |
|
if (_viewManager->activeViewController() != nullptr) { |
|
Q_ASSERT(_viewManager->activeViewController()->session()); |
|
|
|
return _viewManager->activeViewController()->session(); |
|
} |
|
return nullptr; |
|
} |
|
|
|
void Part::startProgram(const QString &program, const QStringList &arguments) |
|
{ |
|
Q_ASSERT(activeSession()); |
|
|
|
// do nothing if the session has already started running |
|
if (activeSession()->isRunning()) { |
|
return; |
|
} |
|
|
|
if (!program.isEmpty() && !arguments.isEmpty()) { |
|
activeSession()->setProgram(program); |
|
activeSession()->setArguments(arguments); |
|
} |
|
|
|
activeSession()->run(); |
|
} |
|
|
|
void Part::openTeletype(int ptyMasterFd, bool runShell) |
|
{ |
|
Q_ASSERT(activeSession()); |
|
|
|
activeSession()->openTeletype(ptyMasterFd, runShell); |
|
} |
|
|
|
void Part::showShellInDir(const QString &dir) |
|
{ |
|
Q_ASSERT(activeSession()); |
|
|
|
// do nothing if the session has already started running |
|
if (activeSession()->isRunning()) { |
|
return; |
|
} |
|
|
|
// All other checking is done in setInitialWorkingDirectory() |
|
if (!dir.isEmpty()) { |
|
activeSession()->setInitialWorkingDirectory(dir); |
|
} |
|
|
|
activeSession()->run(); |
|
} |
|
|
|
void Part::sendInput(const QString &text) |
|
{ |
|
Q_ASSERT(activeSession()); |
|
activeSession()->sendTextToTerminal(text); |
|
} |
|
|
|
int Part::terminalProcessId() |
|
{ |
|
Q_ASSERT(activeSession()); |
|
|
|
return activeSession()->processId(); |
|
} |
|
|
|
int Part::foregroundProcessId() |
|
{ |
|
Q_ASSERT(activeSession()); |
|
|
|
if (activeSession()->isForegroundProcessActive()) { |
|
return activeSession()->foregroundProcessId(); |
|
} |
|
return -1; |
|
} |
|
|
|
QString Part::foregroundProcessName() |
|
{ |
|
Q_ASSERT(activeSession()); |
|
|
|
if (activeSession()->isForegroundProcessActive()) { |
|
return activeSession()->foregroundProcessName(); |
|
} |
|
return QString(); |
|
} |
|
|
|
QString Part::currentWorkingDirectory() const |
|
{ |
|
Q_ASSERT(activeSession()); |
|
|
|
return activeSession()->currentWorkingDirectory(); |
|
} |
|
|
|
#ifdef USE_TERMINALINTERFACEV2 |
|
QVariant Part::profileProperty(const QString &profileProperty) const |
|
{ |
|
const auto metaEnum = QMetaEnum::fromType<Profile::Property>(); |
|
const auto value = metaEnum.keyToValue(profileProperty.toStdString().c_str()); |
|
|
|
if (value == -1) { |
|
return QString(); |
|
} |
|
|
|
const auto p = static_cast<Profile::Property>(value); |
|
return SessionManager::instance()->sessionProfile(activeSession())->property<QVariant>(p); |
|
} |
|
|
|
QStringList Part::availableProfiles() const |
|
{ |
|
return ProfileManager::instance()->availableProfileNames(); |
|
} |
|
|
|
QString Part::currentProfileName() const |
|
{ |
|
return SessionManager::instance()->sessionProfile(activeSession())->name(); |
|
} |
|
|
|
bool Part::setCurrentProfile(const QString &profileName) |
|
{ |
|
|
|
Profile::Ptr profile; |
|
for(auto p : ProfileManager::instance()->allProfiles()) { |
|
if (p->name() == profileName) { |
|
profile = p; |
|
break; |
|
} |
|
} |
|
|
|
if (!profile) { |
|
profile = ProfileManager::instance()->loadProfile(profileName); |
|
} |
|
|
|
SessionManager::instance()->setSessionProfile(activeSession(), profile); |
|
return currentProfileName() == profileName; |
|
} |
|
#endif |
|
|
|
void Part::createSession(const QString &profileName, const QString &directory) |
|
{ |
|
Profile::Ptr profile = ProfileManager::instance()->defaultProfile(); |
|
if (!profileName.isEmpty()) { |
|
profile = ProfileManager::instance()->loadProfile(profileName); |
|
} |
|
|
|
Q_ASSERT(profile); |
|
|
|
Session *session = SessionManager::instance()->createSession(profile); |
|
|
|
// override the default directory specified in the profile |
|
if (!directory.isEmpty() && profile->startInCurrentSessionDir()) { |
|
session->setInitialWorkingDirectory(directory); |
|
} |
|
|
|
auto newView = _viewManager->createView(session); |
|
_viewManager->activeContainer()->addView(newView); |
|
} |
|
|
|
void Part::activeViewChanged(SessionController *controller) |
|
{ |
|
Q_ASSERT(controller); |
|
Q_ASSERT(controller->view()); |
|
|
|
// remove existing controller |
|
if (_pluggedController != nullptr) { |
|
removeChildClient(_pluggedController); |
|
disconnect(_pluggedController, &Konsole::SessionController::titleChanged, this, |
|
&Konsole::Part::activeViewTitleChanged); |
|
disconnect(_pluggedController, &Konsole::SessionController::currentDirectoryChanged, this, |
|
&Konsole::Part::currentDirectoryChanged); |
|
} |
|
|
|
// insert new controller |
|
insertChildClient(controller); |
|
|
|
connect(controller, &Konsole::SessionController::titleChanged, this, |
|
&Konsole::Part::activeViewTitleChanged); |
|
activeViewTitleChanged(controller); |
|
connect(controller, &Konsole::SessionController::currentDirectoryChanged, this, |
|
&Konsole::Part::currentDirectoryChanged); |
|
|
|
const char *displaySignal = SIGNAL(overrideShortcutCheck(QKeyEvent*,bool&)); |
|
const char *partSlot = SLOT(overrideTerminalShortcut(QKeyEvent*,bool&)); |
|
|
|
disconnect(controller->view(), displaySignal, this, partSlot); |
|
connect(controller->view(), displaySignal, this, partSlot); |
|
|
|
_pluggedController = controller; |
|
} |
|
|
|
void Part::overrideTerminalShortcut(QKeyEvent *event, bool &override) |
|
{ |
|
// Shift+Insert is commonly used as the alternate shortcut for |
|
// pasting in KDE apps(including konsole), so it deserves some |
|
// special treatment. |
|
if (((event->modifiers() & Qt::ShiftModifier) != 0u) |
|
&& (event->key() == Qt::Key_Insert)) { |
|
override = false; |
|
return; |
|
} |
|
|
|
// override all shortcuts in the embedded terminal by default |
|
override = true; |
|
emit overrideShortcut(event, override); |
|
} |
|
|
|
void Part::activeViewTitleChanged(ViewProperties *properties) |
|
{ |
|
emit setWindowCaption(properties->title()); |
|
} |
|
|
|
void Part::showManageProfilesDialog(QWidget *parent) |
|
{ |
|
// Make sure this string is unique among all users of this part |
|
if (KConfigDialog::showDialog(QStringLiteral("konsolepartmanageprofiles"))) { |
|
return; |
|
} |
|
|
|
KConfigDialog *settingsDialog = new KConfigDialog(parent, QStringLiteral("konsolepartmanageprofiles"), |
|
KonsoleSettings::self()); |
|
settingsDialog->setFaceType(KPageDialog::Tabbed); |
|
|
|
auto profileSettings = new ProfileSettings(settingsDialog); |
|
settingsDialog->addPage(profileSettings, |
|
i18nc("@title Preferences page name", |
|
"Profiles"), |
|
QStringLiteral("configure")); |
|
|
|
auto partInfoSettings = new PartInfoSettings(settingsDialog); |
|
settingsDialog->addPage(partInfoSettings, |
|
i18nc("@title Preferences page name", |
|
"Part Info"), |
|
QStringLiteral("dialog-information")); |
|
|
|
settingsDialog->show(); |
|
} |
|
|
|
void Part::showEditCurrentProfileDialog(QWidget *parent) |
|
{ |
|
Q_ASSERT(activeSession()); |
|
|
|
auto dialog = new EditProfileDialog(parent); |
|
dialog->setAttribute(Qt::WA_DeleteOnClose); |
|
dialog->setProfile(SessionManager::instance()->sessionProfile(activeSession())); |
|
dialog->show(); |
|
} |
|
|
|
void Part::changeSessionSettings(const QString &text) |
|
{ |
|
Q_ASSERT(activeSession()); |
|
|
|
// send a profile change command, the escape code format |
|
// is the same as the normal X-Term commands used to change the window title or icon, |
|
// but with a magic value of '50' for the parameter which specifies what to change |
|
QString command = QStringLiteral("\033]50;%1\a").arg(text); |
|
|
|
sendInput(command); |
|
} |
|
|
|
// Konqueror integration |
|
bool Part::openUrl(const QUrl &url) |
|
{ |
|
if (KParts::ReadOnlyPart::url() == url) { |
|
emit completed(); |
|
return true; |
|
} |
|
|
|
setUrl(url); |
|
emit setWindowCaption(url.toDisplayString(QUrl::PreferLocalFile)); |
|
////qDebug() << "Set Window Caption to " << url.pathOrUrl(); |
|
emit started(nullptr); |
|
|
|
if (url.isLocalFile()) { |
|
showShellInDir(url.path()); |
|
} else { |
|
showShellInDir(QDir::homePath()); |
|
} |
|
|
|
emit completed(); |
|
return true; |
|
} |
|
|
|
void Part::setMonitorSilenceEnabled(bool enabled) |
|
{ |
|
Q_ASSERT(activeSession()); |
|
|
|
if (enabled) { |
|
activeSession()->setMonitorSilence(true); |
|
connect(activeSession(), &Konsole::Session::notificationsChanged, |
|
this, &Konsole::Part::notificationChanged, |
|
Qt::UniqueConnection); |
|
} else { |
|
activeSession()->setMonitorSilence(false); |
|
if (!activeSession()->isMonitorActivity()) { |
|
disconnect(activeSession(), &Konsole::Session::notificationsChanged, |
|
this, &Konsole::Part::notificationChanged); |
|
} |
|
} |
|
} |
|
|
|
void Part::setMonitorActivityEnabled(bool enabled) |
|
{ |
|
Q_ASSERT(activeSession()); |
|
|
|
if (enabled) { |
|
activeSession()->setMonitorActivity(true); |
|
connect(activeSession(), &Konsole::Session::notificationsChanged, |
|
this, &Konsole::Part::notificationChanged, |
|
Qt::UniqueConnection); |
|
} else { |
|
activeSession()->setMonitorActivity(false); |
|
if (!activeSession()->isMonitorSilence()) { |
|
disconnect(activeSession(), &Konsole::Session::notificationsChanged, |
|
this, |
|
&Konsole::Part::notificationChanged); |
|
} |
|
} |
|
} |
|
|
|
bool Part::isBlurEnabled() |
|
{ |
|
return ViewManager::profileHasBlurEnabled(SessionManager::instance()->sessionProfile(activeSession())); |
|
} |
|
|
|
void Part::notificationChanged(Session::Notification notification, bool enabled) |
|
{ |
|
if (notification == Session::Notification::Silence && enabled) { |
|
emit silenceDetected(); |
|
} else if (notification == Session::Notification::Activity && enabled) { |
|
emit activityDetected(); |
|
} |
|
} |
|
|
|
#include "Part.moc"
|
|
|