好的,我已经为在Qt GUI中集成matplotlib自行实现了imline...现在,实现imrect等内容也很简单。如果有人需要imrect等内容,我将更新代码。以下是我实现imline的代码:
from PyQt4.QtCore import *
from PyQt4.QtGui import *
import time
import matplotlib as mpl
import matplotlib.pyplot as plt
import numpy as np
import scipy.optimize as opt
class Imline(QObject):
'''
Plot interactive line
'''
def __init__(self, plt, image = None, scale = 1, *args, **kwargs):
'''
Initialize imline
'''
super(Imline, self).__init__(None)
self.__plt = plt
self.scale = scale
self.startX = None
self.startY = None
self.endX = None
self.endY = None
self.__line2d = None
self.mask = None
if(image is not None):
height, width = image.shape
else:
height = None
width = None
self.__width = width
self.__height = height
self.__c1 = self.__plt.figure.canvas.mpl_connect('button_press_event', self.__mousePressEvent)
self.__c2 = self.__plt.figure.canvas.mpl_connect('motion_notify_event', self.__mouseMoveEvent)
self.__c3 = self.__plt.figure.canvas.mpl_connect('button_release_event', self.__mouseReleaseEvent)
self.imlineEventFinished = SIGNAL('imlineEventFinished')
def __mousePressEvent(self, event):
'''
Starting point
'''
xdata = event.xdata
ydata = event.ydata
if((xdata is None) | (ydata is None) | (self.startX is not None) | (self.startY is not None) | (self.endX is not None) | (self.endY is not None)):
return
self.startX = xdata
self.startY = ydata
def __mouseMoveEvent(self, event):
'''
Draw interactive line
'''
xdata = event.xdata
ydata = event.ydata
if((xdata is None) | (ydata is None) | (self.startX is None) | (self.startY is None) | (self.endX is not None) | (self.endY is not None)):
return
if(self.__line2d is not None):
self.__line2d[0].remove()
x = [self.startX, xdata]
y = [self.startY, ydata]
self.__plt.axes.hold(True)
xlim = self.__plt.axes.get_xlim()
ylim = self.__plt.axes.get_ylim()
self.__line2d = self.__plt.axes.plot(x, y, color = [1, 0, 0])
self.__plt.axes.set_xlim(xlim)
self.__plt.axes.set_ylim(ylim)
self.__plt.draw()
self.__plt.show()
def __mouseReleaseEvent(self, event):
'''
End point
'''
xdata = event.xdata
ydata = event.ydata
if((xdata is None) | (ydata is None) | (self.endX is not None) | (self.endY is not None)):
return
if(self.__line2d is not None):
self.__line2d[0].remove()
self.endX = xdata
self.endY = ydata
P = np.polyfit([self.startX, self.endX], [self.startY, self.endY],1 )
self.__m = P[0]
self.__q = P[1]
self.__plt.draw()
self.__plt.show()
self.__plt.figure.canvas.mpl_disconnect(self.__c1)
self.__plt.figure.canvas.mpl_disconnect(self.__c2)
self.__plt.figure.canvas.mpl_disconnect(self.__c3)
self.emit(SIGNAL('imlineEventFinished'))
def createMask(self):
'''
Create mask from painted line
'''
if((self.__height is None) | (self.__width is None)):
return None
mask = np.zeros((self.__height, self.__width))
m = self.__m
q = self.__q
print m, q
startX = np.int(self.startX)
startY = np.int(self.startY)
endX = np.int(self.endX)
endY = np.int(self.endY)
tempStartX = startX
if(startX > endX):
startX = endX
endX = tempStartX
tempStartY = startY
if(startY > endY):
startY = endY
endY = tempStartY
self.startX = startX
self.endX = endX
self.startY = startY
self.endY = endY
xData = np.arange(startX, endX)
yData = np.arange(startY, endY)
for x in xData:
row = round(m*x + q)
if(row < startY):
row = startY
if(row > endY):
row = endY
mask[row, x] = 1
for y in yData:
col = round((y - q) / m)
if(col < startX):
col = startX
if(col > endX):
col = endX
mask[y, col] = 1
mask = mask == 1
return mask
QRubberBand
等)。虽然对于matplotlib也有类似的工具,但如果您将其嵌入到Qt中,则没有必要使用与GUI无关的matplotlib小部件。 - Joe Kington