当前位置:   article > 正文

Python3使用PyQt5制作简单的画板/手写板

pyqt 画板

0.目录

1.前言
2.简单的画板1.0
  • 在定点和移动中的鼠标所在处画一条线
3.简单的画板2.0
  • 在定点和移动中的鼠标所在处画一条线
  • 并将画过的线都保留在窗体上
4.简单的画板3.0
  • 将按住鼠标后移动的轨迹保留在窗体上
5.简单的画板4.0
  • 将按住鼠标后移动的轨迹保留在窗体上
  • 并解决二次作画时与上次痕迹连续的问题

1.前言

版本:Python3.6.1 + PyQt5

写一个程序的时候需要用到画板/手写板,只需要最简单的那种。原以为网上到处都是,结果找了好几天,都没有找到想要的结果。
网上的要么是非python版的qt程序(要知道qt版本之间差异巨大,还是非同一语言的),改写难度太大。要么是PyQt4的老程序,很多都已经不能在PyQt5上运行了。要么是大神写的特别复杂的程序,简直是直接做出了一个Windows自带的画图版,只能膜拜~

于是我只能在众多代码中慢慢寻找自己需要的那一小部分,然后不断地拼凑,不断地理解大神的代码,最终做出这么一个简单的画板。望着这个简单的画板我真是泪流满面,中间数十次拼不对拼不全导致程序无数次崩溃,差点就放弃了......

以下是参考网站的名单(名单不分先后顺序):
1.PyQt5教程(九)——绘图https://my.oschina.net/wisedream/blog/549989
2.Qt入门 小程序之画图板http://blog.csdn.net/doraemon___/article/details/53026890
3.使用Python编写简单的画图板程序的示例教程http://www.jb51.net/article/76067.htm
4.qt中函数paintEvent(QPaintEvent*)是不是被系统自动调用的https://zhidao.baidu.com/question/1509518984399472660.html
5.从Qt到PyQthttp://www.cnblogs.com/Finley/p/5268861.html
6.python3+PyQt5 重新实现QT事件处理程序http://blog.sina.com.cn/s/blog_c22e36090102wzxq.html
7.PyQt5入门http://www.docin.com/p-1560265564.html
8.PyQt5教程(一)——第一个PyQt5程序https://my.oschina.net/wisedream/blog/536052
9.PyQt之布局&无边框&信号http://www.cnblogs.com/codeAB/p/5019439.html
10.QT界面简单的图形移动和鼠标绘图http://blog.csdn.net/K54387/article/details/77926313
11.qt鼠标事件总结(转)http://blog.sina.com.cn/s/blog_8b97b05e0100v6kk.html
12.一个简单的画图程序http://blog.csdn.net/cutter_point/article/details/43087497
13.Qt双缓冲机制:实现一个简单的绘图工具(纯代码实现)http://blog.csdn.net/rl529014/article/details/51658350

2.简单的画板1.0

