2009-02-25 09:15:00 +01:00
|
|
|
/**************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
|
|
|
|
** Contact: Qt Software Information (qt-info@nokia.com)
|
|
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Commercial Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Licensees holding valid Qt Commercial licenses may use this file in
|
|
|
|
|
** accordance with the Qt Commercial License Agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
|
** a written agreement between you and Nokia.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** GNU Lesser General Public License Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
|
|
|
** General Public License version 2.1 as published by the Free Software
|
|
|
|
|
** Foundation and appearing in the file LICENSE.LGPL included in the
|
|
|
|
|
** packaging of this file. Please review the following information to
|
|
|
|
|
** ensure the GNU Lesser General Public License version 2.1 requirements
|
|
|
|
|
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** If you are unsure which license is appropriate for your use, please
|
|
|
|
|
** contact the sales department at qt-sales@nokia.com.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
**************************************************************************/
|
2008-12-02 15:08:31 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "debuggeroutputwindow.h"
|
2009-03-24 14:41:15 +01:00
|
|
|
#include "debuggeractions.h"
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
|
#include <QtCore/QDebug>
|
|
|
|
|
|
|
|
|
|
#include <QtGui/QAction>
|
|
|
|
|
#include <QtGui/QHBoxLayout>
|
|
|
|
|
#include <QtGui/QVBoxLayout>
|
|
|
|
|
#include <QtGui/QKeyEvent>
|
|
|
|
|
#include <QtGui/QLabel>
|
|
|
|
|
#include <QtGui/QLineEdit>
|
|
|
|
|
#include <QtGui/QMenu>
|
|
|
|
|
#include <QtGui/QSpacerItem>
|
|
|
|
|
#include <QtGui/QSplitter>
|
|
|
|
|
#include <QtGui/QTextBlock>
|
|
|
|
|
|
|
|
|
|
#ifndef GDBDEBUGGERLEAN
|
|
|
|
|
|
|
|
|
|
#include <aggregation/aggregate.h>
|
|
|
|
|
#include <find/basetextfind.h>
|
|
|
|
|
|
|
|
|
|
using namespace Find;
|
|
|
|
|
|
|
|
|
|
#endif // GDBDEBUGGERLEAN
|
|
|
|
|
|
2009-03-24 14:41:15 +01:00
|
|
|
using namespace Debugger::Internal;
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
|
|
|
|
// InputPane
|
|
|
|
|
//
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
2009-03-27 17:31:58 +01:00
|
|
|
class DebuggerPane : public QPlainTextEdit
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
DebuggerPane(QWidget *parent)
|
2009-03-27 17:31:58 +01:00
|
|
|
: QPlainTextEdit(parent)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
m_clearContentsAction = new QAction(this);
|
|
|
|
|
m_clearContentsAction->setText("Clear contents");
|
|
|
|
|
m_clearContentsAction->setEnabled(true);
|
|
|
|
|
m_clearContentsAction->setShortcut(Qt::ControlModifier + Qt::Key_R);
|
|
|
|
|
connect(m_clearContentsAction, SIGNAL(triggered(bool)),
|
|
|
|
|
parent, SLOT(clearContents()));
|
|
|
|
|
|
|
|
|
|
m_saveContentsAction = new QAction(this);
|
|
|
|
|
m_saveContentsAction->setText("Save contents");
|
|
|
|
|
m_saveContentsAction->setEnabled(true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void contextMenuEvent(QContextMenuEvent *ev)
|
|
|
|
|
{
|
|
|
|
|
QMenu *menu = createStandardContextMenu();
|
|
|
|
|
menu->addAction(m_clearContentsAction);
|
|
|
|
|
//menu->addAction(m_saveContentsAction);
|
|
|
|
|
addContextActions(menu);
|
2009-04-03 15:46:40 +02:00
|
|
|
theDebuggerAction(ExecuteCommand)->setData(textCursor().block().text());
|
|
|
|
|
menu->addAction(theDebuggerAction(ExecuteCommand));
|
2009-03-24 14:41:15 +01:00
|
|
|
menu->addSeparator();
|
|
|
|
|
menu->addAction(theDebuggerAction(SettingsDialog));
|
2008-12-02 12:01:29 +01:00
|
|
|
menu->exec(ev->globalPos());
|
|
|
|
|
delete menu;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
virtual void addContextActions(QMenu *) {}
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
QAction *m_clearContentsAction;
|
|
|
|
|
QAction *m_saveContentsAction;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
class InputPane : public DebuggerPane
|
|
|
|
|
{
|
|
|
|
|
Q_OBJECT
|
|
|
|
|
public:
|
2009-04-03 15:46:40 +02:00
|
|
|
InputPane(QWidget *parent)
|
|
|
|
|
: DebuggerPane(parent)
|
|
|
|
|
{}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
|
signals:
|
|
|
|
|
void clearContentsRequested();
|
|
|
|
|
void statusMessageRequested(const QString &, int);
|
|
|
|
|
void commandSelected(int);
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
void keyPressEvent(QKeyEvent *ev)
|
|
|
|
|
{
|
|
|
|
|
if (ev->modifiers() == Qt::ControlModifier && ev->key() == Qt::Key_Return)
|
2009-04-03 15:46:40 +02:00
|
|
|
theDebuggerAction(ExecuteCommand)->trigger(textCursor().block().text());
|
2008-12-02 12:01:29 +01:00
|
|
|
else if (ev->modifiers() == Qt::ControlModifier && ev->key() == Qt::Key_R)
|
|
|
|
|
emit clearContentsRequested();
|
|
|
|
|
else
|
2009-03-27 17:31:58 +01:00
|
|
|
QPlainTextEdit::keyPressEvent(ev);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void mouseDoubleClickEvent(QMouseEvent *ev)
|
|
|
|
|
{
|
|
|
|
|
QString line = cursorForPosition(ev->pos()).block().text();
|
|
|
|
|
int n = 0;
|
|
|
|
|
|
|
|
|
|
// cut time string
|
|
|
|
|
if (line.size() > 18 && line.at(0) == '[')
|
|
|
|
|
line = line.mid(18);
|
|
|
|
|
//qDebug() << line;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i != line.size(); ++i) {
|
|
|
|
|
QChar c = line.at(i);
|
|
|
|
|
if (!c.isDigit())
|
|
|
|
|
break;
|
|
|
|
|
n = 10 * n + c.unicode() - '0';
|
|
|
|
|
}
|
|
|
|
|
emit commandSelected(n);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void addContextActions(QMenu *menu)
|
|
|
|
|
{
|
2009-04-03 15:46:40 +02:00
|
|
|
menu->addAction(theDebuggerAction(ExecuteCommand));
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void focusInEvent(QFocusEvent *ev)
|
|
|
|
|
{
|
|
|
|
|
emit statusMessageRequested("Type Ctrl-<Return> to execute a line.", -1);
|
2009-03-27 17:31:58 +01:00
|
|
|
QPlainTextEdit::focusInEvent(ev);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void focusOutEvent(QFocusEvent *ev)
|
|
|
|
|
{
|
|
|
|
|
emit statusMessageRequested(QString(), -1);
|
2009-03-27 17:31:58 +01:00
|
|
|
QPlainTextEdit::focusOutEvent(ev);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
|
|
|
|
// CombinedPane
|
|
|
|
|
//
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
class CombinedPane : public DebuggerPane
|
|
|
|
|
{
|
|
|
|
|
Q_OBJECT
|
|
|
|
|
public:
|
|
|
|
|
CombinedPane(QWidget *parent)
|
|
|
|
|
: DebuggerPane(parent)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
public slots:
|
|
|
|
|
void gotoResult(int i)
|
|
|
|
|
{
|
|
|
|
|
QString needle = QString::number(i) + '^';
|
|
|
|
|
QString needle2 = "stdout:" + needle;
|
|
|
|
|
QTextCursor cursor(document());
|
|
|
|
|
do {
|
|
|
|
|
const QString line = cursor.block().text();
|
|
|
|
|
if (line.startsWith(needle) || line.startsWith(needle2)) {
|
|
|
|
|
setFocus();
|
|
|
|
|
setTextCursor(cursor);
|
|
|
|
|
ensureCursorVisible();
|
|
|
|
|
cursor.movePosition(QTextCursor::Down, QTextCursor::KeepAnchor);
|
|
|
|
|
setTextCursor(cursor);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
} while (cursor.movePosition(QTextCursor::Down));
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
|
|
|
|
// DebuggerOutputWindow
|
|
|
|
|
//
|
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
DebuggerOutputWindow::DebuggerOutputWindow(QWidget *parent)
|
|
|
|
|
: QWidget(parent)
|
|
|
|
|
{
|
|
|
|
|
setWindowTitle(tr("Gdb"));
|
|
|
|
|
|
|
|
|
|
QSplitter *m_splitter = new QSplitter(Qt::Horizontal, this);
|
|
|
|
|
// mixed input/output
|
|
|
|
|
m_combinedText = new CombinedPane(this);
|
|
|
|
|
m_combinedText->setReadOnly(true);
|
|
|
|
|
m_combinedText->setReadOnly(false);
|
|
|
|
|
m_combinedText->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
|
|
|
|
|
|
|
|
|
|
// input only
|
|
|
|
|
m_inputText = new InputPane(this);
|
|
|
|
|
m_inputText->setReadOnly(false);
|
|
|
|
|
m_inputText->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
|
|
|
|
|
|
|
|
|
|
m_splitter->addWidget(m_inputText);
|
|
|
|
|
m_splitter->addWidget(m_combinedText);
|
|
|
|
|
|
|
|
|
|
QGridLayout *layout = new QGridLayout(this);
|
|
|
|
|
layout->setMargin(0);
|
|
|
|
|
layout->addWidget(m_splitter);
|
|
|
|
|
setLayout(layout);
|
|
|
|
|
|
|
|
|
|
#ifndef GDBDEBUGGERLEAN
|
|
|
|
|
Aggregation::Aggregate *aggregate = new Aggregation::Aggregate;
|
|
|
|
|
aggregate->add(m_combinedText);
|
|
|
|
|
aggregate->add(new BaseTextFind(m_combinedText));
|
|
|
|
|
|
|
|
|
|
aggregate = new Aggregation::Aggregate;
|
|
|
|
|
aggregate->add(m_inputText);
|
|
|
|
|
aggregate->add(new BaseTextFind(m_inputText));
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
connect(m_inputText, SIGNAL(statusMessageRequested(QString,int)),
|
|
|
|
|
this, SIGNAL(statusMessageRequested(QString,int)));
|
|
|
|
|
connect(m_inputText, SIGNAL(commandSelected(int)),
|
|
|
|
|
m_combinedText, SLOT(gotoResult(int)));
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void DebuggerOutputWindow::showOutput(const QString &prefix, const QString &output)
|
|
|
|
|
{
|
|
|
|
|
if (output.isEmpty())
|
|
|
|
|
return;
|
|
|
|
|
foreach (QString line, output.split("\n")) {
|
|
|
|
|
// FIXME: QTextEdit asserts on really long lines...
|
|
|
|
|
const int n = 3000;
|
|
|
|
|
if (line.size() > n)
|
|
|
|
|
line = line.left(n) + " [...] <cut off>";
|
2009-03-27 17:31:58 +01:00
|
|
|
m_combinedText->appendPlainText(prefix + line);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
QTextCursor cursor = m_combinedText->textCursor();
|
|
|
|
|
cursor.movePosition(QTextCursor::End);
|
|
|
|
|
m_combinedText->setTextCursor(cursor);
|
|
|
|
|
m_combinedText->ensureCursorVisible();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DebuggerOutputWindow::showInput(const QString &prefix, const QString &input)
|
|
|
|
|
{
|
2008-12-05 18:27:45 +01:00
|
|
|
Q_UNUSED(prefix);
|
2009-03-27 17:31:58 +01:00
|
|
|
m_inputText->appendPlainText(input);
|
2008-12-02 12:01:29 +01:00
|
|
|
QTextCursor cursor = m_inputText->textCursor();
|
|
|
|
|
cursor.movePosition(QTextCursor::End);
|
|
|
|
|
m_inputText->setTextCursor(cursor);
|
|
|
|
|
m_inputText->ensureCursorVisible();
|
|
|
|
|
showOutput("input:", input);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DebuggerOutputWindow::clearContents()
|
|
|
|
|
{
|
|
|
|
|
m_combinedText->clear();
|
|
|
|
|
m_inputText->clear();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DebuggerOutputWindow::setCursor(const QCursor &cursor)
|
|
|
|
|
{
|
|
|
|
|
m_combinedText->setCursor(cursor);
|
|
|
|
|
m_inputText->setCursor(cursor);
|
|
|
|
|
QWidget::setCursor(cursor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString DebuggerOutputWindow::combinedContents() const
|
|
|
|
|
{
|
|
|
|
|
return m_combinedText->toPlainText();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString DebuggerOutputWindow::inputContents() const
|
|
|
|
|
{
|
|
|
|
|
return m_inputText->toPlainText();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#include "debuggeroutputwindow.moc"
|