forked from qt-creator/qt-creator
Show only the Enabled field for BreakpointAtJavaScriptThrow type. Change-Id: Ib4908b4a7826329a98105112c612963624586115 Reviewed-by: hjk <qthjk@ovi.com>
748 lines
28 KiB
C++
748 lines
28 KiB
C++
/**************************************************************************
|
|
**
|
|
** This file is part of Qt Creator
|
|
**
|
|
** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies).
|
|
**
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
|
**
|
|
**
|
|
** GNU Lesser General Public License Usage
|
|
**
|
|
** 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.
|
|
**
|
|
** In addition, as a special exception, Nokia gives you certain additional
|
|
** rights. These rights are described in the Nokia Qt LGPL Exception
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
**
|
|
** Other Usage
|
|
**
|
|
** Alternatively, this file may be used in accordance with the terms and
|
|
** conditions contained in a signed written agreement between you and Nokia.
|
|
**
|
|
** If you have questions regarding the use of this file, please contact
|
|
** Nokia at qt-info@nokia.com.
|
|
**
|
|
**************************************************************************/
|
|
|
|
#include "breakwindow.h"
|
|
#include "debuggerinternalconstants.h"
|
|
#include "breakhandler.h"
|
|
#include "debuggerengine.h"
|
|
#include "debuggeractions.h"
|
|
#include "debuggercore.h"
|
|
#include "ui_breakpoint.h"
|
|
#include "ui_breakcondition.h"
|
|
|
|
#include <utils/pathchooser.h>
|
|
#include <utils/qtcassert.h>
|
|
#include <utils/savedaction.h>
|
|
|
|
#include <QtCore/QDebug>
|
|
|
|
#include <QtGui/QAction>
|
|
#include <QtGui/QIntValidator>
|
|
#include <QtGui/QKeyEvent>
|
|
#include <QtGui/QMenu>
|
|
|
|
namespace Debugger {
|
|
namespace Internal {
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
//
|
|
// BreakpointDialog: Show a dialog for editing breakpoints. Shows controls
|
|
// for the file-and-line, function and address parameters depending on the
|
|
// breakpoint type. The controls not applicable to the current type
|
|
// (say function name for file-and-line) are disabled and cleared out.
|
|
// However,the values are saved and restored once the respective mode
|
|
// is again chosen, which is done using m_savedParameters and
|
|
// setters/getters taking the parts mask enumeration parameter.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
class BreakpointDialog : public QDialog
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
explicit BreakpointDialog(unsigned engineCapabilities, QWidget *parent = 0);
|
|
bool showDialog(BreakpointParameters *data, BreakpointParts *parts);
|
|
|
|
void setParameters(const BreakpointParameters &data);
|
|
BreakpointParameters parameters() const;
|
|
|
|
public slots:
|
|
void typeChanged(int index);
|
|
|
|
private:
|
|
void setPartsEnabled(unsigned partsMask);
|
|
void clearOtherParts(unsigned partsMask);
|
|
void getParts(unsigned partsMask, BreakpointParameters *data) const;
|
|
void setParts(unsigned partsMask, const BreakpointParameters &data);
|
|
|
|
void setType(BreakpointType type);
|
|
BreakpointType type() const;
|
|
|
|
unsigned m_enabledParts;
|
|
Ui::BreakpointDialog m_ui;
|
|
BreakpointParameters m_savedParameters;
|
|
BreakpointType m_previousType;
|
|
bool m_firstTypeChange;
|
|
};
|
|
|
|
BreakpointDialog::BreakpointDialog(unsigned engineCapabilities, QWidget *parent)
|
|
: QDialog(parent), m_enabledParts(-1), m_previousType(UnknownType),
|
|
m_firstTypeChange(true)
|
|
{
|
|
m_ui.setupUi(this);
|
|
m_ui.comboBoxType->setMaxVisibleItems(20);
|
|
if (!(engineCapabilities & BreakConditionCapability))
|
|
m_enabledParts &= ~ConditionPart;
|
|
if (!(engineCapabilities & BreakModuleCapability))
|
|
m_enabledParts &= ~ModulePart;
|
|
if (!(engineCapabilities & TracePointCapability))
|
|
m_enabledParts &= ~TracePointPart;
|
|
// Match BreakpointType (omitting unknown type).
|
|
QStringList types;
|
|
types << tr("File name and line number")
|
|
<< tr("Function name")
|
|
<< tr("Break on memory address")
|
|
<< tr("Break when C++ exception is thrown")
|
|
<< tr("Break when C++ exception is caught")
|
|
<< tr("Break when function \"main\" starts")
|
|
<< tr("Break when a new process is forked")
|
|
<< tr("Break when a new process is executed")
|
|
<< tr("Break when a system call is executed")
|
|
<< tr("Break on data access at fixed address")
|
|
<< tr("Break on data access at address given by expression")
|
|
<< tr("Break on QML signal handler")
|
|
<< tr("Break when JavaScript exception is thrown");
|
|
|
|
QTC_ASSERT(types.size() == BreakpointAtJavaScriptThrow, return; )
|
|
m_ui.comboBoxType->addItems(types);
|
|
m_ui.pathChooserFileName->setExpectedKind(Utils::PathChooser::File);
|
|
connect(m_ui.comboBoxType, SIGNAL(activated(int)), SLOT(typeChanged(int)));
|
|
const QString moduleToolTip =
|
|
tr("Specifying the module (base name of the library or executable)\n"
|
|
"for function or file type breakpoints can significantly speed up\n"
|
|
"debugger start-up times (CDB, LLDB).");
|
|
m_ui.labelModule->setToolTip(moduleToolTip);
|
|
m_ui.lineEditModule->setToolTip(moduleToolTip);
|
|
const QString commandToolTip =
|
|
tr("Debugger command to be executed when the breakpoint is hit.\n"
|
|
"GDB allows for specifying a sequence of commands separated by "
|
|
"the delimiter '\\n'.");
|
|
m_ui.lineEditCommand->setToolTip(commandToolTip);
|
|
m_ui.labelCommand->setToolTip(commandToolTip);
|
|
m_ui.spinBoxIgnoreCount->setMinimum(0);
|
|
m_ui.spinBoxIgnoreCount->setMaximum(2147483647);
|
|
const QString pathToolTip =
|
|
tr("<html><head/><body><p>Determines how the path is specified "
|
|
"when setting breakpoints:</p><ul>"
|
|
"<li><i>Use Engine Default</i>: Preferred setting of the "
|
|
"debugger engine.</li>"
|
|
"<li><i>Use Full Path</i>: Pass full path, avoiding ambiguities "
|
|
"should files of the same name exist in several modules. "
|
|
"This is the engine default for CDB and LLDB.</li>"
|
|
"<li><i>Use File Name</i>: Pass the file name only. This is "
|
|
"useful when using a source tree whose location does "
|
|
"not match the one used when building the modules. "
|
|
"It is the engine default for GDB as using full paths can "
|
|
"be slow with this engine.</li>"
|
|
"</ul></body></html>");
|
|
m_ui.labelUseFullPath->setToolTip(pathToolTip);
|
|
m_ui.comboBoxPathUsage->setToolTip(pathToolTip);
|
|
}
|
|
|
|
void BreakpointDialog::setType(BreakpointType type)
|
|
{
|
|
const int comboIndex = type - 1; // Skip UnknownType.
|
|
if (comboIndex != m_ui.comboBoxType->currentIndex() || m_firstTypeChange) {
|
|
m_ui.comboBoxType->setCurrentIndex(comboIndex);
|
|
typeChanged(comboIndex);
|
|
m_firstTypeChange = false;
|
|
}
|
|
}
|
|
|
|
BreakpointType BreakpointDialog::type() const
|
|
{
|
|
const int type = m_ui.comboBoxType->currentIndex() + 1; // Skip unknown type.
|
|
return static_cast<BreakpointType>(type);
|
|
}
|
|
|
|
void BreakpointDialog::setParameters(const BreakpointParameters &data)
|
|
{
|
|
m_savedParameters = data;
|
|
setType(data.type);
|
|
setParts(AllParts, data);
|
|
}
|
|
|
|
BreakpointParameters BreakpointDialog::parameters() const
|
|
{
|
|
BreakpointParameters data(type());
|
|
getParts(AllParts, &data);
|
|
return data;
|
|
}
|
|
|
|
void BreakpointDialog::setPartsEnabled(unsigned partsMask)
|
|
{
|
|
partsMask &= m_enabledParts;
|
|
m_ui.labelFileName->setEnabled(partsMask & FileAndLinePart);
|
|
m_ui.pathChooserFileName->setEnabled(partsMask & FileAndLinePart);
|
|
m_ui.labelLineNumber->setEnabled(partsMask & FileAndLinePart);
|
|
m_ui.lineEditLineNumber->setEnabled(partsMask & FileAndLinePart);
|
|
m_ui.labelUseFullPath->setEnabled(partsMask & FileAndLinePart);
|
|
m_ui.comboBoxPathUsage->setEnabled(partsMask & FileAndLinePart);
|
|
|
|
|
|
m_ui.labelFunction->setEnabled(partsMask & FunctionPart);
|
|
m_ui.lineEditFunction->setEnabled(partsMask & FunctionPart);
|
|
|
|
m_ui.labelAddress->setEnabled(partsMask & AddressPart);
|
|
m_ui.lineEditAddress->setEnabled(partsMask & AddressPart);
|
|
m_ui.labelExpression->setEnabled(partsMask & ExpressionPart);
|
|
m_ui.lineEditExpression->setEnabled(partsMask & ExpressionPart);
|
|
|
|
m_ui.labelCondition->setEnabled(partsMask & ConditionPart);
|
|
m_ui.lineEditCondition->setEnabled(partsMask & ConditionPart);
|
|
m_ui.labelIgnoreCount->setEnabled(partsMask & IgnoreCountPart);
|
|
m_ui.spinBoxIgnoreCount->setEnabled(partsMask & IgnoreCountPart);
|
|
m_ui.labelThreadSpec->setEnabled(partsMask & ThreadSpecPart);
|
|
m_ui.lineEditThreadSpec->setEnabled(partsMask & ThreadSpecPart);
|
|
|
|
m_ui.labelModule->setEnabled(partsMask & ModulePart);
|
|
m_ui.lineEditModule->setEnabled(partsMask & ModulePart);
|
|
|
|
m_ui.labelTracepoint->setEnabled(partsMask & TracePointPart);
|
|
m_ui.checkBoxTracepoint->setEnabled(partsMask & TracePointPart);
|
|
|
|
m_ui.labelCommand->setEnabled(partsMask & TracePointPart);
|
|
m_ui.lineEditCommand->setEnabled(partsMask & TracePointPart);
|
|
|
|
m_ui.labelMessage->setEnabled(partsMask & TracePointPart);
|
|
m_ui.lineEditMessage->setEnabled(partsMask & TracePointPart);
|
|
}
|
|
|
|
void BreakpointDialog::clearOtherParts(unsigned partsMask)
|
|
{
|
|
const unsigned invertedPartsMask = ~partsMask;
|
|
if (invertedPartsMask & FileAndLinePart) {
|
|
m_ui.pathChooserFileName->setPath(QString());
|
|
m_ui.lineEditLineNumber->clear();
|
|
m_ui.comboBoxPathUsage->setCurrentIndex(BreakpointPathUsageEngineDefault);
|
|
}
|
|
|
|
if (invertedPartsMask & FunctionPart)
|
|
m_ui.lineEditFunction->clear();
|
|
|
|
if (invertedPartsMask & AddressPart)
|
|
m_ui.lineEditAddress->clear();
|
|
if (invertedPartsMask & ExpressionPart)
|
|
m_ui.lineEditExpression->clear();
|
|
|
|
if (invertedPartsMask & ConditionPart)
|
|
m_ui.lineEditCondition->clear();
|
|
if (invertedPartsMask & IgnoreCountPart)
|
|
m_ui.spinBoxIgnoreCount->clear();
|
|
if (invertedPartsMask & ThreadSpecPart)
|
|
m_ui.lineEditThreadSpec->clear();
|
|
if (invertedPartsMask & ModulePart)
|
|
m_ui.lineEditModule->clear();
|
|
|
|
if (invertedPartsMask & TracePointPart) {
|
|
m_ui.checkBoxTracepoint->setChecked(false);
|
|
m_ui.lineEditCommand->clear();
|
|
m_ui.lineEditMessage->clear();
|
|
}
|
|
}
|
|
|
|
void BreakpointDialog::getParts(unsigned partsMask, BreakpointParameters *data) const
|
|
{
|
|
data->enabled = m_ui.checkBoxEnabled->isChecked();
|
|
|
|
if (partsMask & FileAndLinePart) {
|
|
data->lineNumber = m_ui.lineEditLineNumber->text().toInt();
|
|
data->pathUsage = static_cast<BreakpointPathUsage>(m_ui.comboBoxPathUsage->currentIndex());
|
|
data->fileName = m_ui.pathChooserFileName->path();
|
|
}
|
|
if (partsMask & FunctionPart)
|
|
data->functionName = m_ui.lineEditFunction->text();
|
|
|
|
if (partsMask & AddressPart)
|
|
data->address = m_ui.lineEditAddress->text().toULongLong(0, 0);
|
|
if (partsMask & ExpressionPart)
|
|
data->expression = m_ui.lineEditExpression->text().toUtf8();
|
|
|
|
if (partsMask & ConditionPart)
|
|
data->condition = m_ui.lineEditCondition->text().toUtf8();
|
|
if (partsMask & IgnoreCountPart)
|
|
data->ignoreCount = m_ui.spinBoxIgnoreCount->text().toInt();
|
|
if (partsMask & ThreadSpecPart)
|
|
data->threadSpec =
|
|
BreakHandler::threadSpecFromDisplay(m_ui.lineEditThreadSpec->text());
|
|
if (partsMask & ModulePart)
|
|
data->module = m_ui.lineEditModule->text();
|
|
|
|
if (partsMask & TracePointPart) {
|
|
data->tracepoint = m_ui.checkBoxTracepoint->isChecked();
|
|
data->command = m_ui.lineEditCommand->text().trimmed();
|
|
data->message = m_ui.lineEditMessage->text();
|
|
}
|
|
}
|
|
|
|
void BreakpointDialog::setParts(unsigned mask, const BreakpointParameters &data)
|
|
{
|
|
m_ui.checkBoxEnabled->setChecked(data.enabled);
|
|
m_ui.comboBoxPathUsage->setCurrentIndex(data.pathUsage);
|
|
m_ui.lineEditCommand->setText(data.command);
|
|
m_ui.lineEditMessage->setText(data.message);
|
|
|
|
if (mask & FileAndLinePart) {
|
|
m_ui.pathChooserFileName->setPath(data.fileName);
|
|
m_ui.lineEditLineNumber->setText(QString::number(data.lineNumber));
|
|
}
|
|
|
|
if (mask & FunctionPart)
|
|
m_ui.lineEditFunction->setText(data.functionName);
|
|
|
|
if (mask & AddressPart) {
|
|
if (data.address) {
|
|
m_ui.lineEditAddress->setText(
|
|
QString::fromAscii("0x%1").arg(data.address, 0, 16));
|
|
} else {
|
|
m_ui.lineEditAddress->clear();
|
|
}
|
|
}
|
|
|
|
if (mask & ExpressionPart) {
|
|
if (!data.expression.isEmpty()) {
|
|
m_ui.lineEditExpression->setText(data.expression);
|
|
} else {
|
|
m_ui.lineEditExpression->clear();
|
|
}
|
|
}
|
|
|
|
if (mask & ConditionPart)
|
|
m_ui.lineEditCondition->setText(QString::fromUtf8(data.condition));
|
|
if (mask & IgnoreCountPart)
|
|
m_ui.spinBoxIgnoreCount->setValue(data.ignoreCount);
|
|
if (mask & ThreadSpecPart)
|
|
m_ui.lineEditThreadSpec->
|
|
setText(BreakHandler::displayFromThreadSpec(data.threadSpec));
|
|
if (mask & ModulePart)
|
|
m_ui.lineEditModule->setText(data.module);
|
|
|
|
if (mask & TracePointPart)
|
|
m_ui.checkBoxTracepoint->setChecked(data.tracepoint);
|
|
}
|
|
|
|
void BreakpointDialog::typeChanged(int)
|
|
{
|
|
BreakpointType previousType = m_previousType;
|
|
const BreakpointType newType = type();
|
|
m_previousType = newType;
|
|
// Save current state.
|
|
switch(previousType) {
|
|
case UnknownType:
|
|
break;
|
|
case BreakpointByFileAndLine:
|
|
getParts(FileAndLinePart|ModulePart|AllConditionParts|TracePointPart, &m_savedParameters);
|
|
break;
|
|
case BreakpointByFunction:
|
|
getParts(FunctionPart|ModulePart|AllConditionParts|TracePointPart, &m_savedParameters);
|
|
break;
|
|
case BreakpointAtThrow:
|
|
case BreakpointAtCatch:
|
|
case BreakpointAtMain:
|
|
case BreakpointAtFork:
|
|
case BreakpointAtExec:
|
|
//case BreakpointAtVFork:
|
|
case BreakpointAtSysCall:
|
|
case BreakpointAtJavaScriptThrow:
|
|
break;
|
|
case BreakpointByAddress:
|
|
case WatchpointAtAddress:
|
|
getParts(AddressPart|AllConditionParts|TracePointPart, &m_savedParameters);
|
|
break;
|
|
case WatchpointAtExpression:
|
|
getParts(ExpressionPart|AllConditionParts|TracePointPart, &m_savedParameters);
|
|
break;
|
|
case BreakpointOnQmlSignalHandler:
|
|
getParts(FunctionPart, &m_savedParameters);
|
|
}
|
|
|
|
// Enable and set up new state from saved values.
|
|
switch (newType) {
|
|
case UnknownType:
|
|
break;
|
|
case BreakpointByFileAndLine:
|
|
setParts(FileAndLinePart|AllConditionParts|ModulePart|TracePointPart, m_savedParameters);
|
|
setPartsEnabled(FileAndLinePart|AllConditionParts|ModulePart|TracePointPart);
|
|
clearOtherParts(FileAndLinePart|AllConditionParts|ModulePart|TracePointPart);
|
|
break;
|
|
case BreakpointByFunction:
|
|
setParts(FunctionPart|AllConditionParts|ModulePart|TracePointPart, m_savedParameters);
|
|
setPartsEnabled(FunctionPart|AllConditionParts|ModulePart|TracePointPart);
|
|
clearOtherParts(FunctionPart|AllConditionParts|ModulePart|TracePointPart);
|
|
break;
|
|
case BreakpointAtThrow:
|
|
case BreakpointAtCatch:
|
|
case BreakpointAtFork:
|
|
case BreakpointAtExec:
|
|
//case BreakpointAtVFork:
|
|
case BreakpointAtSysCall:
|
|
clearOtherParts(AllConditionParts|ModulePart|TracePointPart);
|
|
setPartsEnabled(AllConditionParts|TracePointPart);
|
|
break;
|
|
case BreakpointAtJavaScriptThrow:
|
|
clearOtherParts(AllParts);
|
|
setPartsEnabled(0);
|
|
break;
|
|
case BreakpointAtMain:
|
|
m_ui.lineEditFunction->setText(QLatin1String("main")); // Just for display
|
|
clearOtherParts(0);
|
|
setPartsEnabled(0);
|
|
break;
|
|
case BreakpointByAddress:
|
|
case WatchpointAtAddress:
|
|
setParts(AddressPart|AllConditionParts|TracePointPart, m_savedParameters);
|
|
setPartsEnabled(AddressPart|AllConditionParts|TracePointPart|TracePointPart);
|
|
clearOtherParts(AddressPart|AllConditionParts|TracePointPart);
|
|
break;
|
|
case WatchpointAtExpression:
|
|
setParts(ExpressionPart|AllConditionParts|TracePointPart, m_savedParameters);
|
|
setPartsEnabled(ExpressionPart|AllConditionParts|TracePointPart|TracePointPart);
|
|
clearOtherParts(ExpressionPart|AllConditionParts|TracePointPart);
|
|
break;
|
|
case BreakpointOnQmlSignalHandler:
|
|
setParts(FunctionPart, m_savedParameters);
|
|
setPartsEnabled(FunctionPart);
|
|
clearOtherParts(FunctionPart);
|
|
}
|
|
}
|
|
|
|
bool BreakpointDialog::showDialog(BreakpointParameters *data,
|
|
BreakpointParts *parts)
|
|
{
|
|
setParameters(*data);
|
|
if (exec() != QDialog::Accepted)
|
|
return false;
|
|
|
|
// Check if changed.
|
|
const BreakpointParameters newParameters = parameters();
|
|
*parts = data->differencesTo(newParameters);
|
|
if (!*parts)
|
|
return false;
|
|
|
|
*data = newParameters;
|
|
return true;
|
|
}
|
|
|
|
// Dialog allowing changing properties of multiple breakpoints at a time.
|
|
class MultiBreakPointsDialog : public QDialog
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
MultiBreakPointsDialog(unsigned engineCapabilities, QWidget *parent = 0);
|
|
|
|
QString condition() const { return m_ui.lineEditCondition->text(); }
|
|
int ignoreCount() const { return m_ui.spinBoxIgnoreCount->value(); }
|
|
int threadSpec() const
|
|
{ return BreakHandler::threadSpecFromDisplay(m_ui.lineEditThreadSpec->text()); }
|
|
|
|
void setCondition(const QString &c) { m_ui.lineEditCondition->setText(c); }
|
|
void setIgnoreCount(int i) { m_ui.spinBoxIgnoreCount->setValue(i); }
|
|
void setThreadSpec(int t)
|
|
{ return m_ui.lineEditThreadSpec->setText(BreakHandler::displayFromThreadSpec(t)); }
|
|
|
|
private:
|
|
Ui::BreakCondition m_ui;
|
|
};
|
|
|
|
MultiBreakPointsDialog::MultiBreakPointsDialog(unsigned engineCapabilities, QWidget *parent) :
|
|
QDialog(parent)
|
|
{
|
|
setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
|
|
m_ui.setupUi(this);
|
|
setWindowTitle(tr("Edit Breakpoint Properties"));
|
|
m_ui.spinBoxIgnoreCount->setMinimum(0);
|
|
m_ui.spinBoxIgnoreCount->setMaximum(2147483647);
|
|
if (!(engineCapabilities & BreakConditionCapability)) {
|
|
m_ui.labelCondition->setEnabled(false);
|
|
m_ui.lineEditCondition->setEnabled(false);
|
|
}
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
//
|
|
// BreakWindow
|
|
//
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
BreakWindow::BreakWindow(QWidget *parent)
|
|
: BaseWindow(parent)
|
|
{
|
|
setWindowTitle(tr("Breakpoints"));
|
|
setObjectName(QLatin1String("ThreadsWindow"));
|
|
setWindowIcon(QIcon(QLatin1String(":/debugger/images/debugger_breakpoints.png")));
|
|
setSelectionMode(QAbstractItemView::ExtendedSelection);
|
|
setAlwaysAdjustColumnsAction(debuggerCore()->action(AlwaysAdjustBreakpointsColumnWidths));
|
|
connect(debuggerCore()->action(UseAddressInBreakpointsView),
|
|
SIGNAL(toggled(bool)), SLOT(showAddressColumn(bool)));
|
|
}
|
|
|
|
void BreakWindow::showAddressColumn(bool on)
|
|
{
|
|
setColumnHidden(7, !on);
|
|
}
|
|
|
|
void BreakWindow::keyPressEvent(QKeyEvent *ev)
|
|
{
|
|
if (ev->key() == Qt::Key_Delete) {
|
|
QItemSelectionModel *sm = selectionModel();
|
|
QTC_ASSERT(sm, return);
|
|
QModelIndexList si = sm->selectedIndexes();
|
|
if (si.isEmpty())
|
|
si.append(currentIndex());
|
|
const BreakpointModelIds ids = breakHandler()->findBreakpointsByIndex(si);
|
|
int row = qMin(model()->rowCount() - ids.size() - 1, currentIndex().row());
|
|
deleteBreakpoints(ids);
|
|
setCurrentIndex(si.at(0).sibling(row, 0));
|
|
}
|
|
QTreeView::keyPressEvent(ev);
|
|
}
|
|
|
|
void BreakWindow::mouseDoubleClickEvent(QMouseEvent *ev)
|
|
{
|
|
QModelIndex indexUnderMouse = indexAt(ev->pos());
|
|
if (indexUnderMouse.isValid() && indexUnderMouse.column() >= 4) {
|
|
BreakpointModelId id = breakHandler()->findBreakpointByIndex(indexUnderMouse);
|
|
editBreakpoints(BreakpointModelIds() << id);
|
|
}
|
|
QTreeView::mouseDoubleClickEvent(ev);
|
|
}
|
|
|
|
void BreakWindow::setModel(QAbstractItemModel *model)
|
|
{
|
|
BaseWindow::setModel(model);
|
|
resizeColumnToContents(0); // Number
|
|
resizeColumnToContents(3); // Line
|
|
resizeColumnToContents(6); // Ignore count
|
|
connect(model, SIGNAL(layoutChanged()), this, SLOT(expandAll()));
|
|
}
|
|
|
|
void BreakWindow::contextMenuEvent(QContextMenuEvent *ev)
|
|
{
|
|
QMenu menu;
|
|
QItemSelectionModel *sm = selectionModel();
|
|
QTC_ASSERT(sm, return);
|
|
QModelIndexList selectedIndices = sm->selectedIndexes();
|
|
QModelIndex indexUnderMouse = indexAt(ev->pos());
|
|
if (selectedIndices.isEmpty() && indexUnderMouse.isValid())
|
|
selectedIndices.append(indexUnderMouse);
|
|
|
|
BreakHandler *handler = breakHandler();
|
|
BreakpointModelIds selectedIds = handler->findBreakpointsByIndex(selectedIndices);
|
|
|
|
const int rowCount = model()->rowCount();
|
|
QAction *deleteAction = new QAction(tr("Delete Breakpoint"), &menu);
|
|
deleteAction->setEnabled(!selectedIds.isEmpty());
|
|
|
|
QAction *deleteAllAction = new QAction(tr("Delete All Breakpoints"), &menu);
|
|
deleteAllAction->setEnabled(model()->rowCount() > 0);
|
|
|
|
// Delete by file: Find indices of breakpoints of the same file.
|
|
QAction *deleteByFileAction = 0;
|
|
BreakpointModelIds breakpointsInFile;
|
|
if (indexUnderMouse.isValid()) {
|
|
const QModelIndex index = indexUnderMouse.sibling(indexUnderMouse.row(), 2);
|
|
const QString file = index.data().toString();
|
|
if (!file.isEmpty()) {
|
|
for (int i = 0; i != rowCount; ++i)
|
|
if (index.data().toString() == file)
|
|
breakpointsInFile.append(handler->findBreakpointByIndex(index));
|
|
if (breakpointsInFile.size() > 1) {
|
|
deleteByFileAction =
|
|
new QAction(tr("Delete Breakpoints of \"%1\"").arg(file), &menu);
|
|
deleteByFileAction->setEnabled(true);
|
|
}
|
|
}
|
|
}
|
|
if (!deleteByFileAction) {
|
|
deleteByFileAction = new QAction(tr("Delete Breakpoints of File"), &menu);
|
|
deleteByFileAction->setEnabled(false);
|
|
}
|
|
|
|
QAction *adjustColumnAction =
|
|
new QAction(tr("Adjust Column Widths to Contents"), &menu);
|
|
|
|
QAction *editBreakpointAction =
|
|
new QAction(tr("Edit Breakpoint..."), &menu);
|
|
editBreakpointAction->setEnabled(!selectedIds.isEmpty());
|
|
|
|
int threadId = 0;
|
|
// FIXME BP: m_engine->threadsHandler()->currentThreadId();
|
|
QString associateTitle = threadId == -1
|
|
? tr("Associate Breakpoint With All Threads")
|
|
: tr("Associate Breakpoint With Thread %1").arg(threadId);
|
|
QAction *associateBreakpointAction = new QAction(associateTitle, &menu);
|
|
associateBreakpointAction->setEnabled(!selectedIds.isEmpty());
|
|
|
|
QAction *synchronizeAction =
|
|
new QAction(tr("Synchronize Breakpoints"), &menu);
|
|
synchronizeAction->setEnabled(debuggerCore()->hasSnapshots());
|
|
|
|
bool enabled = selectedIds.isEmpty() || handler->isEnabled(selectedIds.at(0));
|
|
|
|
const QString str5 = selectedIds.size() > 1
|
|
? enabled
|
|
? tr("Disable Selected Breakpoints")
|
|
: tr("Enable Selected Breakpoints")
|
|
: enabled
|
|
? tr("Disable Breakpoint")
|
|
: tr("Enable Breakpoint");
|
|
QAction *toggleEnabledAction = new QAction(str5, &menu);
|
|
toggleEnabledAction->setEnabled(!selectedIds.isEmpty());
|
|
|
|
QAction *addBreakpointAction =
|
|
new QAction(tr("Add Breakpoint..."), this);
|
|
|
|
menu.addAction(addBreakpointAction);
|
|
menu.addAction(deleteAction);
|
|
menu.addAction(editBreakpointAction);
|
|
menu.addAction(associateBreakpointAction);
|
|
menu.addAction(toggleEnabledAction);
|
|
menu.addSeparator();
|
|
menu.addAction(deleteAllAction);
|
|
//menu.addAction(deleteByFileAction);
|
|
menu.addSeparator();
|
|
menu.addAction(synchronizeAction);
|
|
menu.addSeparator();
|
|
menu.addAction(debuggerCore()->action(UseToolTipsInBreakpointsView));
|
|
menu.addAction(debuggerCore()->action(UseAddressInBreakpointsView));
|
|
addBaseContextActions(&menu);
|
|
|
|
QAction *act = menu.exec(ev->globalPos());
|
|
|
|
if (act == deleteAction)
|
|
deleteBreakpoints(selectedIds);
|
|
else if (act == deleteAllAction)
|
|
deleteBreakpoints(handler->allBreakpointIds());
|
|
else if (act == deleteByFileAction)
|
|
deleteBreakpoints(breakpointsInFile);
|
|
else if (act == adjustColumnAction)
|
|
resizeColumnsToContents();
|
|
else if (act == editBreakpointAction)
|
|
editBreakpoints(selectedIds);
|
|
else if (act == associateBreakpointAction)
|
|
associateBreakpoint(selectedIds, threadId);
|
|
else if (act == synchronizeAction)
|
|
; //synchronizeBreakpoints();
|
|
else if (act == toggleEnabledAction)
|
|
setBreakpointsEnabled(selectedIds, !enabled);
|
|
else if (act == addBreakpointAction)
|
|
addBreakpoint();
|
|
else
|
|
handleBaseContextAction(act);
|
|
}
|
|
|
|
void BreakWindow::setBreakpointsEnabled(const BreakpointModelIds &ids, bool enabled)
|
|
{
|
|
BreakHandler *handler = breakHandler();
|
|
foreach (const BreakpointModelId id, ids)
|
|
handler->setEnabled(id, enabled);
|
|
}
|
|
|
|
void BreakWindow::deleteBreakpoints(const BreakpointModelIds &ids)
|
|
{
|
|
BreakHandler *handler = breakHandler();
|
|
foreach (const BreakpointModelId id, ids)
|
|
handler->removeBreakpoint(id);
|
|
}
|
|
|
|
void BreakWindow::editBreakpoint(BreakpointModelId id, QWidget *parent)
|
|
{
|
|
BreakpointParameters data = breakHandler()->breakpointData(id);
|
|
unsigned engineCapabilities = AllDebuggerCapabilities;
|
|
if (const DebuggerEngine *engine = breakHandler()->engine(id))
|
|
engineCapabilities = engine->debuggerCapabilities();
|
|
BreakpointParts parts = NoParts;
|
|
BreakpointDialog dialog(engineCapabilities, parent);
|
|
if (dialog.showDialog(&data, &parts))
|
|
breakHandler()->changeBreakpointData(id, data, parts);
|
|
}
|
|
|
|
void BreakWindow::addBreakpoint()
|
|
{
|
|
BreakpointParameters data(BreakpointByFileAndLine);
|
|
BreakpointParts parts = NoParts;
|
|
BreakpointDialog dialog(AllDebuggerCapabilities, this);
|
|
dialog.setWindowTitle(tr("Add Breakpoint"));
|
|
if (dialog.showDialog(&data, &parts))
|
|
breakHandler()->appendBreakpoint(data);
|
|
}
|
|
|
|
void BreakWindow::editBreakpoints(const BreakpointModelIds &ids)
|
|
{
|
|
QTC_ASSERT(!ids.isEmpty(), return);
|
|
|
|
const BreakpointModelId id = ids.at(0);
|
|
|
|
if (ids.size() == 1) {
|
|
editBreakpoint(id, this);
|
|
return;
|
|
}
|
|
|
|
// This allows to change properties of multiple breakpoints at a time.
|
|
BreakHandler *handler = breakHandler();
|
|
unsigned engineCapabilities = AllDebuggerCapabilities;
|
|
if (const DebuggerEngine *engine = breakHandler()->engine(id))
|
|
engineCapabilities = engine->debuggerCapabilities();
|
|
MultiBreakPointsDialog dialog(engineCapabilities);
|
|
const QString oldCondition = QString::fromLatin1(handler->condition(id));
|
|
dialog.setCondition(oldCondition);
|
|
const int oldIgnoreCount = handler->ignoreCount(id);
|
|
dialog.setIgnoreCount(oldIgnoreCount);
|
|
const int oldThreadSpec = handler->threadSpec(id);
|
|
dialog.setThreadSpec(oldThreadSpec);
|
|
|
|
if (dialog.exec() == QDialog::Rejected)
|
|
return;
|
|
|
|
const QString newCondition = dialog.condition();
|
|
const int newIgnoreCount = dialog.ignoreCount();
|
|
const int newThreadSpec = dialog.threadSpec();
|
|
|
|
if (newCondition == oldCondition && newIgnoreCount == oldIgnoreCount
|
|
&& newThreadSpec == oldThreadSpec)
|
|
return;
|
|
|
|
foreach (const BreakpointModelId id, ids) {
|
|
handler->setCondition(id, newCondition.toLatin1());
|
|
handler->setIgnoreCount(id, newIgnoreCount);
|
|
handler->setThreadSpec(id, newThreadSpec);
|
|
}
|
|
}
|
|
|
|
void BreakWindow::associateBreakpoint(const BreakpointModelIds &ids, int threadId)
|
|
{
|
|
BreakHandler *handler = breakHandler();
|
|
foreach (const BreakpointModelId id, ids)
|
|
handler->setThreadSpec(id, threadId);
|
|
}
|
|
|
|
void BreakWindow::rowActivated(const QModelIndex &index)
|
|
{
|
|
breakHandler()->gotoLocation(breakHandler()->findBreakpointByIndex(index));
|
|
}
|
|
|
|
} // namespace Internal
|
|
} // namespace Debugger
|
|
|
|
#include "breakwindow.moc"
|