我正在尝试编写一个可用作MS Paint或其他简单图形编辑器的应用程序。我的目标是提供如下功能:不同的工具(绘制线条,图形等),改变颜色,宽度,大小和撤消/重做功能的可能性。我认为最简单的方法是创建用于绘制和查看对象的separete类。不幸的是,它运作不正常。这是我的代码:
import sys
from PyQt4 import QtGui, QtCore
class Example(QtGui.QWidget):
def __init__(self):
super(Example, self).__init__()
self.update()
self.view = View(self)
self.action = Action(self.view.scene)
self.UI()
def UI(self):
self.setGeometry(300,300,300,300)
self.setWindowTitle('Pen styles')
self.undo = QtGui.QPushButton("undo", self)
self.undo.clicked.connect(self.action.undoStack.undo)
self.redo = QtGui.QPushButton("redo", self)
self.redo.clicked.connect(self.action.undoStack.redo)
self.redo.move(0,50)
self.tool = QtGui.QPushButton("tool", self)
self.tool.setCheckable(True)
self.tool.clicked[bool].connect(self.new)
self.tool.move(0,100)
layout = QtGui.QVBoxLayout()
layout.addWidget(self.view)
self.setLayout(layout)
self.show()
def new(self):
pass
class Action(QtGui.QGraphicsScene):
def __init__(self, scene):
QtGui.QGraphicsScene.__init__(self)
self.undoStack = QtGui.QUndoStack(self)
self.scene = scene
self.scene.addLine(0,0,150,150)
self.undoStack = QtGui.QUndoStack(self)
def mousePressEvent(self, event):
if event.button() == QtCore.Qt.LeftButton:
self.start = event.pos()
self.scene.addLine(0,0,150,250)
def mouseReleaseEvent(self, event):
start = QtCore.QPointF(self.mapToScene(self.start))
end = QtCore.QPointF(self.mapToScene(event.pos()))
self.drawing(start, end)
def drawing(self, start, end):
self.line = QtGui.QGraphicsLineItem(QtCore.QLineF(start, end))
self.line.setPen(QtGui.QPen(QtCore.Qt.blue,3,
QtCore.Qt.SolidLine, QtCore.Qt.RoundCap, QtCore.Qt.RoundJoin))
self.scene.addLine(0,0,150,300)
command = Draw(self.line, self.scene)
self.undoStack.push(command)
class Draw(QtGui.QUndoCommand):
def __init__(self, line, scene):
QtGui.QUndoCommand.__init__(self)
self.line = line
self.scene = scene
def redo(self):
self.scene.addItem(self.line)
def undo(self):
self.scene.removeItem(self.line)
class View(QtGui.QGraphicsView):
def __init__(self, parent):
QtGui.QGraphicsView.__init__(self, parent)
self.scene = QtGui.QGraphicsScene()
self.action = Action(self.scene)
self.setScene(self.scene)
self.setSceneRect(QtCore.QRectF(self.viewport().rect()))
def main():
app = QtGui.QApplication(sys.argv)
ex = Example()
sys.exit(app.exec_())
if __name__ == '__main__':
main()
我的问题是我无法使用鼠标绘制任何东西。如果组合Action和View一起工作正常,但是当我把它们放好时没有任何事情发生(也没有错误)。我做这种分离的原因是我很简单想要添加另一个具有其他功能的类(绘制elipses,rects ...)并将它们与Action类交换。我在场景中添加了一行(在Action init 中)并且它被绘制为corectly,但MouseEvents根本不起作用。按钮"工具"是为了更改绘图工具。
在我看来,这是获得在同一画布(场景)上使用不同工具进行涂鸦的最佳方法。我的方式不错,所以我在寻求帮助。我该怎么做才能解决这个问题,所以它可以按照我的意愿运行?错误在哪里?或者整个方法可能是错误的,它应该以另一种方式完成?我使用PYQT 4和Python 3.4。
答案 0 :(得分:1)
您当前的代码在QGraphicsScene
内创建了View.__init__
的两个实例。一个是标准QGraphicsScene
,另一个是您的子类Action
。但是您的视图只能附加到一个场景,因此其中一个是多余的,无法正常运行。您已将其附加到QGraphicsScene
,因此Action
对象无效。
相反,你应该杀掉普通无聊的QGraphicsScene
,并且只实例化Action
类。在Action
的调用中使用实例化的view.setScene()
对象。同样,在Action
类中,不需要传递另一个场景。只需使用自己(所以在self.scene
类中用self
替换Action
的所有实例