weixin_52310393 2022-05-19 11:49 采纳率: 90%
浏览 75
已结题

怎样擦除已绘制的path?要鼠标控制擦除,不能清空



```python
# -*- coding: utf-8 -*-

# Form implementation generated from reading ui file 'Demo.ui'
#
# Created by: PyQt5 UI code generator 5.15.4
#
# WARNING: Any manual changes made to this file will be lost when pyuic5 is
# run again.  Do not edit this file unless you know what you are doing.
import math
import sys

from PyQt5 import QtCore, QtGui, QtWidgets
from PyQt5.QtCore import Qt, QPoint, QPointF
from PyQt5.QtGui import QPainter, QPen, QPainterPath, QPolygonF, QColor, QPixmap
from PyQt5.QtWidgets import QApplication, QWidget, QFrame, QSizePolicy, QColorDialog, QHBoxLayout, QPushButton, \
    QComboBox, QSpinBox, QFormLayout, QLabel


class MyLabel(QLabel):
    def __init__(self, parent=None):
        super(MyLabel, self).__init__(parent)
        self.flag = None
        self.setLineWidth(1)
        self.setMidLineWidth(0)
        self.setSizePolicy(QSizePolicy.Preferred, QSizePolicy.Preferred)

        self.x0 = 0
        self.y0 = 0
        self.x1 = 0
        self.y1 = 0
        self.penColor = (QColor(255, 0, 0, 50))
        self.lastpenColor = (QColor(255, 0, 0, 50))
        self.status = True
        self.penWidth = 30
        self.penStyle = Qt.SolidLine
        self.capStyle = Qt.RoundCap

        self.pathList = []
        self.path = QPainterPath()

    def setPenColor(self, color):
        self.lastpenColor = color
        self.penColor = color

    def setDrawStatus(self, status):
        if status == True:
            self.penColor = self.lastpenColor
            self.status = True
        elif status == False:
            self.penColor = (QColor(0, 0, 0, 0))
            self.status = False

    def widthChange(self, value):
        self.penWidth = value

    def paintEvent(self, event):
        painter = QPainter(self)
        painter.setRenderHint(QPainter.Antialiasing, True)

        # 绘制边框线
        painter.drawRect(self.rect())

    #    painter.setPen(QPen(self.penColor, self.penWidth, self.penStyle, self.capStyle))
    #    painter.drawLine(self.x0, self.y0, self.x1, self.y1)

        for p, d in self.pathList:
            painter.setPen(QPen(*d))
            painter.drawPath(p)

    def mousePressEvent(self, event):
        self.path = QPainterPath()
        self.pathList.append([self.path, (self.penColor, self.penWidth, self.penStyle, self.capStyle)])
        self.path.moveTo(event.pos())
        self.flag = True
        self.update()


    def mouseMoveEvent(self, event):
        if self.flag:
            self.path.lineTo(event.pos())
            self.update()

    def mouseReleaseEvent(self, event):
        self.flag = False

class Widget(QWidget):
    def __init__(self,parent=None):
        super(Widget,self).__init__(parent)
        self.resize(1000, 700)
        self.initUi()

    def initUi(self):
        self.canvas = MyLabel()
        layout = QHBoxLayout()

        btnPen = QPushButton('画笔')
        btnPen.clicked.connect(self.DrawOn)

        penColor = QComboBox()
        penColor.addItems(['红色 肿瘤','绿色 间质', '紫色 坏死', '蓝色 淋巴'])
        penColor.currentTextChanged.connect(self.ColorChange)

        btnEraser = QPushButton('橡皮擦')
        btnEraser.clicked.connect(self.DrawOff)

        penWidth = QSpinBox()
        penWidth.resize(20,30)
        penWidth.setRange(20,60)
        penWidth.setValue(30)
        penWidth.valueChanged.connect(self.canvas.widthChange)

        fLayout = QFormLayout()
        fLayout.setContentsMargins(0, 0, 0, 0)
        fLayout.addRow(btnPen)
        fLayout.addRow(penColor)
        fLayout.addRow(btnEraser)
        fLayout.addRow(penWidth)

        wid_left = QWidget()
        wid_left.setMaximumWidth(200)
        wid_left.setLayout(fLayout)

        layout.addWidget(wid_left)
        layout.addWidget(self.canvas)

        self.setLayout(layout)

    def DrawOn(self):
        self.canvas.setDrawStatus(True)

    def ColorChange(self, color):
        if color == '红色 肿瘤':
            self.canvas.setPenColor(QColor(255, 0, 0, 50))
        elif color == '绿色 间质':
            self.canvas.setPenColor(QColor(0, 255, 0, 50))
        elif color == '紫色 坏死':
            self.canvas.setPenColor(QColor(153, 0, 255, 50))
        elif color == '蓝色 淋巴':
            self.canvas.setPenColor(QColor(0, 0, 255, 50))

    def DrawOff(self):
        self.canvas.setDrawStatus(False)

if __name__ == '__main__':
    app = QApplication(sys.argv)
    window = Widget()
    window.show()
    sys.exit(app.exec())


  • 写回答

3条回答 默认 最新

  • 关注

    提供两个思路:
    方法一:点击擦除按钮后,再鼠标移动槽函数中,记录鼠标移动的点,并判断点是否再已绘的path中,如果在就把原来的path分割成两个path,并把点从新path中删掉,然后update重绘。
    方法二:点击擦除按钮后,在鼠标移动槽函数中,记录鼠标移动的点,并用画板的背景色绘制这条path(你原来的path怎么绘制的,这个擦除的path也怎么绘制,区别是,擦除的path采用的颜色是画板的背景色)
    两种方法的比较:
    方法一:逻辑会比较复杂,特别是在分割path的时候,频繁的重绘可能影响程序的稳定性,容易出错。
    方法二:逻辑相对简单,但是需要记录每条path的绘制顺序及颜色,如果你之前绘制的时候没有记录这些数据,可能需要调整你的数据结构。
    推荐采用第二种方法。

    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论
查看更多回答(2条)

报告相同问题?

问题事件

  • 系统已结题 5月27日
  • 已采纳回答 5月19日
  • 创建了问题 5月19日

悬赏问题

  • ¥15 c程序不知道为什么得不到结果
  • ¥40 复杂的限制性的商函数处理
  • ¥15 程序不包含适用于入口点的静态Main方法
  • ¥15 素材场景中光线烘焙后灯光失效
  • ¥15 请教一下各位,为什么我这个没有实现模拟点击
  • ¥15 执行 virtuoso 命令后,界面没有,cadence 启动不起来
  • ¥50 comfyui下连接animatediff节点生成视频质量非常差的原因
  • ¥20 有关区间dp的问题求解
  • ¥15 多电路系统共用电源的串扰问题
  • ¥15 slam rangenet++配置