[docs]classScatter3D:def__init__(self,coordinates,smarties=False,colors=None,hemispheres=None,half_slice_thickness=None):self.__coordinates=Noneself.__has_hemispheres=hemispheresisnotNoneself.__has_colours=colorsisnotNoneorsmartiesself.default_symbol='+'self.alternate_symbol='p'self.half_slice_thickness=half_slice_thicknessself.axis=2ifsmartiesandcolorsisNone:n_samples=coordinates.shape[0]colors=pseudo_random_rgb_array(n_samples)ifcolorsisnotNoneandnotisinstance(colors[0],str):# Convert to hex if not yetcolors=[to_hex(c)forcincolors]ifhemispheresisnotNone:self.symbol_map={0:self.default_symbol,255:self.alternate_symbol}# symbols = symbols.decode()# colors = if colors is None else np.array([QColor( * col.astype(int)) for col in colors]self.data=pd.DataFrame({'x':coordinates[:,0],'y':coordinates[:,1],'z':coordinates[:,2],'hemisphere':hemispheres,'colour':colors})# TODO: could use id instead of colourifcolorsisnotNone:unique_colors=np.unique(colors)self.point_map=pd.DataFrame({'colour':unique_colors,'pen':[pg.mkPen(c)forcinunique_colors],'brush':[pg.mkBrush(c)forcinunique_colors]})@propertydefcoordinates(self):ifself.__coordinatesisNone:self.__coordinates=self.data[['x','y','z']].valuesreturnself.__coordinates
[docs]defset_data(self,df):# print(self.data['colour'].values, df['colour'])ifisinstance(df,dict):df=pd.DataFrame(df)if'colour'indf.columns:ifisinstance(df['colour'][0],np.ndarray):# TODO: should be iterabledf['colour']=[to_hex(c)forcindf['colour']]# OPTIMISE: see mapelifisinstance(df['colour'][0],QColor):df['colour']=[c.name()forcindf['colour']]# OPTIMISE: see mapunique_colors=np.unique(df['colour'])self.point_map=pd.DataFrame({'colour':unique_colors,'pen':[pg.mkPen(c)forcinunique_colors],'brush':[pg.mkBrush(c)forcinunique_colors]})ifset(self.data.columns)>=set(df.columns):forcolinset(self.data.columns)-set(df.columns):# Add if missingdf[col]=Noneself.data=df# print(self.data['colour'].values, df['colour'].values)self.__coordinates=None
[docs]defget_all_data(self,main_slice_idx,half_slice_thickness=3):# FIXME: rename""" Get surrounding markers Parameters ---------- main_slice_idx int half_slice_thickness int Returns ------- """ifnothalf_slice_thickness:# WARNING: optimisationoutput={'pos':np.empty(0),'size':np.empty(0),}ifself.has_colours:output['pen']=np.empty(0)returnoutputhalf_slice_thickness=self.half_slice_thicknessifself.half_slice_thicknessisnotNoneelsehalf_slice_thicknessrows=[pd.DataFrame(columns=['x','y','colour','size'])]foriinrange(main_slice_idx-half_slice_thickness,main_slice_idx+half_slice_thickness):current_z_data=pd.DataFrame(columns=['x','y','colour','size'])# WARNING: this is x/y of the view, not the 3D imageifi<0:# or i > self.coordinates[:, 2].max()continueelse:current_slice=ipos=self.get_pos(current_slice)ifnotall(pos.shape):# emptycontinuecurrent_z_data[['x','y']]=poscurrent_z_data['colour']=self.get_colours(current_slice).values# Otherwise uses index from sourcecurrent_z_data['size']=self.get_symbol_sizes(main_slice_idx,current_slice,half_size=half_slice_thickness)rows.append(current_z_data)data_df=pd.concat(rows)output={'pos':data_df[['x','y']].values,# WARNING: this is x/y of the view, not the 3D image'size':data_df['size'].values}ifself.has_colours:output['pen']=data_df['colour'].map(dict(self.point_map[['colour','pen']].values)).valuesreturnoutput
[docs]defget_pos(self,current_slice):indices=self.current_slice_indices(current_slice)ifindicesisnotNone:axes=[0,1,2]axes.pop(self.axis)# coordinates in the two other axesreturnself.coordinates[np.ix_(indices,axes)]else:returnnp.empty((0,2))