/***************************************************************************
- * Copyright (C) 2007 by Peter Penz <peter.penz@gmx.at> *
+ * Copyright (C) 2007-2010 by Peter Penz <peter.penz@gmx.at> *
* *
* 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 *
#include <QBoxLayout>
#include <QShowEvent>
-#include <kdebug.h>
-
TerminalPanel::TerminalPanel(QWidget* parent) :
Panel(parent),
+ m_clearTerminal(true),
+ m_mostLocalUrlJob(0),
m_layout(0),
m_terminal(0),
m_terminalWidget(0)
return size;
}
-void TerminalPanel::setUrl(const KUrl& url)
+void TerminalPanel::terminalExited()
{
- if (!url.isValid() || (url == Panel::url())) {
- return;
- }
+ emit hideTerminalPanel();
+ m_terminal = 0;
+}
- Panel::setUrl(url);
+bool TerminalPanel::urlChanged()
+{
+ if (!url().isValid()) {
+ return false;
+ }
const bool sendInput = (m_terminal != 0)
&& (m_terminal->foregroundProcessId() == -1)
&& isVisible();
if (sendInput) {
- cdUrl(url);
+ changeDir(url());
}
-}
-void TerminalPanel::terminalExited()
-{
- emit hideTerminalPanel();
- m_terminal = 0;
+ return true;
}
void TerminalPanel::showEvent(QShowEvent* event)
}
if (m_terminal == 0) {
+ m_clearTerminal = true;
KPluginFactory* factory = KPluginLoader("libkonsolepart").factory();
KParts::ReadOnlyPart* part = factory ? (factory->create<KParts::ReadOnlyPart>(this)) : 0;
if (part != 0) {
}
if (m_terminal != 0) {
m_terminal->showShellInDir(url().toLocalFile());
- cdUrl(url());
- m_terminal->sendInput("clear\n"); // TODO do clear after slotMostLocalUrlResult is called, for remote dirs?
+ changeDir(url());
m_terminalWidget->setFocus();
}
Panel::showEvent(event);
}
-void TerminalPanel::cdUrl(const KUrl& url)
+void TerminalPanel::changeDir(const KUrl& url)
{
+ delete m_mostLocalUrlJob;
+ m_mostLocalUrlJob = 0;
+
if (url.isLocalFile()) {
- cdDirectory(url.toLocalFile());
+ sendCdToTerminal(url.toLocalFile());
} else {
- KIO::StatJob* job = KIO::mostLocalUrl(url, KIO::HideProgressInfo);
- job->ui()->setWindow(this);
- connect(job, SIGNAL(result(KJob*)), this, SLOT(slotMostLocalUrlResult(KJob*)));
+ m_mostLocalUrlJob = KIO::mostLocalUrl(url, KIO::HideProgressInfo);
+ m_mostLocalUrlJob->ui()->setWindow(this);
+ connect(m_mostLocalUrlJob, SIGNAL(result(KJob*)), this, SLOT(slotMostLocalUrlResult(KJob*)));
}
}
-void TerminalPanel::cdDirectory(const QString& dir)
+void TerminalPanel::sendCdToTerminal(const QString& dir)
{
+ if (!m_clearTerminal) {
+ // The TerminalV2 interface does not provide a way to delete the
+ // current line before sending a new input. This is mandatory,
+ // otherwise sending a 'cd x' to a existing 'rm -rf *' might
+ // result in data loss. As workaround Ctrl+C is send.
+ QString cancel;
+ cancel.append(QChar(3));
+ cancel.append(QChar('c'));
+ m_terminal->sendInput(cancel);
+ }
+
m_terminal->sendInput("cd " + KShell::quoteArg(dir) + '\n');
+
+ if (m_clearTerminal) {
+ m_terminal->sendInput("clear\n");
+ m_clearTerminal = false;
+ }
}
void TerminalPanel::slotMostLocalUrlResult(KJob* job)
KIO::StatJob* statJob = static_cast<KIO::StatJob *>(job);
const KUrl url = statJob->mostLocalUrl();
if (url.isLocalFile()) {
- cdDirectory(url.toLocalFile());
+ sendCdToTerminal(url.toLocalFile());
}
+
+ m_mostLocalUrlJob = 0;
}
#include "terminalpanel.moc"