在简单的画板1.0这里,实现的功能是:在定点和移动中的鼠标所在处画一条线
如图所示:
1250397-20171018210508381-248522440.png
鼠标按住移动的话,线也会跟着移动,从这个简单的程序开始理解PyQt5的运行机制吧。

  1. '''
  2. 简单的画板1.0
  3. 功能:在定点和移动中的鼠标所在处画一条线
  4. 作者:PyLearn
  5. 博客: http://www.cnblogs.com/PyLearn/
  6. 最后修改日期: 2017/10/18
  7. '''
  8. import sys
  9. from PyQt5.QtWidgets import (QApplication, QWidget)
  10. from PyQt5.QtGui import (QPainter, QPen)
  11. from PyQt5.QtCore import Qt
  12. class Example(QWidget):
  13. def __init__(self):
  14. super(Example, self).__init__()
  15. #resize设置宽高,move设置位置
  16. self.resize(400, 300)
  17. self.move(100, 100)
  18. self.setWindowTitle("简单的画板1.0")
  19. #setMouseTracking设置为False,否则不按下鼠标时也会跟踪鼠标事件
  20. self.setMouseTracking(False)
  21. #设置两个变量接收移动中的点的x、y坐标
  22. self.pos_x = 20
  23. self.pos_y = 20
  24. def paintEvent(self, event):
  25. painter = QPainter()
  26. painter.begin(self)
  27. pen = QPen(Qt.black, 2, Qt.SolidLine)
  28. painter.setPen(pen)
  29. #定点(20, 20) 到 (self.pos_x, self.pos_y)之间画线
  30. painter.drawLine(20, 20, self.pos_x, self.pos_y)
  31. painter.end()
  32. def mouseMoveEvent(self, event):
  33. '''
  34. 按住鼠标移动事件:更新pos_x和pos_y的值
  35. 调用update()函数在这里相当于调用paintEvent()函数
  36. 每次update()时,之前调用的paintEvent()留下的痕迹都会清空
  37. '''
  38. self.pos_x = event.pos().x()
  39. self.pos_y = event.pos().y()
  40. self.update()
  41. if __name__ == "__main__":
  42. app = QApplication(sys.argv)
  43. pyqt_learn = Example()
  44. pyqt_learn.show()
  45. app.exec_()

3.简单的画板2.0

从以上的简单的画板1.0程序的运行可以发现,按住鼠标移动的时候,线也会跟着移动,那如何让之前的线留下痕迹,而不是消失呢?
在简单的画板2.0中,使用一个列表保存所有移动过的点,然后要画线的时候,循环遍历列表,依次画出列表中点到定点之间的线即可。
效果如图所示:
1250397-20171018211314490-1654661333.png

  1. '''
  2. 简单的画板2.0
  3. 功能:
  4. 在定点和移动中的鼠标所在处画一条线
  5. 并将画过的线都保留在窗体上
  6. 作者:PyLearn
  7. 博客: http://www.cnblogs.com/PyLearn/
  8. 最后修改日期: 2017/10/18
  9. '''
  10. import sys
  11. from PyQt5.QtWidgets import (QApplication, QWidget)
  12. from PyQt5.QtGui import (QPainter, QPen)
  13. from PyQt5.QtCore import Qt
  14. class Example(QWidget):
  15. def __init__(self):
  16. super(Example, self).__init__()
  17. #resize设置宽高,move设置位置
  18. self.resize(400, 300)
  19. self.move(100, 100)
  20. self.setWindowTitle("简单的画板2.0")
  21. #setMouseTracking设置为False,否则不按下鼠标时也会跟踪鼠标事件
  22. self.setMouseTracking(False)
  23. '''
  24. 要想将画过的线都保留在窗体上
  25. 需要一个列表来保存所有移动过的点
  26. '''
  27. self.pos_xy = []
  28. def paintEvent(self, event):
  29. painter = QPainter()
  30. painter.begin(self)
  31. pen = QPen(Qt.black, 2, Qt.SolidLine)
  32. painter.setPen(pen)
  33. #循环遍历self.pos_xy中每个点,然后画点到定点之间的线
  34. for pos_tmp in self.pos_xy:
  35. painter.drawLine(20, 20, pos_tmp[0], pos_tmp[1])
  36. painter.end()
  37. def mouseMoveEvent(self, event):
  38. '''
  39. 按住鼠标移动事件:将当前点添加到pos_xy列表中
  40. 调用update()函数在这里相当于调用paintEvent()函数
  41. 每次update()时,之前调用的paintEvent()留下的痕迹都会清空
  42. '''
  43. #中间变量pos_tmp提取当前点
  44. pos_tmp = (event.pos().x(), event.pos().y())
  45. #pos_tmp添加到self.pos_xy中
  46. self.pos_xy.append(pos_tmp)
  47. self.update()
  48. if __name__ == "__main__":
  49. app = QApplication(sys.argv)
  50. pyqt_learn = Example()
  51. pyqt_learn.show()
  52. app.exec_()

4.简单的画板3.0

