Skip to content
Snippets Groups Projects
watchhandler.h 10.5 KiB
Newer Older
/**************************************************************************
con's avatar
con committed
**
** This file is part of Qt Creator
**
** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
con's avatar
con committed
**
** Contact: Nokia Corporation (qt-info@nokia.com)
con's avatar
con committed
**
** Commercial Usage
** 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.
** GNU Lesser General Public License Usage
** 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.
** If you are unsure which license is appropriate for your use, please
hjk's avatar
hjk committed
** contact the sales department at http://qt.nokia.com/contact.
con's avatar
con committed
**
**************************************************************************/
hjk's avatar
hjk committed

con's avatar
con committed
#ifndef DEBUGGER_WATCHHANDLER_H
#define DEBUGGER_WATCHHANDLER_H

#include <QtCore/QPointer>
#include <QtCore/QObject>
#include <QtCore/QHash>
#include <QtCore/QSet>
#include <QtCore/QAbstractItemModel>
con's avatar
con committed
#include <QtScript/QScriptValue>

QT_BEGIN_NAMESPACE
class QDebug;
QT_END_NAMESPACE

con's avatar
con committed
namespace Debugger {

class DebuggerManager;

con's avatar
con committed
namespace Internal {

hjk's avatar
hjk committed
class WatchHandler;
enum WatchType { LocalsWatch, WatchersWatch, TooltipsWatch };
hjk's avatar
hjk committed

con's avatar
con committed
class WatchData
{
public:
    WatchData();

    enum State
    {
        Complete = 0,
con's avatar
con committed
        ValueNeeded = 2,
        TypeNeeded = 4,
        ChildrenNeeded = 8,

        NeededMask = ValueNeeded
            | TypeNeeded
            | ChildrenNeeded
con's avatar
con committed

        InitialState = ValueNeeded
            | TypeNeeded
            | ChildrenNeeded
con's avatar
con committed
    };

    void setValue(const QString &);
Friedemann Kleint's avatar
Friedemann Kleint committed
    void setType(const QString &, bool guessChildrenFromType = true);
con's avatar
con committed
    void setValueToolTip(const QString &);
    void setError(const QString &);
    void setAddress(const QString &address);

    bool isSomethingNeeded() const { return state & NeededMask; }
    void setAllNeeded() { state = NeededMask; }
    void setAllUnneeded() { state = State(0); }

    bool isTypeNeeded() const { return state & TypeNeeded; }
    bool isTypeKnown() const { return !(state & TypeNeeded); }
    void setTypeNeeded() { state = State(state | TypeNeeded); }
    void setTypeUnneeded() { state = State(state & ~TypeNeeded); }

    bool isValueNeeded() const { return state & ValueNeeded; }
    bool isValueKnown() const { return !(state & ValueNeeded); }
    void setValueNeeded() { state = State(state | ValueNeeded); }
    void setValueUnneeded() { state = State(state & ~ValueNeeded); }

    bool isChildrenNeeded() const { return state & ChildrenNeeded; }
    bool isChildrenKnown() const { return !(state & ChildrenNeeded); }
    void setChildrenNeeded() { state = State(state | ChildrenNeeded); }
    void setChildrenUnneeded() { state = State(state & ~ChildrenNeeded); }

    bool isHasChildrenNeeded() const { return state & HasChildrenNeeded; }
    bool isHasChildrenKnown() const { return !(state & HasChildrenNeeded); }
    void setHasChildrenNeeded() { state = State(state | HasChildrenNeeded); }
    void setHasChildrenUnneeded() { state = State(state & ~HasChildrenNeeded); }
    void setHasChildren(bool c) { hasChildren = c; setHasChildrenUnneeded();
        if (!c) setChildrenUnneeded(); }
con's avatar
con committed

    QString toString() const;
    QString toToolTip() const;
    bool isLocal() const { return iname.startsWith("local."); }
    bool isWatcher() const { return iname.startsWith("watch."); }
con's avatar
con committed
    bool isValid() const { return !iname.isEmpty(); }
    
