Newer
Older
self.nextmask.set_data((self.nextplotmask % 10 +1 )*(self.nextplotmask != 0))
self.ax3.draw_artist(self.nextmask)
# Plot cell numbers
self.ShowCellNumbers()
"""
In order to just display the cells so regions with value > 0
and also to assign to each of the cell values one color,
the modulo 10 of the value is take and we add 1, to distinguish
the values of 10,20,30,... from the background (although the bckgrnd
gets with the addition the value 1) and the result of the
modulo is multiplied with a matrix containing a False value for the
background coordinates, setting the background to 0 again.
"""
if flag:
self.currmask.set_data((self.plotmask%10+1)*(self.plotmask!=0))
self.currmask.set_data((self.tempmask%10+1)*(self.tempmask!=0))
# show the updates by redrawing the array using draw_artist, it is faster
# to use as it only redraws the array itself, and not everything else.
self.ax.draw_artist(self.currplot)
self.ax.draw_artist(self.currmask)
self.update()
self.flush_events()
def _getCellCenters(self, plotmask):
"""Get approximate locations for cell centers"""
vals = np.unique(plotmask).astype(int)
vals = np.delete(vals,np.where(vals==0))
xtemp = []
ytemp = []
for k in vals:
y,x = (plotmask==k).nonzero()
sample = np.random.choice(len(x), size=20, replace=True)
meanx = np.mean(x[sample])
meany = np.mean(y[sample])
xtemp.append(int(round(meanx)))
ytemp.append(int(round(meany)))
return vals, xtemp, ytemp
def ShowCellNumbers(self):
"""Checks whether to show cell numbers, and does so if button is
checked"""
if self.button_showval_check.isChecked():
self.ShowCellNumbersCurr()
self.ShowCellNumbersNext()
self.ShowCellNumbersPrev()
"""This function is called to display the cell values and it
takes 10 random points inside of the cell, computes the mean of these
points and this gives the coordinate where the number will be
displayed. The number to be displayed is just given by the value
in the mask of the cell.
This function is just used for the current time subplot.
vals, xtemp, ytemp = self._getCellCenters(self.plotmask)
if xtemp:
for i in range(0,len(xtemp)):
ann = self.ax.annotate(str(int(vals[i])), (xtemp[i], ytemp[i]))
self.ann_list.append(ann)
self.draw()
"""This function is called to display the cell values and it
takes 10 random points inside of the cell, computes the mean of these
points and this gives the coordinate where the number will be
displayed. The number to be displayed is just given by the value
in the mask of the cell.
This function is just used for the previous time subplot.
"""
vals, xtemp, ytemp = self._getCellCenters(self.prevplotmask)
if xtemp:
for i in range(0,len(xtemp)):
ann = self.ax2.annotate(str(vals[i]), (xtemp[i], ytemp[i]))
self.ann_list_prev.append(ann)
self.draw()
"""This function is called to display the cell values and it
takes 10 random points inside of the cell, computes the mean of these
points and this gives the coordinate where the number will be
displayed. The number to be displayed is just given by the value
in the mask of the cell.
This function is just used for the next time subplot.
"""
for i,a in enumerate(self.ann_list_next):
vals, xtemp, ytemp = self._getCellCenters(self.nextplotmask)
if xtemp:
for i in range(0,len(xtemp)):
ann = self.ax3.annotate(str(vals[i]), (xtemp[i], ytemp[i]))
self.ann_list_next.append(ann)
self.draw()
"""
it updates the plot once the user clicks on the plot and draws a 4x4 pixel dot
at the coordinate of the click
"""
# remove the first coordinate as it should only coorespond
# to the value that the user wants to attribute to the drawn region
xtemp, ytemp = self.storemouseclicks[0]
# here we initialize the value attributed to the pixels.
# it means that the first click selects the value that will be attributed to
# the pixels inside the polygon (drawn by the following mouse clicks of the user)
self.cellval = self.plotmask[ytemp, xtemp]
# drawing the 2x2 square ot of the mouse click
if ((self.button_newcell_check.isChecked() or self.button_drawmouse_check.isChecked())
and self.cellval == 0):
self.tempmask[posy:posy+2, posx:posx+2] = 9
else:
self.tempmask[posy:posy+2,posx:posx+2] = self.cellval
# plot the mouseclick
self.updatedata(False)
this method is used to draw either a new cell (flag = true) or to add a region to
an existing cell (flag = false). The flag will just be used to set the
value of pixels (= self.cellval) in the drawn region.
If flag = true, then the value will be the maximal value plus 1. Such
that it attributes a new value to the new cell.
If flag = false, then it will use the value of the first click to set
the value of the pixels in the new added region.
# here the values that have been changed to mark the mouse clicks are
# restored such that they don't appear when the region/new cell is
# drawn.
# if new cell is added, it sets the value of the drawn pixels to a new value
# corresponding to the new cell
# The first value is taken out as it is just used to set the value
# to the new region.
# if only two points or less have been click, it cannot make a area
# so it justs discards these values and returns.
self.storemouseclicks = list(self.storemouseclicks)
self.storemouseclicks.clear()
self.updatedata(True)
return
# add the first point because to use the path function, one has to close
# the path by returning to the initial point.
self.storemouseclicks.append(self.storemouseclicks[0])
# codes are requested by the path function in order to make a polygon
# out of the points that have been selected.
codes = np.zeros(len(self.storemouseclicks))
codes[0] = Path.MOVETO
codes[len(codes)-1]= Path.CLOSEPOLY
codes[1:len(codes)-1] = Path.LINETO
codes = list(codes)
# out of the coordinates of the mouse clicks and of the code, it makes
# a path/contour which corresponds to the added region/new cell.
self.storemouseclicks = np.array(self.storemouseclicks)
# Take a square around the drawn region, where the drawn region fits inside.
minx = min(self.storemouseclicks[:,0])
maxx = max(self.storemouseclicks[:,0])
miny = min(self.storemouseclicks[:,1])
maxy= max(self.storemouseclicks[:,1])
# creates arrays of coordinates of the whole square surrounding
# the drawn region
array_x = np.arange(minx, maxx, 1)
array_y = np.arange(miny, maxy, 1)
# takes all the coordinates to couple them and store them in array_coord
for xi in range(0,len(array_x)):
for yi in range(0,len(array_y)):
array_coord.append((array_x[xi], array_y[yi]))
# path_contains_points returns an array of bool values
# where for each coordinates it tests if it is inside the path
pix_inside_path = path.contains_points(array_coord)
# for each coordinate where the contains_points method returned true
# the value of the self.currpicture matrix is changed, it draws the region
# defined by the user
if pix_inside_path[j]:
x,y = array_coord[j]
self.plotmask[y,x]= self.cellval
# once the self.currpicture has been updated it is drawn by callinf the
# updatedata method.
self.storemouseclicks = list(self.storemouseclicks)
# empty the lists ready for the next region to be drawn.
if __name__ == '__main__':
app = QApplication(sys.argv)
# If two arguments are given, make them nd2name and hdfname
if len(sys.argv)==3:
nd2name1 = sys.argv[1]
hdfname1 = sys.argv[2]
ex = App(nd2name1, hdfname1, '')
wind = dfb.FileBrowser()
if wind.exec_():
nd2name1 = wind.nd2name
hdfname1 = wind.hdfname
hdfnewname = wind.newhdfentry.text()
ex = App(nd2name1, hdfname1, hdfnewname)
sys.exit(app.exec_())
else:
app.exit()