使用PYQT4向Scribble应用程序添加工具

时间:2015-06-01 18:29:54

标签: python pyqt qgraphicsview qgraphicsscene

我正在尝试编写一个可用作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。

1 个答案:

答案 0 :(得分:1)

您当前的代码在QGraphicsScene内创建了View.__init__的两个实例。一个是标准QGraphicsScene,另一个是您的子类Action。但是您的视图只能附加到一个场景,因此其中一个是多余的,无法正常运行。您已将其附加到QGraphicsScene,因此Action对象无效。

相反,你应该杀掉普通无聊的QGraphicsScene,并且只实例化Action类。在Action的调用中使用实例化的view.setScene()对象。同样,在Action类中,不需要传递另一个场景。只需使用自己(所以在self.scene类中用self替换Action的所有实例