    bool isEqual(const WatchData &other) const;
con's avatar
con committed

    static QString msgNotInScope();
    static QString shadowedName(const QString &name, int seen);

con's avatar
con committed
public:
    QByteArray iname;     // internal name sth like 'local.baz.public.a'
    QByteArray exp;       // the expression
con's avatar
con committed
    QString name;         // displayed name
    QString value;        // displayed value
    QByteArray editvalue; // displayed value
    QString valuetooltip; // tooltip in value column
    QString type;         // type for further processing
    QString displayedType;// displayed type (optional)
    QByteArray variable;  // name of internal Gdb variable if created
Tobias Hunger's avatar
Tobias Hunger committed
    QByteArray addr;      // displayed address
    QByteArray saddr;     // stored address (pointer in container)
con's avatar
con committed
    QString framekey;     // key for type cache
    QScriptValue scriptValue; // if needed...
hjk's avatar
hjk committed
    int generation;       // when updated?
    bool valueEnabled;    // value will be greyed out or not
    bool valueEditable;   // value will be editable
con's avatar
con committed

private:

public:
    int source;  // Used by some debuggers (CDB) to tell where it originates from (dumper or symbol evaluation)
con's avatar
con committed
    int state;
    bool changed;
};

enum WatchRoles
{
    INameRole = Qt::UserRole,
    ExpressionRole,
Tobias Hunger's avatar
Tobias Hunger committed
    ExpandedRole,    // used to communicate preferred expanded state to the view
    ActiveDataRole,  // used for tooltip
    TypeFormatListRole,
    TypeFormatRole,  // used to communicate alternative formats to the view
    IndividualFormatRole,
    AddressRole,     // some memory address related to the object
con's avatar
con committed

enum IntegerFormat
{
    DecimalFormat = 0, // keep that at 0 as default
    HexadecimalFormat,
    BinaryFormat,
    OctalFormat,
};


enum PointerFomat
{
    BaldPointerFormat = 0, // keep that at 0 as default
    Latin1StringFormat,
    Local8BitStringFormat,
    Utf8StringFormat,
    Utf16StringFormat,
    Ucs4StringFormat,
};


enum DumpableFormat
{
    PrettyFormat = 0, // keep that at 0 as default
    PlainFomat, 
};

hjk's avatar
hjk committed
class WatchModel : public QAbstractItemModel
con's avatar
con committed
{
    Q_OBJECT
private:
    explicit WatchModel(WatchHandler *handler, WatchType type);
    virtual ~WatchModel();

    QVariant data(const QModelIndex &index, int role) const;
    bool setData(const QModelIndex &index, const QVariant &value, int role);
    QModelIndex index(int, int, const QModelIndex &idx) const;
    QModelIndex parent(const QModelIndex &idx) const;
    int rowCount(const QModelIndex &idx) const;
    int columnCount(const QModelIndex &idx) const;
    bool hasChildren(const QModelIndex &idx) const;
    Qt::ItemFlags flags(const QModelIndex &idx) const;
    QVariant headerData(int section, Qt::Orientation orientation,
        int role = Qt::DisplayRole) const;
    bool canFetchMore(const QModelIndex &parent) const;
    void fetchMore(const QModelIndex &parent);

    friend class WatchHandler;
    friend class GdbEngine;
hjk's avatar
hjk committed
    WatchItem *watchItem(const QModelIndex &) const;
    QModelIndex watchIndex(const WatchItem *needle) const;
    QModelIndex watchIndexHelper(const WatchItem *needle,
        const WatchItem *parentItem, const QModelIndex &parentIndex) const;

    void insertData(const WatchData &data);
    void insertBulkData(const QList<WatchData> &data);
    WatchItem *findItem(const QByteArray &iname, WatchItem *root) const;
hjk's avatar
hjk committed
    void reinitialize();
    void removeOutdated();
    void removeOutdatedHelper(WatchItem *item);
    WatchItem *rootItem() const;
    void destroyItem(WatchItem *item);
hjk's avatar
hjk committed

    void emitDataChanged(int column,
        const QModelIndex &parentIndex = QModelIndex());
    void beginCycle(); // called at begin of updateLocals() cycle
    void endCycle(); // called after all results have been received
    friend QDebug operator<<(QDebug d, const WatchModel &m);

    void dump();
    void dumpHelper(WatchItem *item);
    void emitAllChanged();
    Q_SLOT void resetExtraLayoutChanged();
signals:
    void enableUpdates(bool);

hjk's avatar
hjk committed
private:
    QString niceType(const QString &typeIn) const;

    WatchType m_type;
    WatchItem *m_root;
    QSet<QByteArray> m_fetchTriggered;
    // Part of the workaround to update the [+] marker when items
    // are added to a container.
    bool m_inExtraLayoutChanged;
hjk's avatar
hjk committed
};

class WatchHandler : public QObject
{
    Q_OBJECT

public:
    explicit WatchHandler(DebuggerManager *manager);
hjk's avatar
hjk committed
    WatchModel *model(WatchType type) const;
    WatchModel *modelForIName(const QByteArray &iname) const;
hjk's avatar
hjk committed

con's avatar
con committed
//public slots:
    void cleanup();
    Q_SLOT void watchExpression(); // data in action->data().toString()
    Q_SLOT void watchExpression(const QString &exp);
    Q_SLOT void removeWatchExpression();
    Q_SLOT void removeWatchExpression(const QString &exp);
    Q_SLOT void emitAllChanged();
    void beginCycle(); // called at begin of updateLocals() cycle
    void updateWatchers(); // called after locals are fetched
    void endCycle(); // called after all results have been received
hjk's avatar
hjk committed
    void showEditValue(const WatchData &data);
con's avatar
con committed