好了,接下来进入正题了。简单的画板2.0不过是画鼠标所在点到定点的线,那么如何将按住鼠标后移动的轨迹保留在窗体上?
这个就需要一个列表来保存所有移动过的点,然后把所有相邻两个点之间都画一条线,就能断断续续连成鼠标的痕迹了。
效果如图所示:
1250397-20171018211840365-1701304713.png
是不是就画出鼠标移动的轨迹了!
不过这也是有缺点的,比如说写个5看看:
1250397-20171018212046302-564988239.png
硬生生变成了一个5不是5, 6不是6的数字。这是因为再次提笔画时,5上面的那一横跟之前画的尾巴那里连起来了。好好想想,这个问题怎么解决呢?

  1. '''
  2. 简单的画板3.0
  3. 功能:将按住鼠标后移动的轨迹保留在窗体上
  4. 作者:PyLearn
  5. 博客: http://www.cnblogs.com/PyLearn/
  6. 最后修改日期: 2017/10/18
  7. '''
  8. import sys
  9. from PyQt5.QtWidgets import (QApplication, QWidget)
  10. from PyQt5.QtGui import (QPainter, QPen)
  11. from PyQt5.QtCore import Qt
  12. class Example(QWidget):
  13. def __init__(self):
  14. super(Example, self).__init__()
  15. #resize设置宽高,move设置位置
  16. self.resize(400, 300)
  17. self.move(100, 100)
  18. self.setWindowTitle("简单的画板3.0")
  19. #setMouseTracking设置为False,否则不按下鼠标时也会跟踪鼠标事件
  20. self.setMouseTracking(False)
  21. '''
  22. 要想将按住鼠标后移动的轨迹保留在窗体上
  23. 需要一个列表来保存所有移动过的点
  24. '''
  25. self.pos_xy = []
  26. def paintEvent(self, event):
  27. painter = QPainter()
  28. painter.begin(self)
  29. pen = QPen(Qt.black, 2, Qt.SolidLine)
  30. painter.setPen(pen)
  31. '''
  32. 首先判断pos_xy列表中是不是至少有两个点了
  33. 然后将pos_xy中第一个点赋值给point_start
  34. 利用中间变量pos_tmp遍历整个pos_xy列表
  35. point_end = pos_tmp
  36. 画point_start到point_end之间的线
  37. point_start = point_end
  38. 这样,不断地将相邻两个点之间画线,就能留下鼠标移动轨迹了
  39. '''
  40. if len(self.pos_xy) > 1:
  41. point_start = self.pos_xy[0]
  42. for pos_tmp in self.pos_xy:
  43. point_end = pos_tmp
  44. painter.drawLine(point_start[0], point_start[1], point_end[0], point_end[1])
  45. point_start = point_end
  46. painter.end()
  47. def mouseMoveEvent(self, event):
  48. '''
  49. 按住鼠标移动事件:将当前点添加到pos_xy列表中
  50. 调用update()函数在这里相当于调用paintEvent()函数
  51. 每次update()时,之前调用的paintEvent()留下的痕迹都会清空
  52. '''
  53. #中间变量pos_tmp提取当前点
  54. pos_tmp = (event.pos().x(), event.pos().y())
  55. #pos_tmp添加到self.pos_xy中
  56. self.pos_xy.append(pos_tmp)
  57. self.update()
  58. if __name__ == "__main__":
  59. app = QApplication(sys.argv)
  60. pyqt_learn = Example()
  61. pyqt_learn.show()
  62. app.exec_()

5.简单的画板4.0

简单的画板3.0中有一个致命的问题,那就是连续的问题,比如说要写一个三位数123:
1250397-20171018212413771-1371572092.png
很难看对不对?

解决这个问题的方法应该是有很多种的,我也没有深入想,就直接用了这个麻烦点的方法。

