forked from qt-creator/qt-creator
Change-Id: I183fb0f59826291d3c8e3141ab12c77afe5d6213 Reviewed-by: Thomas Hartmann <thomas.hartmann@qt.io>
642 lines
19 KiB
C++
642 lines
19 KiB
C++
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
|
|
|
#include "debugview.h"
|
|
#include "debugviewwidget.h"
|
|
|
|
#include <qmldesignerplugin.h>
|
|
|
|
#include <qmldesignerbase/settings/designersettings.h>
|
|
|
|
#include <bindingproperty.h>
|
|
#include <modelutils.h>
|
|
#include <nodeabstractproperty.h>
|
|
#include <nodelistproperty.h>
|
|
#include <nodemetainfo.h>
|
|
#include <signalhandlerproperty.h>
|
|
#include <variantproperty.h>
|
|
|
|
#include <qmlitemnode.h>
|
|
|
|
#include <utils/algorithm.h>
|
|
#include <utils/smallstringio.h>
|
|
|
|
namespace {
|
|
const QString lineBreak = QStringLiteral("<br>");
|
|
|
|
bool isDebugViewEnabled()
|
|
{
|
|
return QmlDesigner::QmlDesignerPlugin::settings().value(
|
|
QmlDesigner::DesignerSettingsKey::ENABLE_DEBUGVIEW).toBool();
|
|
}
|
|
|
|
bool isDebugViewShown()
|
|
{
|
|
return QmlDesigner::QmlDesignerPlugin::settings().value(
|
|
QmlDesigner::DesignerSettingsKey::SHOW_DEBUGVIEW).toBool();
|
|
}
|
|
|
|
}
|
|
|
|
namespace QmlDesigner {
|
|
|
|
namespace Internal {
|
|
|
|
DebugView::DebugView(ExternalDependenciesInterface &externalDependencies)
|
|
: AbstractView{externalDependencies}
|
|
, m_debugViewWidget(new DebugViewWidget)
|
|
{
|
|
}
|
|
|
|
DebugView::~DebugView()
|
|
{
|
|
delete m_debugViewWidget;
|
|
}
|
|
|
|
void DebugView::modelAttached(Model *model)
|
|
{
|
|
log("::modelAttached:", QString("filename %1").arg(model->fileUrl().toLocalFile()));
|
|
m_debugViewWidget->setDebugViewEnabled(isDebugViewEnabled());
|
|
if (isDebugViewEnabled())
|
|
qDebug() << tr("Debug view is enabled");
|
|
AbstractView::modelAttached(model);
|
|
}
|
|
|
|
void DebugView::modelAboutToBeDetached(Model *model)
|
|
{
|
|
log("::modelAboutToBeDetached:", QString("filename %1").arg(model->fileUrl().toLocalFile()));
|
|
AbstractView::modelAboutToBeDetached(model);
|
|
}
|
|
|
|
void DebugView::importsChanged(const Imports &addedImports, const Imports &removedImports)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QString message;
|
|
message += QString("added imports:") += lineBreak;
|
|
for (const Import &import : addedImports) {
|
|
message += import.toImportString() += lineBreak;
|
|
}
|
|
|
|
message += QString("removed imports:") += lineBreak;
|
|
for (const Import &import : removedImports) {
|
|
message += import.toImportString() += lineBreak;
|
|
}
|
|
|
|
log("::importsChanged:", message);
|
|
}
|
|
}
|
|
|
|
void DebugView::nodeCreated(const ModelNode &createdNode)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
message << createdNode;
|
|
message << createdNode.majorVersion() << "." << createdNode.minorVersion();
|
|
message << createdNode.nodeSource();
|
|
message << "MetaInfo " << createdNode.metaInfo().isValid() << " ";
|
|
if (auto metaInfo = createdNode.metaInfo()) {
|
|
#ifndef QDS_USE_PROJECTSTORAGE
|
|
message << metaInfo.majorVersion() << "." << metaInfo.minorVersion();
|
|
#endif
|
|
message << ModelUtils::componentFilePath(createdNode);
|
|
}
|
|
log("::nodeCreated:", message.readAll());
|
|
}
|
|
}
|
|
|
|
void DebugView::nodeAboutToBeRemoved(const ModelNode &removedNode)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
message << removedNode << lineBreak;
|
|
const QList<ModelNode> modelNodes = removedNode.allSubModelNodes();
|
|
for (const ModelNode &modelNode : modelNodes) {
|
|
message << "child node:" << modelNode << lineBreak;
|
|
}
|
|
|
|
log("::nodeAboutToBeRemoved:", message.readAll());
|
|
}
|
|
}
|
|
|
|
void DebugView::nodeReparented(const ModelNode &node, const NodeAbstractProperty &newPropertyParent,
|
|
const NodeAbstractProperty &oldPropertyParent, AbstractView::PropertyChangeFlags propertyChange)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
message << node;
|
|
message << "new parent property:";
|
|
message << lineBreak;
|
|
message << newPropertyParent;
|
|
message << "old parent property:";
|
|
message << lineBreak;
|
|
message << oldPropertyParent;
|
|
message << "property change flag";
|
|
message << lineBreak;
|
|
message << propertyChange;
|
|
log(tr("::nodeReparented:"), message.readAll());
|
|
}
|
|
}
|
|
|
|
void DebugView::nodeIdChanged(const ModelNode &node, const QString &newId, const QString &oldId)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
message << node;
|
|
message << QString("new id:") << ' ' << newId << lineBreak;
|
|
message << QString("old id:") << ' ' << oldId << lineBreak;
|
|
log(tr("::nodeIdChanged:"), string);
|
|
}
|
|
}
|
|
|
|
void DebugView::propertiesAboutToBeRemoved(const QList<AbstractProperty> &propertyList)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
for (const AbstractProperty &property : propertyList) {
|
|
message << property;
|
|
if (property.isNodeAbstractProperty())
|
|
message << " is NodeAbstractProperty";
|
|
if (property.isDefaultProperty())
|
|
message << " is DefaultProperty";
|
|
}
|
|
log("::propertiesAboutToBeRemoved:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::variantPropertiesChanged(const QList<VariantProperty> &propertyList,
|
|
AbstractView::PropertyChangeFlags /*propertyChange*/)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
for (const VariantProperty &property : propertyList) {
|
|
message << property;
|
|
}
|
|
log("::variantPropertiesChanged:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::bindingPropertiesChanged(const QList<BindingProperty> &propertyList,
|
|
AbstractView::PropertyChangeFlags /*propertyChange*/)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
for (const BindingProperty &property : propertyList) {
|
|
message << property;
|
|
}
|
|
log("::Binding properties changed:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::signalHandlerPropertiesChanged(const QVector<SignalHandlerProperty> &propertyList, AbstractView::PropertyChangeFlags /*propertyChange*/)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
for (const SignalHandlerProperty &property : propertyList) {
|
|
message << property;
|
|
}
|
|
log("::signalHandlerPropertiesChanged:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::rootNodeTypeChanged(const QString &type, int majorVersion, int minorVersion)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QString message;
|
|
message += type;
|
|
message += QStringLiteral(" ");
|
|
message += QString::number(majorVersion);
|
|
message += QStringLiteral(" ");
|
|
message += QString::number(minorVersion);
|
|
log("::rootNodeTypeChanged:", message);
|
|
}
|
|
}
|
|
|
|
namespace {
|
|
QTextStream &operator<<(QTextStream &stream, AuxiliaryDataType type)
|
|
{
|
|
switch (type) {
|
|
case AuxiliaryDataType::None:
|
|
stream << "None";
|
|
break;
|
|
case AuxiliaryDataType::NodeInstancePropertyOverwrite:
|
|
stream << "NodeInstancePropertyOverwrite";
|
|
break;
|
|
case AuxiliaryDataType::NodeInstanceAuxiliary:
|
|
stream << "NodeInstanceAuxiliary";
|
|
break;
|
|
case AuxiliaryDataType::Document:
|
|
stream << "Permanent";
|
|
break;
|
|
case AuxiliaryDataType::Temporary:
|
|
stream << "Temporary";
|
|
break;
|
|
case AuxiliaryDataType::Persistent:
|
|
stream << "Persistent";
|
|
break;
|
|
}
|
|
|
|
return stream;
|
|
}
|
|
} // namespace
|
|
|
|
static QString rectFToString(const QRectF &rect)
|
|
{
|
|
return QString::number(rect.x()) + " " + QString::number(rect.y()) + " "
|
|
+ QString::number(rect.width()) + " " + QString::number(rect.height());
|
|
}
|
|
|
|
static QString sizeToString(const QSize &size)
|
|
{
|
|
return QString::number(size.width()) + " " + QString::number(size.height());
|
|
}
|
|
|
|
void DebugView::selectedNodesChanged(const QList<ModelNode> &selectedNodes /*selectedNodeList*/,
|
|
const QList<ModelNode> & /*lastSelectedNodeList*/)
|
|
{
|
|
for (const ModelNode &selectedNode : selectedNodes) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
message << selectedNode;
|
|
message << " version: " << selectedNode.majorVersion() << '.' << selectedNode.minorVersion();
|
|
for (const VariantProperty &property : selectedNode.variantProperties())
|
|
message << property << lineBreak;
|
|
|
|
message << lineBreak;
|
|
|
|
for (const SignalHandlerProperty &property : selectedNode.signalProperties())
|
|
message << property << lineBreak;
|
|
|
|
message << lineBreak;
|
|
|
|
message << lineBreak;
|
|
message << "metaInfo valid: " << selectedNode.metaInfo().isValid();
|
|
message << lineBreak;
|
|
|
|
if (selectedNode.metaInfo().isValid()) {
|
|
#ifndef QDS_USE_PROJECTSTORAGE
|
|
for (const NodeMetaInfo &metaInfo : selectedNode.metaInfo().selfAndPrototypes()) {
|
|
message << metaInfo.typeName() << " " << metaInfo.majorVersion() << "."
|
|
<< metaInfo.minorVersion() << lineBreak;
|
|
}
|
|
|
|
message << lineBreak;
|
|
message << selectedNode.metaInfo().typeName();
|
|
#endif
|
|
message << lineBreak;
|
|
|
|
message << "Node Source" << selectedNode.nodeSource();
|
|
message << lineBreak;
|
|
|
|
message << "Is Component" << selectedNode.isComponent();
|
|
message << lineBreak;
|
|
|
|
message << "Node Source Type" << selectedNode.nodeSourceType();
|
|
message << lineBreak;
|
|
|
|
message << lineBreak;
|
|
|
|
for (const PropertyName &name : selectedNode.metaInfo().slotNames())
|
|
message << name << " ";
|
|
|
|
message << lineBreak;
|
|
message << "Is QtQuickItem: "
|
|
<< selectedNode.metaInfo().isBasedOn(model()->qtQuickItemMetaInfo());
|
|
}
|
|
|
|
message << lineBreak;
|
|
message << "Is item or window: " << QmlItemNode::isItemOrWindow(selectedNode);
|
|
message << lineBreak;
|
|
|
|
message << lineBreak;
|
|
message << "Is graphical item: " << selectedNode.metaInfo().isGraphicalItem();
|
|
message << lineBreak;
|
|
|
|
message << lineBreak;
|
|
message << "Is valid object node: " << QmlItemNode::isValidQmlObjectNode(selectedNode);
|
|
message << lineBreak;
|
|
|
|
#ifndef QDS_USE_PROJECTSTORAGE
|
|
message << "version: "
|
|
<< QString::number(selectedNode.metaInfo().majorVersion()) + "."
|
|
+ QString::number(selectedNode.metaInfo().minorVersion());
|
|
#endif
|
|
message << lineBreak;
|
|
|
|
QmlItemNode itemNode(selectedNode);
|
|
if (itemNode.isValid()) {
|
|
message << lineBreak;
|
|
message << "Item Node";
|
|
message << lineBreak;
|
|
message << "BR ";
|
|
message << rectFToString(itemNode.instanceBoundingRect());
|
|
message << lineBreak;
|
|
message << "BR content ";
|
|
message << rectFToString(itemNode.instanceContentItemBoundingRect());
|
|
message << lineBreak;
|
|
message << "PM ";
|
|
message << sizeToString(itemNode.instanceRenderPixmap().size());
|
|
message << lineBreak;
|
|
}
|
|
|
|
auto auxiliaryData = selectedNode.auxiliaryData();
|
|
AuxiliaryDatas sortedaAuxiliaryData{auxiliaryData.begin(), auxiliaryData.end()};
|
|
|
|
Utils::sort(sortedaAuxiliaryData, [](const auto &first, const auto &second) {
|
|
return first.first < second.first;
|
|
});
|
|
for (const auto &element : sortedaAuxiliaryData) {
|
|
message << element.first.type << ' ' << element.first.name.data() << ' '
|
|
<< element.second.toString() << lineBreak;
|
|
}
|
|
|
|
log("::selectedNodesChanged:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::scriptFunctionsChanged(const ModelNode & /*node*/, const QStringList & /*scriptFunctionList*/)
|
|
{
|
|
}
|
|
|
|
void DebugView::propertiesRemoved(const QList<AbstractProperty> &propertyList)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
for (const AbstractProperty &property : propertyList) {
|
|
message << property;
|
|
}
|
|
log("::propertiesRemoved:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::auxiliaryDataChanged(const ModelNode &node,
|
|
AuxiliaryDataKeyView key,
|
|
const QVariant &data)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
|
|
message << node;
|
|
message << key.type;
|
|
message << key.name.toByteArray();
|
|
message << data.toString();
|
|
|
|
log("::auxiliaryDataChanged:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::documentMessagesChanged(const QList<DocumentMessage> &errors, const QList<DocumentMessage> &warnings)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
|
|
for (const DocumentMessage &error : errors) {
|
|
message << error.toString();
|
|
}
|
|
|
|
for (const DocumentMessage &warning : warnings) {
|
|
message << warning.toString();
|
|
}
|
|
|
|
log("::documentMessageChanged:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::rewriterBeginTransaction()
|
|
{
|
|
if (isDebugViewEnabled())
|
|
log("::rewriterBeginTransaction:", QString(), true);
|
|
}
|
|
|
|
void DebugView::rewriterEndTransaction()
|
|
{
|
|
if (isDebugViewEnabled())
|
|
log("::rewriterEndTransaction:", QString(), true);
|
|
}
|
|
|
|
WidgetInfo DebugView::widgetInfo()
|
|
{
|
|
return createWidgetInfo(m_debugViewWidget.data(),
|
|
QStringLiteral("DebugView"),
|
|
WidgetInfo::LeftPane,
|
|
tr("Debug View"));
|
|
}
|
|
|
|
bool DebugView::hasWidget() const
|
|
{
|
|
if (isDebugViewShown())
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
void DebugView::instancePropertyChanged(const QList<QPair<ModelNode, PropertyName> > &propertyList)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
|
|
using Pair = QPair<ModelNode, PropertyName>;
|
|
|
|
for (const Pair &pair : propertyList) {
|
|
message << pair.first;
|
|
message << " ";
|
|
message << pair.second;
|
|
message << ": ";
|
|
message << QmlObjectNode(pair.first).instanceValue(pair.second).toString();
|
|
message << lineBreak;
|
|
}
|
|
|
|
logInstance(":instancePropertyChanged::", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::instanceErrorChanged(const QVector<ModelNode> &/*errorNodeList*/)
|
|
{
|
|
}
|
|
|
|
void DebugView::instancesCompleted(const QVector<ModelNode> &completedNodeList)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
|
|
for (const ModelNode &modelNode : completedNodeList) {
|
|
message << modelNode << lineBreak;
|
|
if (QmlItemNode::isValidQmlItemNode(modelNode)) {
|
|
message << "parent: " << QmlItemNode(modelNode).instanceParent() << lineBreak;
|
|
}
|
|
}
|
|
|
|
logInstance(":instancesCompleted::", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::instanceInformationsChanged(const QMultiHash<ModelNode, InformationName> &informationChangedHash)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
|
|
const QList<ModelNode> modelNodes = informationChangedHash.keys();
|
|
for (const ModelNode &modelNode : modelNodes) {
|
|
message << modelNode;
|
|
message << informationChangedHash.value(modelNode);
|
|
message << ": ";
|
|
message << QmlItemNode(modelNode).instanceSize().width();
|
|
message << " ";
|
|
message << QmlItemNode(modelNode).instanceSize().height();
|
|
}
|
|
|
|
logInstance("::instanceInformationsChanged:", string);
|
|
}
|
|
|
|
}
|
|
|
|
void DebugView::instancesRenderImageChanged(const QVector<ModelNode> & /*nodeList*/)
|
|
{
|
|
}
|
|
|
|
void DebugView::instancesPreviewImageChanged(const QVector<ModelNode> & /*nodeList*/)
|
|
{
|
|
}
|
|
|
|
void DebugView::instancesChildrenChanged(const QVector<ModelNode> & nodeList)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
|
|
for (const ModelNode &modelNode : nodeList) {
|
|
message << modelNode << lineBreak;
|
|
if (QmlItemNode::isValidQmlItemNode(modelNode)) {
|
|
message << "parent: " << QmlItemNode(modelNode).instanceParent() << lineBreak;
|
|
}
|
|
}
|
|
|
|
logInstance("::instancesChildrenChanged:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::customNotification(const AbstractView *view, const QString &identifier, const QList<ModelNode> &nodeList, const QList<QVariant> &data)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
|
|
message << view;
|
|
message << identifier;
|
|
for (const ModelNode &node : nodeList) {
|
|
message << node;
|
|
}
|
|
|
|
for (const QVariant &variant : data) {
|
|
message << variant.toString();
|
|
}
|
|
|
|
log("::customNotification:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::nodeSourceChanged(const ModelNode &modelNode, const QString &newNodeSource)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
|
|
message << modelNode;
|
|
message << newNodeSource;
|
|
|
|
log("::nodeSourceChanged:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::nodeRemoved(const ModelNode &removedNode, const NodeAbstractProperty &/*parentProperty*/, AbstractView::PropertyChangeFlags /*propertyChange*/)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
|
|
message << removedNode;
|
|
|
|
log("::nodeRemoved:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::nodeAboutToBeReparented(const ModelNode &/*node*/, const NodeAbstractProperty &/*newPropertyParent*/, const NodeAbstractProperty &/*oldPropertyParent*/, AbstractView::PropertyChangeFlags /*propertyChange*/)
|
|
{
|
|
|
|
}
|
|
|
|
void DebugView::instancesToken(const QString &/*tokenName*/, int /*tokenNumber*/, const QVector<ModelNode> &/*nodeVector*/)
|
|
{
|
|
|
|
}
|
|
|
|
void DebugView::currentStateChanged(const ModelNode &node)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
|
|
message << node;
|
|
|
|
log("::currentStateChanged:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::nodeOrderChanged([[maybe_unused]] const NodeListProperty &listProperty)
|
|
{
|
|
if (isDebugViewEnabled()) {
|
|
QTextStream message;
|
|
QString string;
|
|
message.setString(&string);
|
|
|
|
log("::nodeSlide:", string);
|
|
}
|
|
}
|
|
|
|
void DebugView::log(const QString &title, const QString &message, bool highlight)
|
|
{
|
|
m_debugViewWidget->addLogMessage(title, message, highlight);
|
|
}
|
|
|
|
void DebugView::logInstance(const QString &title, const QString &message, bool highlight)
|
|
{
|
|
m_debugViewWidget->addLogInstanceMessage(title, message, highlight);
|
|
}
|
|
|
|
} // namesapce Internal
|
|
|
|
} // namespace QmlDesigner
|