    void insertData(const WatchData &data);
    void insertBulkData(const QList<WatchData> &data);
    void removeData(const QByteArray &iname);
    WatchData *findItem(const QByteArray &iname) const;
con's avatar
con committed

hjk's avatar
hjk committed
    void loadSessionData();
    void saveSessionData();
con's avatar
con committed

    bool isDisplayedIName(const QByteArray &iname) const
con's avatar
con committed
        { return m_displayedINames.contains(iname); }
    bool isExpandedIName(const QByteArray &iname) const
con's avatar
con committed
        { return m_expandedINames.contains(iname); }
    QSet<QByteArray> expandedINames() const
        { return m_expandedINames; }
    QStringList watchedExpressions() const;
    QHash<QByteArray, int> watcherNames() const
        { return m_watcherNames; }
con's avatar
con committed

    static QString watcherEditPlaceHolder();

con's avatar
con committed
private:
    friend class WatchModel;
con's avatar
con committed

    void loadWatchers();
    void saveWatchers();

    void loadTypeFormats();
    void saveTypeFormats();
    void setFormat(const QString &type, int format);

con's avatar
con committed
    bool m_expandPointers;
    bool m_inChange;

    typedef QMap<QString, QPointer<QWidget> > EditWindows;
    EditWindows m_editWindows;

    QHash<QByteArray, int> m_watcherNames;
    QByteArray watcherName(const QByteArray &exp);
    QHash<QString, int> m_typeFormats;
    QHash<QString, int> m_individualFormats;
con's avatar
con committed

    void setDisplayedIName(const QString &iname, bool on);
    QSet<QByteArray> m_expandedINames;  // those expanded in the treeview
    QSet<QByteArray> m_displayedINames; // those with "external" viewers
con's avatar
con committed

    WatchModel *m_locals;
    WatchModel *m_watchers;
    WatchModel *m_tooltips;
    DebuggerManager *m_manager;
con's avatar
con committed
};

} // namespace Internal
} // namespace Debugger

Q_DECLARE_METATYPE(Debugger::Internal::WatchData);

#endif // DEBUGGER_WATCHHANDLER_H