我的办法是当鼠标按住移动然后松开的时候,往保存所有移动过的点的列表中添加一个断点(-1, -1)。然后在每次画线的时候,都判断一下是不是断点,如果是断点的话就想办法跳过去,并且不连续的开始接着画线。
效果如图所示:
1250397-20171018212938740-1011784932.png
以下是具体实现代码:

  1. '''
  2. 简单的画板4.0
  3. 功能:
  4. 将按住鼠标后移动的轨迹保留在窗体上
  5. 并解决二次作画时与上次痕迹连续的问题
  6. 作者:PyLearn
  7. 博客: http://www.cnblogs.com/PyLearn/
  8. 最后修改日期: 2017/10/18
  9. '''
  10. import sys
  11. from PyQt5.QtWidgets import (QApplication, QWidget)
  12. from PyQt5.QtGui import (QPainter, QPen)
  13. from PyQt5.QtCore import Qt
  14. class Example(QWidget):
  15. def __init__(self):
  16. super(Example, self).__init__()
  17. #resize设置宽高,move设置位置
  18. self.resize(400, 300)
  19. self.move(100, 100)
  20. self.setWindowTitle("简单的画板4.0")
  21. #setMouseTracking设置为False,否则不按下鼠标时也会跟踪鼠标事件
  22. self.setMouseTracking(False)
  23. '''
  24. 要想将按住鼠标后移动的轨迹保留在窗体上
  25. 需要一个列表来保存所有移动过的点
  26. '''
  27. self.pos_xy = []
  28. def paintEvent(self, event):
  29. painter = QPainter()
  30. painter.begin(self)
  31. pen = QPen(Qt.black, 2, Qt.SolidLine)
  32. painter.setPen(pen)
  33. '''
  34. 首先判断pos_xy列表中是不是至少有两个点了
  35. 然后将pos_xy中第一个点赋值给point_start
  36. 利用中间变量pos_tmp遍历整个pos_xy列表
  37. point_end = pos_tmp
  38. 判断point_end是否是断点,如果是
  39. point_start赋值为断点
  40. continue
  41. 判断point_start是否是断点,如果是
  42. point_start赋值为point_end
  43. continue
  44. 画point_start到point_end之间的线
  45. point_start = point_end
  46. 这样,不断地将相邻两个点之间画线,就能留下鼠标移动轨迹了
  47. '''
  48. if len(self.pos_xy) > 1:
  49. point_start = self.pos_xy[0]
  50. for pos_tmp in self.pos_xy:
  51. point_end = pos_tmp
  52. if point_end == (-1, -1):
  53. point_start = (-1, -1)
  54. continue
  55. if point_start == (-1, -1):
  56. point_start = point_end
  57. continue
  58. painter.drawLine(point_start[0], point_start[1], point_end[0], point_end[1])
  59. point_start = point_end
  60. painter.end()
  61. def mouseMoveEvent(self, event):
  62. '''
  63. 按住鼠标移动事件:将当前点添加到pos_xy列表中
  64. 调用update()函数在这里相当于调用paintEvent()函数
  65. 每次update()时,之前调用的paintEvent()留下的痕迹都会清空
  66. '''
  67. #中间变量pos_tmp提取当前点
  68. pos_tmp = (event.pos().x(), event.pos().y())
  69. #pos_tmp添加到self.pos_xy中
  70. self.pos_xy.append(pos_tmp)
  71. self.update()
  72. def mouseReleaseEvent(self, event):
  73. '''
  74. 重写鼠标按住后松开的事件
  75. 在每次松开后向pos_xy列表中添加一个断点(-1, -1)
  76. 然后在绘画时判断一下是不是断点就行了
  77. 是断点的话就跳过去,不与之前的连续
  78. '''
  79. pos_test = (-1, -1)
  80. self.pos_xy.append(pos_test)
  81. self.update()
  82. if __name__ == "__main__":
  83. app = QApplication(sys.argv)
  84. pyqt_learn = Example()
  85. pyqt_learn.show()
  86. app.exec_()

至此,终于完成了简单的画板程序的实现!
另外,如果在使用这个代码的过程中有遇到什么问题,也欢迎向我反馈。

转载于:https://www.cnblogs.com/PyLearn/p/7689170.html

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/笔触狂放9/article/detail/532730
推荐阅读
相关标签
  

闽ICP备14008679号