forked from qt-creator/qt-creator
Refactored QmlJS Live Preview into a library
QMLObserver links against the library dynamically. User's apps can also link to it statically, so that the app can be deployed with debugging features to a device, without having to worry about library deployment.
This commit is contained in:
326
src/libs/qmljsdebugger/editor/subcomponenteditortool.cpp
Normal file
326
src/libs/qmljsdebugger/editor/subcomponenteditortool.cpp
Normal file
@@ -0,0 +1,326 @@
|
||||
#include "subcomponenteditortool.h"
|
||||
#include "qdeclarativedesignview.h"
|
||||
#include "subcomponentmasklayeritem.h"
|
||||
#include "layeritem.h"
|
||||
#include "crumblepath.h"
|
||||
|
||||
#include <QGraphicsItem>
|
||||
#include <QGraphicsObject>
|
||||
#include <QTimer>
|
||||
#include <QMouseEvent>
|
||||
#include <QKeyEvent>
|
||||
|
||||
#include <QDebug>
|
||||
|
||||
|
||||
namespace QmlViewer {
|
||||
|
||||
const qreal MaxOpacity = 0.5f;
|
||||
|
||||
SubcomponentEditorTool::SubcomponentEditorTool(QDeclarativeDesignView *view)
|
||||
: AbstractFormEditorTool(view),
|
||||
m_animIncrement(0.05f),
|
||||
m_animTimer(new QTimer(this)),
|
||||
m_crumblePathWidget(0)
|
||||
{
|
||||
m_mask = new SubcomponentMaskLayerItem(view, view->manipulatorLayer());
|
||||
connect(m_animTimer, SIGNAL(timeout()), SLOT(animate()));
|
||||
m_animTimer->setInterval(20);
|
||||
}
|
||||
|
||||
SubcomponentEditorTool::~SubcomponentEditorTool()
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::mousePressEvent(QMouseEvent * /*event*/)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::mouseMoveEvent(QMouseEvent * /*event*/)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
bool SubcomponentEditorTool::containsCursor(const QPoint &mousePos) const
|
||||
{
|
||||
if (!m_currentContext.size())
|
||||
return false;
|
||||
|
||||
QPointF scenePos = view()->mapToScene(mousePos);
|
||||
QRectF itemRect = m_currentContext.top()->boundingRect() | m_currentContext.top()->childrenBoundingRect();
|
||||
QRectF polyRect = m_currentContext.top()->mapToScene(itemRect).boundingRect();
|
||||
|
||||
return polyRect.contains(scenePos);
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::mouseReleaseEvent(QMouseEvent * /*event*/)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::mouseDoubleClickEvent(QMouseEvent *event)
|
||||
{
|
||||
if (event->buttons() & Qt::LeftButton
|
||||
&& !containsCursor(event->pos())
|
||||
&& m_currentContext.size() > 1)
|
||||
{
|
||||
aboutToPopContext();
|
||||
}
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::hoverMoveEvent(QMouseEvent *event)
|
||||
{
|
||||
if (!containsCursor(event->pos()) && m_currentContext.size() > 1) {
|
||||
view()->clearHighlight();
|
||||
}
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::wheelEvent(QWheelEvent * /*event*/)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::keyPressEvent(QKeyEvent * /*event*/)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::keyReleaseEvent(QKeyEvent * /*keyEvent*/)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::itemsAboutToRemoved(const QList<QGraphicsItem*> &/*itemList*/)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::animate()
|
||||
{
|
||||
if (m_animIncrement > 0) {
|
||||
if (m_mask->opacity() + m_animIncrement < MaxOpacity) {
|
||||
m_mask->setOpacity(m_mask->opacity() + m_animIncrement);
|
||||
} else {
|
||||
m_animTimer->stop();
|
||||
m_mask->setOpacity(MaxOpacity);
|
||||
}
|
||||
} else {
|
||||
if (m_mask->opacity() + m_animIncrement > 0) {
|
||||
m_mask->setOpacity(m_mask->opacity() + m_animIncrement);
|
||||
} else {
|
||||
m_animTimer->stop();
|
||||
m_mask->setOpacity(0);
|
||||
popContext();
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::clear()
|
||||
{
|
||||
m_currentContext.clear();
|
||||
m_mask->setCurrentItem(0);
|
||||
m_animTimer->stop();
|
||||
m_mask->hide();
|
||||
|
||||
if (m_crumblePathWidget)
|
||||
m_crumblePathWidget->clear();
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::graphicsObjectsChanged(const QList<QGraphicsObject*> &/*itemList*/)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::selectedItemsChanged(const QList<QGraphicsItem*> &/*itemList*/)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::setCurrentItem(QGraphicsItem* contextItem)
|
||||
{
|
||||
if (!contextItem)
|
||||
return;
|
||||
|
||||
QGraphicsObject *gfxObject = contextItem->toGraphicsObject();
|
||||
if (!gfxObject)
|
||||
return;
|
||||
|
||||
connect(gfxObject, SIGNAL(xChanged()), SLOT(refresh()));
|
||||
connect(gfxObject, SIGNAL(yChanged()), SLOT(refresh()));
|
||||
connect(gfxObject, SIGNAL(scaleChanged()), SLOT(refresh()));
|
||||
connect(gfxObject, SIGNAL(widthChanged()), SLOT(refresh()));
|
||||
connect(gfxObject, SIGNAL(heightChanged()), SLOT(refresh()));
|
||||
|
||||
//QString parentClassName = gfxObject->metaObject()->className();
|
||||
//if (parentClassName.contains(QRegExp("_QMLTYPE_\\d+")))
|
||||
|
||||
bool containsSelectableItems = false;
|
||||
foreach(QGraphicsItem *item, gfxObject->childItems()) {
|
||||
if (item->type() == Constants::EditorItemType
|
||||
|| item->type() == Constants::ResizeHandleItemType)
|
||||
{
|
||||
continue;
|
||||
}
|
||||
containsSelectableItems = true;
|
||||
break;
|
||||
}
|
||||
|
||||
if (containsSelectableItems) {
|
||||
m_mask->setCurrentItem(gfxObject);
|
||||
m_mask->setOpacity(0);
|
||||
m_mask->show();
|
||||
m_animIncrement = 0.05f;
|
||||
m_animTimer->start();
|
||||
|
||||
view()->clearHighlight();
|
||||
view()->setSelectedItems(QList<QGraphicsItem*>());
|
||||
|
||||
pushContext(gfxObject);
|
||||
}
|
||||
}
|
||||
|
||||
QGraphicsItem *SubcomponentEditorTool::firstChildOfContext(QGraphicsItem *item) const
|
||||
{
|
||||
if (!item)
|
||||
return 0;
|
||||
|
||||
if (isDirectChildOfContext(item))
|
||||
return item;
|
||||
|
||||
QGraphicsItem *parent = item->parentItem();
|
||||
while (parent) {
|
||||
if (isDirectChildOfContext(parent))
|
||||
return parent;
|
||||
parent = parent->parentItem();
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
bool SubcomponentEditorTool::isChildOfContext(QGraphicsItem *item) const
|
||||
{
|
||||
return (firstChildOfContext(item) != 0);
|
||||
}
|
||||
|
||||
bool SubcomponentEditorTool::isDirectChildOfContext(QGraphicsItem *item) const
|
||||
{
|
||||
return (item->parentItem() == m_currentContext.top());
|
||||
}
|
||||
|
||||
bool SubcomponentEditorTool::itemIsChildOfQmlSubComponent(QGraphicsItem *item) const
|
||||
{
|
||||
if (item->parentItem() && item->parentItem() != m_currentContext.top()) {
|
||||
QGraphicsObject *parent = item->parentItem()->toGraphicsObject();
|
||||
QString parentClassName = parent->metaObject()->className();
|
||||
|
||||
if (parentClassName.contains(QRegExp("_QMLTYPE_\\d+"))) {
|
||||
return true;
|
||||
} else {
|
||||
return itemIsChildOfQmlSubComponent(parent);
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::pushContext(QGraphicsObject *contextItem)
|
||||
{
|
||||
connect(contextItem, SIGNAL(destroyed(QObject*)), SLOT(contextDestroyed(QObject*)));
|
||||
m_currentContext.push(contextItem);
|
||||
if (m_crumblePathWidget)
|
||||
m_crumblePathWidget->pushElement(titleForItem(contextItem));
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::aboutToPopContext()
|
||||
{
|
||||
if (m_currentContext.size() > 2) {
|
||||
popContext();
|
||||
} else {
|
||||
m_animIncrement = -0.05f;
|
||||
m_animTimer->start();
|
||||
}
|
||||
}
|
||||
|
||||
QGraphicsObject *SubcomponentEditorTool::popContext()
|
||||
{
|
||||
QGraphicsObject *popped = m_currentContext.pop();
|
||||
|
||||
if (m_crumblePathWidget)
|
||||
m_crumblePathWidget->popElement();
|
||||
|
||||
disconnect(popped, SIGNAL(xChanged()), this, SLOT(refresh()));
|
||||
disconnect(popped, SIGNAL(yChanged()), this, SLOT(refresh()));
|
||||
disconnect(popped, SIGNAL(scaleChanged()), this, SLOT(refresh()));
|
||||
disconnect(popped, SIGNAL(widthChanged()), this, SLOT(refresh()));
|
||||
disconnect(popped, SIGNAL(heightChanged()), this, SLOT(refresh()));
|
||||
|
||||
if (m_currentContext.size() > 1) {
|
||||
m_mask->setCurrentItem(m_currentContext.top());
|
||||
m_mask->setOpacity(MaxOpacity);
|
||||
m_mask->setVisible(true);
|
||||
} else {
|
||||
m_mask->setVisible(false);
|
||||
}
|
||||
|
||||
return popped;
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::refresh()
|
||||
{
|
||||
m_mask->setCurrentItem(m_currentContext.top());
|
||||
}
|
||||
|
||||
QGraphicsObject *SubcomponentEditorTool::currentRootItem() const
|
||||
{
|
||||
return m_currentContext.top();
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::contextDestroyed(QObject *contextToDestroy)
|
||||
{
|
||||
disconnect(contextToDestroy, SIGNAL(destroyed(QObject*)), this, SLOT(contextDestroyed(QObject*)));
|
||||
|
||||
// pop out the whole context - it might not be safe anymore.
|
||||
while (m_currentContext.size() > 1) {
|
||||
m_currentContext.pop();
|
||||
if (m_crumblePathWidget)
|
||||
m_crumblePathWidget->popElement();
|
||||
}
|
||||
|
||||
|
||||
m_mask->setVisible(false);
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::setCrumblePathWidget(CrumblePath *pathWidget)
|
||||
{
|
||||
m_crumblePathWidget = pathWidget;
|
||||
|
||||
if (m_crumblePathWidget) {
|
||||
connect(m_crumblePathWidget, SIGNAL(elementClicked(int)), SLOT(setContext(int)));
|
||||
connect(m_crumblePathWidget, SIGNAL(elementContextMenuRequested(int)), SLOT(openContextMenuForContext(int)));
|
||||
}
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::setContext(int contextIndex)
|
||||
{
|
||||
Q_ASSERT(contextIndex >= 0);
|
||||
|
||||
// sometimes we have to delete the context while user was still clicking around,
|
||||
// so just bail out.
|
||||
if (contextIndex >= m_currentContext.size() -1)
|
||||
return;
|
||||
|
||||
while (m_currentContext.size() - 1 > contextIndex) {
|
||||
popContext();
|
||||
}
|
||||
}
|
||||
|
||||
void SubcomponentEditorTool::openContextMenuForContext(int /*contextIndex*/)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
|
||||
} // namespace QmlViewer
|
||||
Reference in New Issue
Block a user