python鼠标绘图附代码
作者:陈杉菜 发布时间:2021-09-09 17:10:26
标签:python,鼠标绘图
github指路
作业要求
友情提示
ldw老师给の友情提示(虽然感觉也还好/dbq其实还挺有用的)
课上讲的例子是图片展示器(能够实现打开图片+镜像显示图片+保存图片+操作日志的功能)
讲真那啥猫咪图片还挺好看
演示之后差不多是这样👇
分析
然后按照上面这个样例我们要做一个鼠标绘图的gui界面(这就有点难度了
先分析一下就是👇
(我本来是想保留日志和水平翻转垂直翻转这种操作的/后来放弃了/有点难/而且……那啥……我做不到在现有的图上绘制内容就很抱歉)
老师上课其实有提到说那啥控件要改(但是我改完之后貌似不对
然后开始干活
结果呈现
悬浮状态
2020/02/04 更新code
大声说话,文件路径&命名(((小声逃走
#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time : 2019/12/3 14:22
# @Author : Chen Shan
# Function :GUI programming - a naive Sketchpad tool
import os
import platform
import sys
from PyQt5.QtCore import *
from PyQt5.QtGui import *
from PyQt5.QtWidgets import *
class PenWidthDlg(QDialog):
def __init__(self, parent=None):
super(PenWidthDlg, self).__init__(parent)
widthLabel = QLabel("宽度:")
self.widthSpinBox = QSpinBox()
widthLabel.setBuddy(self.widthSpinBox)
self.widthSpinBox.setAlignment(Qt.AlignRight|Qt.AlignVCenter)
self.widthSpinBox.setRange(0, 50)
okButton = QPushButton("ok")
cancelButton = QPushButton("cancle")
layout = QGridLayout()
layout.addWidget(widthLabel,0,0)
layout.addWidget(self.widthSpinBox,0,1)
layout.addWidget(okButton,1,0)
layout.addWidget(cancelButton,1,1)
self.setLayout(layout)
self.setWindowTitle("宽度设置")
okButton.clicked.connect(self.accept)
cancelButton.clicked.connect(self.reject)
class myMainWindow(QMainWindow):
def __init__(self,parent=None):
super().__init__(parent)
self.setWindowTitle("draw")
self.pix=QPixmap()
self.lastPoint=QPoint()
self.endPoint=QPoint()
#初始化参数
self.initData()
#清空画布
self.initView()
#菜单栏
self.Menu = self.menuBar().addMenu("菜单")
#清空
self.ClearAction = QAction(QIcon("images/clear.png"), "清空", self)
self.ClearAction.triggered.connect(self.initView)
self.Menu.addAction(self.ClearAction)
#调画笔颜色
self.changeColor = QAction(QIcon("images/icon.png"), "颜色", self)
self.changeColor.triggered.connect(self.showColorDialog)
self.Menu.addAction(self.changeColor)
#调画笔粗细
self.changeWidth = QAction(QIcon("images/width.png"), "宽度", self)
self.changeWidth.triggered.connect(self.showWidthDialog)
self.Menu.addAction(self.changeWidth)
# #右侧停靠窗口
# logDockWidget=QDockWidget("Log",self)
# logDockWidget.setAllowedAreas(Qt.LeftDockWidgetArea|Qt.RightDockWidgetArea)
# self.listWidget=QListWidget()
# logDockWidget.setWidget(self.listWidget)
# self.addDockWidget(Qt.RightDockWidgetArea,logDockWidget)
#各种动作
self.fileOpenAction = QAction(QIcon("images/fileopen.png"), "&Open", self)
self.fileOpenAction.setShortcut(QKeySequence.Open)
self.fileOpenAction.setToolTip("Open an image.")
self.fileOpenAction.setStatusTip("Open an image.")
self.fileOpenAction.triggered.connect(self.fileOpen)
self.fileSaveAction = QAction(QIcon("images/filesave.png"), "&Save", self)
self.fileSaveAction.setShortcut(QKeySequence.Save)
self.fileSaveAction.setToolTip("Save an image.")
self.fileSaveAction.setStatusTip("Save an image.")
self.fileSaveAction.triggered.connect(self.fileSaveAs)
#工具栏
fileToolbar = self.addToolBar("文件")
fileToolbar.addAction(self.fileOpenAction)
fileToolbar.addAction(self.fileSaveAction)
editToolbar = self.addToolBar("清空")
editToolbar.addAction(self.ClearAction)
colorToolbar = self.addToolBar("颜色")
colorToolbar.addAction(self.changeColor)
widthToolbar = self.addToolBar("宽度")
widthToolbar.addAction(self.changeWidth)
#状态栏
self.sizeLabel=QLabel()
self.sizeLabel.setFrameStyle(QFrame.StyledPanel|QFrame.Sunken)
status=self.statusBar()
status.setSizeGripEnabled(False)
status.addPermanentWidget(self.sizeLabel)
status.showMessage("Ready",5000)
def initData(self):
self.size = QSize(1000,1040)
self.pixmap = QPixmap(self.size)
self.dirty = False
self.filename = None
self.recentFiles = []
#新建画笔
self.width = 5
self.color = QColor(0, 0, 0)
self.pen = QPen() # 实例化画笔对象
self.pen.setColor(self.color) #设置画笔颜色
self.pen = QPen(Qt.SolidLine) #实例化画笔对象.参数:画笔样式
self.pen.setWidth(self.width) #设置画笔粗细
#新建绘图工具
self.painter = QPainter(self.pixmap)
self.painter.setPen(self.pen)
#鼠标位置
self.__lastPos = QPoint(0,0)#上一次鼠标位置
self.__currentPos = QPoint(0,0)#当前的鼠标位置
self.image = QImage()
def initView(self):
#设置界面的尺寸为__size
self.Clear()
self.imageLabel = QLabel()
self.imageLabel.setPixmap(self.pixmap)
self.setCentralWidget(self.imageLabel)
def Clear(self):
#清空画板
self.pixmap.fill(Qt.white)
self.update()
self.dirty = False
def mousePressEvent(self,event):
#鼠标按下时,获取鼠标的当前位置保存为上一次位置
pointX = event.globalX()
pointY = event.globalY()
self.__currentPos = QPoint(pointX,pointY)
self.dirty = True
self.__currentPos = event.pos()
self.__lastPos = self.__currentPos
def mouseMoveEvent(self,event):
#鼠标移动时,更新当前位置,并在上一个位置和当前位置间画线
self.__currentPos = event.pos()
#pointX = event.globalX()
#pointY = event.globalY()
#self.__currentPos = QPoint(pointX,pointY)
#画线
#用begin和end抱起来,表示针对这个对象,就可以在pixmap有图的情况下继续画画
self.painter.begin(self.pixmap)
self.painter.setPen(self.pen)
self.painter.drawLine(self.__lastPos, self.__currentPos)
self.__lastPos = self.__currentPos
self.painter.end()
self.update() #更新显示
self.imageLabel.setPixmap(self.pixmap)
#调画笔颜色
def showColorDialog(self):
col = QColorDialog.getColor()
self.pen.setColor(col)
self.painter.setPen(self.pen)
def updateWidth(self):
self.pen.setWidth(self.width)
self.painter.setPen(self.pen)
def showWidthDialog(self):
dialog = PenWidthDlg(self)
dialog.widthSpinBox.setValue(self.width)
if dialog.exec_():
self.width = dialog.widthSpinBox.value()
self.updateWidth()
###########################################################
def okToContinue(self): #警告当前图像未保存
if self.dirty:
reply = QMessageBox.question(self,
"Image Changer - Unsaved Changes",
"图片已被更改,请问要保存吗?",
QMessageBox.Yes|QMessageBox.No|QMessageBox.Cancel)
if reply == QMessageBox.Cancel:
return False
elif reply == QMessageBox.Yes:
return self.fileSaveAs()
return True
def fileOpen(self):
if not self.okToContinue():
return
dir = (os.path.dirname(self.filename)
if self.filename is not None else ".")
formats = (["*.{}".format(format.data().decode("ascii").lower())
for format in QImageReader.supportedImageFormats()])
fname = QFileDialog.getOpenFileName(self,
"Image Changer - Choose Image", dir,
"Image files ({})".format(" ".join(formats)))
if fname:
print(fname[0])
self.loadFile(fname[0])
self.updateFileMenu()
def loadFile(self, fname=None):
if fname is None:
action = self.sender()
if isinstance(action, QAction):
fname = action.data()
if not self.okToContinue():
return
else:
return
if fname:
self.filename = None
image = QImage(fname)
if image.isNull():
message = "Failed to read {}".format(fname)
else:
self.addRecentFile(fname)
self.image = QImage()
#self.editUnMirrorAction.setChecked(True)
self.image = image
self.filename = fname
self.showImage()
self.dirty = False
self.sizeLabel.setText("{} x {}".format(
image.width(), image.height()))
message = "Loaded {}".format(os.path.basename(fname))
self.updateStatus(message)
def updateStatus(self, message):
self.statusBar().showMessage(message, 5000)
#self.listWidget.addItem(message)
if self.filename:
self.setWindowTitle("Image Changer - {}[*]".format(
os.path.basename(self.filename)))
elif not self.image.isNull():
self.setWindowTitle("Image Changer - Unnamed[*]")
else:
self.setWindowTitle("Image Changer[*]")
self.setWindowModified(self.dirty)
def updateFileMenu(self):
self.Menu.clear()
self.Menu.addAction(self.fileOpenAction)
self.Menu.addAction(self.fileSaveAction)
current = self.filename
recentFiles = []
print(self.recentFiles)
for fname in self.recentFiles:
if fname != current and QFile.exists(fname):
recentFiles.append(fname)
if recentFiles:
self.fileMenu.addSeparator()
for i, fname in enumerate(recentFiles):
action = QAction(QIcon("images/icon.png"),
"&{} {}".format(i + 1, QFileInfo(
fname).fileName()), self)
action.setData(fname)
action.triggered.connect(lambda: self.loadFile(fname))
self.fileMenu.addAction(action)
def addRecentFile(self, fname):
if fname is None:
return
if fname not in self.recentFiles:
if len(self.recentFiles) < 10:
self.recentFiles = [fname] + self.recentFiles
else:
self.recentFiles = [fname] + self.recentFiles[:8]
print(len(self.recentFiles))
def fileSaveAs(self):
savePath = QFileDialog.getSaveFileName(self, 'Save Your Paint', '.\\', '*.png')
print(savePath)
if savePath[0] == "":
print("Save cancel")
return
image = self.pixmap
print("save...")
image.save(savePath[0])
self.updateStatus("Saved as {}".format(savePath))
def showImage(self, percent=None):
if self.image.isNull():
return
self.pixmap = QPixmap.fromImage(self.image)
self.imageLabel.setPixmap(self.pixmap)
app=QApplication(sys.argv)
form=myMainWindow()
form.setMinimumSize(1000,1000)
form.show()
app.exec_()
来源:https://blog.csdn.net/qq_44702847/article/details/103374914
0
投稿
猜你喜欢
- 一. 网页挂马的概念: 网页挂马是指:在获取网站或者网站服务器的部分或者全部权限后,在网
- Mysql数据库备份和还原常用的命令是进行Mysql数据库备份和还原的关键,没有命令,什么都无从做起,更谈不上什么备份还原,只有给系统这个命
- 今天在看框架的时候无意间看到了document.compatMode,经过一番资料查找,终于搞懂了。文档模式在开发中貌似很少用到,最常见的是
- 【原文地址】 Tip/Trick: Supporting Full Screen Mode with Silverlight 【原文发表日期
- 经过一个阶段的asp学习,下面我们结合所学过的内容建立一个最简单的聊天室,虽然很简单,但是大家可以通过他来掌握一个聊天室建立的基本过程,并且
- Softmax原理Softmax函数用于将分类结果归一化,形成一个概率分布。作用类似于二分类中的Sigmoid函数。对于一个k维向量z,我们
- 我们在编写asp程序时经常会使用到这个功能,一般我们这样判断如:if a="" then ,而这个函数考虑的就比较全面了
- 这篇文章主要介绍了微信小程序 云开发模糊查询实现详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友
- 如下所示:#先下载psutil库:pip install psutilimport psutilimport os,datetime,tim
- 当代码已经写得差不多,发现某个变量名需要修改,但代码中很多地方都有该变量,一一修改太麻烦了,在不同的情景下,可以采取更加简便的方法,如下介绍
- 目标:创建一个字典,记录几对python词语,使用OrderedDict类来写,并按顺序输出。写完报错:[root@centos7 tmp]
- 运行以下代码: Dim com As ADODB.Command Dim rst
- 之前有看过一个博文写的是白社会的设计很好但运营却有些遭,因为对每一个WebGame的推出时间把握不准,会有几个应用同时上线造成影响力的冲突,
- 初学python,抓取搜狗微信公众号文章存入mysqlmysql表:代码:import requestsimport jsonimport
- 假设有这样一个任务,希望对某个文件夹(包括所有子文件夹与文件)中的所有文件进行处理。这就需要遍历整理目录树, 处理遇到的每个文件。impor
- 微软 Office 提供基于 COM 接口的编程。Python 通过 pywin32 可以方便地调用各组件。如果下载和安装 pywin32
- 以前大家谈了很多有关打开数据库连接安全的问题,现在我再提出一种思路:使用activex dll来保护你的代码。(既可以不用为使用共享的加密软
- 1. 确认已经安装了NT/2000和SQL Server的最新补丁程序,不用说大家应该已经安装好了,但是我觉得最好还是在这里提醒一下。2.
- 从MySQL 5.0.2开始,通过mysql_stmt_attr_set() C API函数实现了服务器端光标。服务器端光标允许在服务器端生
- 该章节我们来学习一下在 Python 中去创建并使用多进程的方法,通过学习该章节,我们将可以通过创建多个进程来帮助我们提高脚本执行的效率。可