- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试围绕 QGraphicsRectItem
(PySide 或 PyQt4)构建一个 python 类,它通过悬停提供鼠标交互、可移动和可调整大小。我几乎所有的东西都在工作,除了:
出于某种原因,当项目重新调整大小或移动时,鼠标悬停区域似乎没有改变。我需要帮助解决这个问题。
也许问题是由于在 QGraphicsView
上反转 y 轴引起的:
QGraphicsView.scale(1,-1)
class BoxResizable(QtGui.QGraphicsRectItem):
def __init__(self, rect, parent = None, scene = None):
QtGui.QGraphicsRectItem.__init__(self, rect, parent, scene)
self.setZValue(1000)
self._rect = rect
self._scene = scene
self.mouseOver = False
self.resizeHandleSize = 4.0
self.mousePressPos = None
self.mouseMovePos = None
self.mouseIsPressed = False
self.setFlags(QtGui.QGraphicsItem.ItemIsSelectable|QtGui.QGraphicsItem.ItemIsFocusable)
self.setAcceptsHoverEvents(True)
self.updateResizeHandles()
def hoverEnterEvent(self, event):
self.updateResizeHandles()
self.mouseOver = True
self.prepareGeometryChange()
def hoverLeaveEvent(self, event):
self.mouseOver = False
self.prepareGeometryChange()
def hoverMoveEvent(self, event):
if self.topLeft.contains(event.scenePos()) or self.bottomRight.contains(event.scenePos()):
self.setCursor(QtCore.Qt.SizeFDiagCursor)
elif self.topRight.contains(event.scenePos()) or self.bottomLeft.contains(event.scenePos()):
self.setCursor(QtCore.Qt.SizeBDiagCursor)
else:
self.setCursor(QtCore.Qt.SizeAllCursor)
QtGui.QGraphicsRectItem.hoverMoveEvent(self, event)
def mousePressEvent(self, event):
"""
Capture mouse press events and find where the mosue was pressed on the object
"""
self.mousePressPos = event.scenePos()
self.mouseIsPressed = True
self.rectPress = copy.deepcopy(self._rect)
# Top left corner
if self.topLeft.contains(event.scenePos()):
self.mousePressArea = 'topleft'
# top right corner
elif self.topRight.contains(event.scenePos()):
self.mousePressArea = 'topright'
# bottom left corner
elif self.bottomLeft.contains(event.scenePos()):
self.mousePressArea = 'bottomleft'
# bottom right corner
elif self.bottomRight.contains(event.scenePos()):
self.mousePressArea = 'bottomright'
# entire rectangle
else:
self.mousePressArea = None
QtGui.QGraphicsRectItem.mousePressEvent(self, event)
def mouseReleaseEvent(self, event):
"""
Capture nmouse press events.
"""
self.mouseIsPressed = False
self.updateResizeHandles()
self.prepareGeometryChange()
QtGui.QGraphicsRectItem.mouseReleaseEvent(self, event)
def mouseMoveEvent(self, event):
"""
Handle mouse move events.
"""
self.mouseMovePos = event.scenePos()
if self.mouseIsPressed:
# Move top left corner
if self.mousePressArea=='topleft':
self._rect.setTopLeft(self.rectPress.topLeft()-(self.mousePressPos-self.mouseMovePos))
# Move top right corner
elif self.mousePressArea=='topright':
self._rect.setTopRight(self.rectPress.topRight()-(self.mousePressPos-self.mouseMovePos))
# Move bottom left corner
elif self.mousePressArea=='bottomleft':
self._rect.setBottomLeft(self.rectPress.bottomLeft()-(self.mousePressPos-self.mouseMovePos))
# Move bottom right corner
elif self.mousePressArea=='bottomright':
self._rect.setBottomRight(self.rectPress.bottomRight()-(self.mousePressPos-self.mouseMovePos))
# Move entire rectangle, don't resize
else:
self._rect.moveCenter(self.rectPress.center()-(self.mousePressPos-self.mouseMovePos))
self.updateResizeHandles()
self.prepareGeometryChange()
QtGui.QGraphicsRectItem.mousePressEvent(self, event)
def boundingRect(self):
"""
Return bounding rectangle
"""
return self._boundingRect
def updateResizeHandles(self):
"""
Update bounding rectangle and resize handles
"""
self.offset = self.resizeHandleSize*(self._scene.graphicsView.mapToScene(1,0)-self._scene.graphicsView.mapToScene(0,1)).x()
self._boundingRect = self._rect.adjusted(-self.offset, self.offset, self.offset, -self.offset)
# Note: this draws correctly on a view with an inverted y axes. i.e. QGraphicsView.scale(1,-1)
self.topLeft = QtCore.QRectF(self._boundingRect.topLeft().x(), self._boundingRect.topLeft().y() - 2*self.offset,
2*self.offset, 2*self.offset)
self.topRight = QtCore.QRectF(self._boundingRect.topRight().x() - 2*self.offset, self._boundingRect.topRight().y() - 2*self.offset,
2*self.offset, 2*self.offset)
self.bottomLeft = QtCore.QRectF(self._boundingRect.bottomLeft().x(), self._boundingRect.bottomLeft().y(),
2*self.offset, 2*self.offset)
self.bottomRight = QtCore.QRectF(self._boundingRect.bottomRight().x() - 2*self.offset, self._boundingRect.bottomRight().y(),
2*self.offset, 2*self.offset)
def paint(self, painter, option, widget):
"""
Paint Widget
"""
# show boundingRect for debug purposes
painter.setPen(QtGui.QPen(QtCore.Qt.red, 0, QtCore.Qt.DashLine))
painter.drawRect(self._boundingRect)
# Paint rectangle
painter.setPen(QtGui.QPen(QtCore.Qt.black, 0, QtCore.Qt.SolidLine))
painter.drawRect(self._rect)
# If mouse is over, draw handles
if self.mouseOver:
# if rect selected, fill in handles
if self.isSelected():
painter.setBrush(QtGui.QBrush(QtGui.QColor(0,0,0)))
painter.drawRect(self.topLeft)
painter.drawRect(self.topRight)
painter.drawRect(self.bottomLeft)
painter.drawRect(self.bottomRight)
class graphicsScene(QtGui.QGraphicsScene):
def __init__ (self, parent = None):
QtGui.QGraphicsScene.__init__(self, parent)
def setGraphicsView(self, view):
self.graphicsView = view
app = QtGui.QApplication(sys.argv)
app.setStyle('GTK')
mainWindow = QtGui.QMainWindow()
scene = graphicsScene()
scene.setSceneRect(-100,-100, 200, 200)
# Set up view properties
view = QtGui.QGraphicsView()
view.setScene(scene)
view.scale(1,-1)
view.setRenderHint(QtGui.QPainter.Antialiasing)
view.setViewportUpdateMode(QtGui.QGraphicsView.BoundingRectViewportUpdate)
view.setHorizontalScrollBarPolicy ( QtCore.Qt.ScrollBarAlwaysOff )
view.setVerticalScrollBarPolicy ( QtCore.Qt.ScrollBarAlwaysOff )
view.setUpdatesEnabled(True)
view.setMouseTracking(True)
view.setCacheMode(QtGui.QGraphicsView.CacheBackground)
view.setTransformationAnchor(QtGui.QGraphicsView.AnchorUnderMouse)
scene.setGraphicsView(view)
# Add box
BoxResizable(QtCore.QRectF(-50, 50, 100.0, -100.0), parent = None, scene = scene)
mainWindow.setCentralWidget(view)
mainWindow.show()
app.exec_()
app.deleteLater()
sys.exit()
将上述类添加到 QGraphicsScene
和随后的反转 y 轴 QGraphicsView
将产生如下内容:
如有任何帮助或建议,我们将不胜感激!谢谢!
最佳答案
发现部分问题:如果我覆盖形状函数(将此函数添加到类中):
def shape(self):
path = QtGui.QPainterPath()
path.addRect(self.boundingRect())
return path
悬停区域有时会随着框的大小和位置而变化。根据docs对于 QGraphicsItem.shape()
:
The default implementation calls boundingRect() to return a simple rectangular shape...
这是 PyQt4 中的错误吗?
第二个问题:我认为boundingRect 和shape rectangle 需要有正的宽度和高度?这很容易通过添加来完成:
self._rect = self._rect.normalized()
到 mouseReleaseEvent
函数。
关于Python Qt : Interactive Re-Sizable QGraphicsItem, 鼠标悬停区域不调整大小,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22226587/
我在 Qt 中遇到了帖子标题中描述的错误。 我有一个类调用“ball”,它继承类调用“tableItem”,后者继承了 QGraphicsItem。我正在尝试使用原型(prototype)设计模式,因
我想让 QGraphicsScene 中的一个 QGraphicsItem 在另一个移动时移动(或改变大小)。但是,尝试从 QGraphicsScene 对象访问任一 QGraphicsItem 会导
这个问题是关于:Forcing QGraphicsItem To Stay Put 我想在场景中四处移动时在固定位置放置一个QGraphicsItem。 建议的解决方案是覆盖子类 QGraphicsV
我想我的问题与此类似post但在 C++ 中和在 QGraphicsItem 中。 我想将对象的可移动区域固定在另一个 QGraphicsItem 中。如果我试图将它移到外面,我希望我的对象留在里面。
总结 我在SUSE 64位下使用qt 4.8.7 我有 2 个具有不同刷新率的 QGraphicsItem。但是当我对其中一个调用“update()”时,对它们两个都调用了“paint()”。所以两个
我从 OOAD 类(class)中了解到 dynamic_cast 是一个糟糕的设计但我不知道在 Qt 中没有 dynamic_cast 我怎么能做我想做的事因为我不能只在 QGraphicsItem
我有一个 QGraphicsItem 的子类,我想通过“Control+LMB click”将它的实例添加到场景中。问题是项目被添加到坐标比它们应该大两倍的位置。同时使用 scene.addEllip
我有一个自定义 QGraphicsItem 实现。我需要能够限制可以移动项目的位置 - 即将其限制在某个区域。当我检查 Qt 文档时,这是它的建议: QVariant Component::itemC
我被困在如何解决这个问题上。我在一个场景中有一个 QGraphicsItem,我正在将一个悬停事件从场景传递给这个 child 。当移动事件发生时(我只是使用带有鼠标跟踪的 mouseMoveEven
我应该怎么做才能删除 QGraphicsItem? 从我使用的场景中删除项目 QGraphicsScene::removeItem(QGraphicsItem * item); 来自此方法的文档:
我正在尝试围绕鼠标光标缩放对象。我很容易获得鼠标位置,并且可以使用 item->setScale(n) 毫无问题地缩放对象。但是,我不确定如何实际合并翻译以解释任意点。 有没有办法设置刻度中心?如果没
是否可以为QGraphicsItem设置动画,使其沿着QGraphicsScene/QGraphicsView内的某个路径移动? 类似于 this demo ,其中白点沿着齿轮移动 - 一条闭合曲线。
我想在按下鼠标按钮时检测鼠标光标何时移入QGraphicsItem,即在鼠标进入项目之前按下按钮。我的第一个想法是使用 hoverEnterEvent,但按下鼠标左键时似乎没有触发。我的另一个想法是使
我目前正在创建一个使用 QGraphicsView 的应用程序,并允许用户移动 QGraphicsItems,以便他们可以创建类似图表的结构。 我需要这些项目在单击时改变颜色,但在释放鼠标按钮时变回原
我正在尝试使用rubberBand来选择一些与某些预定的X和Y坐标关联的QGraphicItems。用户可以使用鼠标滚轮进行缩放,并使用鼠标中心按钮进行平移。用户可以单击鼠标左键并绘制一个选择窗口,但
我想在两个或多个 QGraphicsItem 之间实现像素完美碰撞检测器。 QGraphicsItem 类提供了一个使用 QPainterPath 对象的碰撞检测器,所以现在我想将图像从文件加载到 Q
我有很多 qgraphicsitem,它们是 map 上的航路点。我想将它们一起移动。因此,我使用了一个 for 循环来调用它们的 setPos() 函数。但是当项目数量变大(超过 100)时。 Ac
我怎样才能像这张图片一样绘制矩形和椭圆形。在这段代码中,它创建了带有单线边框的矩形和椭圆形。但是我需要像这张给定的图片那样更改边框样式。 Widget::Widget(QWidget *parent)
我正在使用一个 QGraphincsView,它包含几个继承自 QGraphicsItem 的元素。一切正常,我可以根据需要选择它们。当我按住 Ctrl 键时,我可以选择其中的几个。 现在我想实现一个
我有 QGraphicsScene,上面显示了一些自定义 QGraphicsItems。这些项目在 MeasurePoint 类中描述,该类继承自 QGraphicsItem。它们也存储在 QList
我是一名优秀的程序员,十分优秀!