source: sasview/sansguiframe/src/sans/guiframe/local_perspectives/plotting/masking.py @ 4752c31

ESS_GUIESS_GUI_DocsESS_GUI_batch_fittingESS_GUI_bumps_abstractionESS_GUI_iss1116ESS_GUI_iss879ESS_GUI_iss959ESS_GUI_openclESS_GUI_orderingESS_GUI_sync_sascalccostrafo411magnetic_scattrelease-4.1.1release-4.1.2release-4.2.2release_4.0.1ticket-1009ticket-1094-headlessticket-1242-2d-resolutionticket-1243ticket-1249ticket885unittest-saveload
Last change on this file since 4752c31 was 4752c31, checked in by Jae Cho <jhjcho@…>, 13 years ago

fixing pylint warnings

  • Property mode set to 100644
File size: 28.1 KB
Line 
1"""
2    Mask editor
3"""
4################################################################################
5#This software was developed by the University of Tennessee as part of the
6#Distributed Data Analysis of Neutron Scattering Experiments (DANSE)
7#project funded by the US National Science Foundation.
8#
9#If you use DANSE applications to do scientific research that leads to
10#publication, we ask that you acknowledge the use of the software with the
11#following sentence:
12#
13#This work benefited from DANSE software developed under NSF award DMR-0520547.
14#
15#copyright 2008, University of Tennessee
16################################################################################
17
18
19##Todo: cleaning up, improving the maskplotpanel initialization, and testing.
20import wx
21import sys
22import time
23import matplotlib.cm as cm
24import math
25import copy
26import numpy
27from danse.common.plottools.PlotPanel import PlotPanel
28from danse.common.plottools.plottables import Graph
29from binder import BindArtist
30from sans.guiframe.dataFitting import Data1D, Data2D
31from boxMask import BoxMask
32from sans.guiframe.events import SlicerEvent
33from sans.guiframe.events import StatusEvent
34(InternalEvent, EVT_INTERNAL) = wx.lib.newevent.NewEvent()
35
36DEFAULT_CMAP = cm.jet
37_BOX_WIDTH = 76
38_SCALE = 1e-6
39_STATICBOX_WIDTH = 380
40
41#SLD panel size
42if sys.platform.count("win32") > 0:
43    PANEL_SIZE = 350
44    FONT_VARIANT = 0
45else:
46    PANEL_SIZE = 300
47    FONT_VARIANT = 1
48   
49from data_util.calcthread import CalcThread
50
51class CalcPlot(CalcThread):
52    """
53    Compute Resolution
54    """
55    def __init__(self,
56                 id = -1,
57                 panel = None,
58                 image = None,
59                 completefn = None,
60                 updatefn   = None,
61                 elapsed = 0,
62                 yieldtime  = 0.01,
63                 worktime   = 0.01
64                 ):
65        """
66        """
67        CalcThread.__init__(self, completefn,
68                 updatefn,
69                 yieldtime,
70                 worktime)
71        self.starttime = 0
72        self.id = id 
73        self.panel = panel
74        self.image = image
75       
76       
77    def compute(self):
78        """
79        excuting computation
80        """
81        elapsed = time.time() - self.starttime
82       
83        self.complete(panel=self.panel, image=self.image, elapsed=elapsed)
84       
85       
86class MaskPanel(wx.Dialog):
87    """
88    Provides the Mask Editor GUI.
89    """
90    ## Internal nickname for the window, used by the AUI manager
91    window_name = "Mask Editor"
92    ## Name to appear on the window title bar
93    window_caption = "Mask Editor"
94    ## Flag to tell the AUI manager to put this panel in the center pane
95    CENTER_PANE = True
96    def __init__(self, parent=None, base=None, 
97                 data=None, id=-1, *args, **kwds):
98        kwds["style"] = wx.DEFAULT_DIALOG_STYLE
99        kwds["size"] = wx.Size(_STATICBOX_WIDTH*0.8, PANEL_SIZE) 
100        wx.Dialog.__init__(self, parent, id=id,  *args, **kwds)
101       
102        if data != None:
103            #Font size
104            kwds = []
105            self.SetWindowVariant(variant=FONT_VARIANT)
106            self.SetTitle("Mask Editor for " + data.name)
107            self.parent = base
108            self.data = data
109            self.str = self.data.__str__()
110            ## mask for 2D
111            self.mask = data.mask
112            self.default_mask = copy.deepcopy(data.mask)
113            ## masked data from GUI
114            self.slicer_mask = None
115            self.slicer = None
116            self.slicer_z = 5
117            self.data.interactive = True
118            ## when 2 data have the same id override the 1 st plotted
119            self.name = self.data.name
120            # Panel for 2D plot
121            self.plotpanel = Maskplotpanel(self, -1,
122                                           style=wx.TRANSPARENT_WINDOW)
123            self.cmap = DEFAULT_CMAP
124            ## Create Artist and bind it
125            self.subplot = self.plotpanel.subplot
126            self.connect = BindArtist(self.subplot.figure)
127            self._setup_layout()
128            self.newplot = Data2D(image=self.data.data)
129            self.newplot.setValues(self.data)
130            self.plotpanel.add_image(self.newplot) 
131            self._update_mask(self.mask)
132            self.Centre()
133            self.Layout()
134            # bind evt_close to _draw in fitpage
135            self.Bind(wx.EVT_CLOSE, self.OnClose)
136           
137    def ShowMessage(self, msg=''):
138        """
139        Show error message when mask covers whole data area
140        """
141        mssg = 'Erase, redraw or clear the mask. \n\r'
142        mssg += 'The data range can not be completely masked... \n\r'
143        mssg += msg
144        wx.MessageBox(mssg, 'Error', wx.OK | wx.ICON_ERROR)
145   
146    def _setup_layout(self):
147        """
148        Set up the layout
149        """
150        note = "Note: This masking applies\n     only to %s." % self.data.name
151        note_txt = wx.StaticText(self, -1, note) 
152        note_txt.SetForegroundColour(wx.RED)
153        shape = "Select a Shape for Masking:"
154        #  panel
155        sizer = wx.GridBagSizer(10, 10)
156        #---------inputs----------------
157        shape_txt = wx.StaticText(self, -1, shape) 
158        sizer.Add(shape_txt, (1, 1), flag=wx.TOP|wx.LEFT|wx.BOTTOM, border=5)
159        self.innersector_rb = wx.RadioButton(self, -1, "Double Wings")
160        self.Bind(wx.EVT_RADIOBUTTON, self.onInnerSectorMask,
161                  id=self.innersector_rb.GetId())
162        sizer.Add(self.innersector_rb, (2, 1), 
163                  flag=wx.RIGHT|wx.BOTTOM, border=5)
164        self.innercircle_rb = wx.RadioButton(self, -1, "Circular Disk")
165        self.Bind(wx.EVT_RADIOBUTTON, self.onInnerRingMask,
166                  id=self.innercircle_rb.GetId())
167        sizer.Add(self.innercircle_rb, (3, 1),
168                   flag=wx.RIGHT|wx.BOTTOM, border=5)
169       
170        self.innerbox_rb = wx.RadioButton(self, -1, "Rectangular Disk")
171        self.Bind(wx.EVT_RADIOBUTTON, self.onInnerBoxMask,
172                  id=self.innerbox_rb.GetId())
173        sizer.Add(self.innerbox_rb, (4, 1), flag=wx.RIGHT|wx.BOTTOM, border=5)
174
175        self.outersector_rb = wx.RadioButton(self, -1, "Double Wing Window")
176        self.Bind(wx.EVT_RADIOBUTTON, self.onOuterSectorMask, 
177                  id=self.outersector_rb.GetId())
178        sizer.Add(self.outersector_rb, (5, 1),
179                  flag=wx.RIGHT|wx.BOTTOM, border=5)
180       
181        #outersector_y_txt = wx.StaticText(self, -1, 'Outer Sector')
182        self.outercircle_rb = wx.RadioButton(self, -1, "Circular Window")
183        self.Bind(wx.EVT_RADIOBUTTON, self.onOuterRingMask,
184                  id=self.outercircle_rb.GetId())
185        sizer.Add(self.outercircle_rb, (6, 1), 
186                  flag=wx.RIGHT|wx.BOTTOM, border=5)
187        #outerbox_txt = wx.StaticText(self, -1, 'Outer Box')
188        self.outerbox_rb = wx.RadioButton(self, -1, "Rectangular Window")
189        self.Bind(wx.EVT_RADIOBUTTON, self.onOuterBoxMask, 
190                  id=self.outerbox_rb.GetId())
191        sizer.Add(self.outerbox_rb, (7, 1), flag=wx.RIGHT|wx.BOTTOM, border=5)
192        sizer.Add(note_txt, (8, 1), flag=wx.RIGHT|wx.BOTTOM, border=5)
193        self.innercircle_rb.SetValue(False)
194        self.outercircle_rb.SetValue(False)       
195        self.innerbox_rb.SetValue(False)
196        self.outerbox_rb.SetValue(False)
197        self.innersector_rb.SetValue(False)
198        self.outersector_rb.SetValue(False)
199        sizer.Add(self.plotpanel, (0, 2), (13, 13), 
200                  wx.EXPAND|wx.LEFT|wx.RIGHT, 15)
201
202        #-----Buttons------------1
203        id_button = wx.NewId()
204        button_add = wx.Button(self, id_button, "Add")
205        button_add.SetToolTipString("Add the mask drawn.")
206        button_add.Bind(wx.EVT_BUTTON, self.onAddMask, id=button_add.GetId()) 
207        sizer.Add(button_add, (13, 7))
208        id_button = wx.NewId()
209        button_erase = wx.Button(self, id_button, "Erase")
210        button_erase.SetToolTipString("Erase the mask drawn.")
211        button_erase.Bind(wx.EVT_BUTTON, self.onEraseMask,
212                          id=button_erase.GetId()) 
213        sizer.Add(button_erase, (13, 8))
214        id_button = wx.NewId()
215        button_reset = wx.Button(self, id_button, "Reset")
216        button_reset.SetToolTipString("Reset the mask.")
217        button_reset.Bind(wx.EVT_BUTTON, self.onResetMask,
218                          id=button_reset.GetId()) 
219        sizer.Add(button_reset, (13, 9), flag=wx.RIGHT|wx.BOTTOM, border=15)
220        id_button = wx.NewId()
221        button_reset = wx.Button(self, id_button, "Clear")
222        button_reset.SetToolTipString("Clear all mask.")
223        button_reset.Bind(wx.EVT_BUTTON, self.onClearMask,
224                          id=button_reset.GetId()) 
225        sizer.Add(button_reset, (13, 10), flag=wx.RIGHT|wx.BOTTOM, border=15)
226        sizer.AddGrowableCol(3)
227        sizer.AddGrowableRow(2)
228        self.SetSizerAndFit(sizer)
229        self.Centre()
230        self.Show(True)
231
232    def onInnerBoxMask(self, event=None):
233        """
234        Call Draw Box Slicer and get mask inside of the box
235        """
236        #get ready for next evt
237        event.Skip()       
238        #from boxMask import BoxMask
239        if event != None:
240            self.onClearSlicer(event)         
241        self.slicer_z += 1
242        self.slicer =  BoxMask(self, self.subplot,
243                               zorder=self.slicer_z, side=True)
244        self.subplot.set_ylim(self.data.ymin, self.data.ymax)
245        self.subplot.set_xlim(self.data.xmin, self.data.xmax)
246        self.update()
247        self.slicer_mask = self.slicer.update()
248       
249    def onOuterBoxMask(self, event=None):
250        """
251        Call Draw Box Slicer and get mask outside of the box
252        """
253        event.Skip()       
254        #from boxMask import BoxMask
255        if event != None:
256            self.onClearSlicer(event)     
257        self.slicer_z += 1
258        self.slicer =  BoxMask(self, self.subplot,
259                               zorder=self.slicer_z, side=False)
260        self.subplot.set_ylim(self.data.ymin, self.data.ymax)
261        self.subplot.set_xlim(self.data.xmin, self.data.xmax)
262        self.update()
263        self.slicer_mask = self.slicer.update()
264
265    def onInnerSectorMask(self, event=None):
266        """
267        Call Draw Sector Slicer and get mask inside of the sector
268        """
269        event.Skip()
270        from sectorMask import SectorMask
271        if event != None:
272            self.onClearSlicer(event)
273        self.slicer_z += 1
274        self.slicer =  SectorMask(self, self.subplot,
275                                  zorder=self.slicer_z, side=True)
276        self.subplot.set_ylim(self.data.ymin, self.data.ymax)
277        self.subplot.set_xlim(self.data.xmin, self.data.xmax)   
278        self.update()
279        self.slicer_mask = self.slicer.update() 
280
281    def onOuterSectorMask(self, event=None):
282        """
283        Call Draw Sector Slicer and get mask outside of the sector
284        """
285        event.Skip()
286        from sectorMask import SectorMask
287        if event != None:
288            self.onClearSlicer(event)
289        self.slicer_z += 1
290        self.slicer =  SectorMask(self, self.subplot,
291                                  zorder=self.slicer_z, side=False)
292        self.subplot.set_ylim(self.data.ymin, self.data.ymax)
293        self.subplot.set_xlim(self.data.xmin, self.data.xmax)   
294        self.update()     
295        self.slicer_mask = self.slicer.update()   
296
297    def onInnerRingMask(self, event=None):
298        """
299        Perform inner circular cut on Phi and draw circular slicer
300        """
301        event.Skip()
302        from AnnulusSlicer import CircularMask
303        if event != None:
304            self.onClearSlicer(event)
305        self.slicer_z += 1
306        self.slicer = CircularMask(self, self.subplot,
307                                   zorder=self.slicer_z, side=True)
308        self.subplot.set_ylim(self.data.ymin, self.data.ymax)
309        self.subplot.set_xlim(self.data.xmin, self.data.xmax)   
310        self.update()
311        self.slicer_mask = self.slicer.update() 
312
313    def onOuterRingMask(self, event=None):
314        """
315        Perform outer circular cut on Phi and draw circular slicer
316        """
317        event.Skip()
318        from AnnulusSlicer import CircularMask
319        if event != None:
320            self.onClearSlicer(event)
321        self.slicer_z += 1
322        self.slicer = CircularMask(self, self.subplot,
323                                   zorder=self.slicer_z, side=False)   
324        self.subplot.set_ylim(self.data.ymin, self.data.ymax)
325        self.subplot.set_xlim(self.data.xmin, self.data.xmax)
326        self.update()
327        self.slicer_mask = self.slicer.update()     
328       
329    def onAddMask(self, event):
330        """
331        Add new mask to old mask
332        """
333        if not self.slicer == None:
334            data = Data2D()
335            data = self.data
336            self.slicer_mask = self.slicer.update()
337            data.mask = self.data.mask & self.slicer_mask
338            self._check_display_mask(data.mask, event)
339           
340    def _check_display_mask(self, mask, event):
341        """
342        check if the mask valid and update the plot
343       
344        :param mask: mask data
345        """
346        ## Redraw the current image
347        self._update_mask(mask)
348
349    def onEraseMask(self, event):
350        """
351        Erase new mask from old mask
352        """
353        if not self.slicer == None:
354            self.slicer_mask = self.slicer.update()
355            mask = self.data.mask
356            mask[self.slicer_mask == False] = True
357            self._check_display_mask(mask, event)
358           
359    def onResetMask(self, event):
360        """
361        Reset mask to the original mask
362        """       
363        self.slicer_z += 1
364        self.slicer =  BoxMask(self, self.subplot, 
365                               zorder=self.slicer_z, side=True)
366        self.subplot.set_ylim(self.data.ymin, self.data.ymax)
367        self.subplot.set_xlim(self.data.xmin, self.data.xmax)   
368        mask = copy.deepcopy(self.default_mask)
369        self.data.mask = mask
370        # update mask plot
371        self._check_display_mask(mask, event)
372       
373    def onClearMask(self, event):
374        """
375        Clear mask
376        """           
377        self.slicer_z += 1
378        self.slicer =  BoxMask(self, self.subplot,
379                               zorder=self.slicer_z, side=True)
380        self.subplot.set_ylim(self.data.ymin, self.data.ymax)
381        self.subplot.set_xlim(self.data.xmin, self.data.xmax)   
382        #mask = copy.deepcopy(self.default_mask)
383        mask = numpy.ones(len(self.data.mask), dtype=bool)
384        self.data.mask = mask
385        # update mask plot
386        self._check_display_mask(mask, event)
387       
388    def onClearSlicer(self, event):
389        """
390        Clear the slicer on the plot
391        """
392        if not self.slicer == None:
393            self.slicer.clear()
394            self.subplot.figure.canvas.draw()
395            self.slicer = None
396
397    def update(self, draw=True):
398        """
399        Respond to changes in the model by recalculating the
400        profiles and resetting the widgets.
401        """
402        self.plotpanel.draw()
403       
404    def _set_mask(self, mask):
405        """
406        Set mask
407        """
408        self.data.mask = mask
409       
410    def set_plot_unfocus(self):
411        """
412        Not implemented
413        """
414        pass
415   
416    def _update_mask(self, mask):
417        """
418        Respond to changes in masking
419        """ 
420        # the case of liitle numbers of True points
421        if (len(mask[mask]) < 10 and self.data != None):
422            self.ShowMessage()
423            mask = copy.deepcopy(self.mask)
424            self.data.mask = mask
425        else:
426            self.mask = mask
427        # make temperary data to plot
428        temp_mask = numpy.zeros(len(mask))
429        temp_data = copy.deepcopy(self.data)
430        # temp_data default is None
431        # This method is to distinguish between masked point and data point = 0.
432        temp_mask = temp_mask/temp_mask
433        temp_mask[mask] = temp_data.data[mask]
434        # set temp_data value for self.mask==True, else still None
435        #temp_mask[mask] = temp_data[mask]
436       
437        #TODO: refactor this horrible logic
438        temp_data.data[mask == False] = temp_mask[mask == False]
439        self.plotpanel.clear()
440        if self.slicer != None:
441            self.slicer.clear()
442            self.slicer = None
443        # Post slicer None event
444        event = self._getEmptySlicerEvent()
445        wx.PostEvent(self, event)
446       
447        ##use this method
448        #set zmax and zmin to plot: Fix it w/ data.
449        if self.plotpanel.scale == 'log_{10}':
450            zmax = math.log10(max(self.data.data[self.data.data>0]))
451            zmin = math.log10(min(self.data.data[self.data.data>0]))
452        else:
453            zmax = max(self.data.data[self.data.data>0])
454            zmin = min(self.data.data[self.data.data>0])
455        #plot   
456        plot = self.plotpanel.image(data=temp_mask,
457                       qx_data=self.data.qx_data,
458                       qy_data=self.data.qy_data,
459                       xmin=self.data.xmin,
460                       xmax=self.data.xmax,
461                       ymin=self.data.ymin,
462                       ymax=self.data.ymax,
463                       zmin=zmin,
464                       zmax=zmax,
465                       cmap=self.cmap,
466                       color=0, symbol=0, label=self.data.name)
467        # axis labels
468        self.plotpanel.axes[0].set_xlabel('$\\rm{Q}_{x}(A^{-1})$')
469        self.plotpanel.axes[0].set_ylabel('$\\rm{Q}_{y}(A^{-1})$')
470        self.plotpanel.render()
471        self.plotpanel.subplot.figure.canvas.draw_idle()
472       
473    def _getEmptySlicerEvent(self):
474        """
475        create an empty slicervent
476        """
477        self.innerbox_rb.SetValue(False)
478        self.outerbox_rb.SetValue(False)
479        self.innersector_rb.SetValue(False)
480        self.outersector_rb.SetValue(False)
481        self.innercircle_rb.SetValue(False)
482        self.outercircle_rb.SetValue(False)
483        return SlicerEvent(type=None,
484                           params=None,
485                           obj_class=None) 
486             
487    def _draw_model(self, event):
488        """
489         on_close, update the model2d plot
490        """
491        pass
492       
493    def freeze_axes(self):
494        """
495        freeze axes
496        """
497        self.plotpanel.axes_frozen = True
498       
499    def thaw_axes(self):
500        """
501        thaw axes
502        """
503        self.plotpanel.axes_frozen = False       
504         
505    def onMouseMotion(self, event):
506        """
507        onMotion event
508        """
509        pass
510   
511    def onWheel(self, event):
512        """
513        on wheel event
514        """
515        pass 
516   
517    def OnClose(self, event):
518        """
519            Processing close event
520        """
521        try:
522            self.parent._draw_masked_model(event)
523        except:
524            # when called by data panel
525            event.Skip()
526            pass   
527
528class FloatPanel(wx.Dialog):
529    """
530    Provides the Mask Editor GUI.
531    """
532    ## Internal nickname for the window, used by the AUI manager
533    window_name = "Plot"
534    ## Name to appear on the window title bar
535    window_caption = "Plot"
536    ## Flag to tell the AUI manager to put this panel in the center pane
537    CENTER_PANE = False
538    ID = wx.NewId()
539    def __init__(self, parent=None, base=None, 
540                 data=None, dimension=1, id=ID, *args, **kwds):
541        kwds["style"] = wx.DEFAULT_DIALOG_STYLE
542        kwds["size"] = wx.Size(PANEL_SIZE*1.5, PANEL_SIZE*1.5) 
543        wx.Dialog.__init__(self, parent, id=id,  *args, **kwds)
544       
545        if data != None:
546            #Font size
547            kwds = []
548            self.SetWindowVariant(variant=FONT_VARIANT)
549            self.SetTitle("Plot " + data.name)
550            self.parent = base
551            self.data = data
552            self.str = self.data.__str__()
553            ## when 2 data have the same id override the 1 st plotted
554            self.name = self.data.name
555            self.dimension = dimension
556            # Panel for 2D plot
557            self.plotpanel = Maskplotpanel(self, -1, dimension,
558                                           style=wx.TRANSPARENT_WINDOW)
559            self.plotpanel._SetInitialSize()
560           
561            self.cmap = DEFAULT_CMAP
562            ## Create Artist and bind it
563            self.subplot = self.plotpanel.subplot
564            self._setup_layout()
565            if self.dimension == 1:
566                self.newplot = Data1D(x=data.x, y=data.y, 
567                                      dx=data.dx, dy=data.dy)
568                self.newplot.name = data.name
569            else:   
570                self.newplot = Data2D(image=self.data.data)
571                self.newplot.setValues(self.data)
572                    # Compute and get the image plot
573            self.get_plot()
574            #self.plotpanel.add_image(self.newplot)
575            self.Centre()
576            self.Layout()
577   
578    def get_plot(self):
579        """
580        Get Plot panel
581        """
582        cal_plot = CalcPlot(panel=self.plotpanel, 
583                                   image=self.newplot, 
584                                   completefn=self.complete)
585        cal_plot.queue()
586   
587    def complete(self, panel, image, elapsed=None):
588        """
589        Plot image
590       
591        :param image: newplot [plotpanel]
592        """
593        wx.CallAfter(panel.add_image, image) 
594       
595    def _setup_layout(self):
596        """
597        Set up the layout
598        """
599        #  panel
600        sizer = wx.GridBagSizer(10, 10)
601        if self.dimension == 3:
602            note = "Note: I am very SLOW.     Please be PATIENT...\n"
603            if len(self.data.data) > 3600:
604                note += "Rotation disabled for pixels > 60x60."
605            note_txt = wx.StaticText(self, -1, note) 
606            note_txt.SetForegroundColour(wx.RED)
607            sizer.Add(note_txt, (0, 2), flag=wx.RIGHT|wx.TOP, border=5)
608       
609        sizer.Add(self.plotpanel, (1, 0), (9, 9), 
610                  wx.EXPAND|wx.ALL, 15)
611
612        sizer.AddGrowableCol(3)
613        sizer.AddGrowableRow(2)
614       
615        self.SetSizerAndFit(sizer)
616        self.Centre()
617        self.Show(True)
618       
619    def set_plot_unfocus(self):
620        """
621        Not implemented
622        """
623        pass
624   
625
626    def _draw_model(self, event):
627        """
628         on_close, update the model2d plot
629        """
630        pass
631       
632    def freeze_axes(self):
633        """
634        freeze axes
635        """
636        self.plotpanel.axes_frozen = True
637       
638    def thaw_axes(self):
639        """
640        thaw axes
641        """
642        self.plotpanel.axes_frozen = False       
643   
644    def OnClose(self, event):
645        """
646        """
647        try:
648            self.plotpanel.subplot.figure.clf()
649            self.plotpanel.Close()
650        except:
651            # when called by data panel
652            event.Skip()
653            pass
654
655               
656class Maskplotpanel(PlotPanel):
657    """
658    PlotPanel for Quick plot and masking plot
659    """
660    def __init__(self, parent, id=-1, dimension=2, color=None, 
661                                            dpi=None, **kwargs):
662        """
663        """
664        PlotPanel.__init__(self, parent, id=id, color=color, dpi=dpi, **kwargs)
665       
666        # Keep track of the parent Frame
667        self.parent = parent
668        # Internal list of plottable names (because graph
669        # doesn't have a dictionary of handles for the plottables)
670        self.dimension = dimension
671        self.plots = {}
672        self.graph = Graph()
673        #add axis labels
674        self.graph.xaxis('\\rm{x} ', '')
675        self.graph.yaxis('\\rm{y} ', '')
676       
677    def add_toolbar(self):
678        """
679        Add toolbar
680        """
681        # Not implemented
682        pass
683   
684    def on_set_focus(self, event):
685        """
686        send to the parenet the current panel on focus
687        """
688        if self.dimension == 3:
689            pass
690        else:
691            self.draw()   
692         
693    def add_image(self, plot):
694        """
695        Add Image
696        """
697        self.plots[plot.name] = plot
698        #init graph
699        self.graph = Graph()
700        #add plot
701        self.graph.add(plot)
702        #add axes
703        if self.dimension == 1:
704            self.xaxis_label = '\\rm{x} '
705            self.xaxis_unit = ''
706            self.yaxis_label = '\\rm{y} '
707            self.yaxis_unit = ''
708        #draw
709        # message
710        status_type = 'progress' 
711        msg = 'Plotting...'
712        self._status_info(msg, status_type)
713        status_type = 'stop'           
714        self.graph.render(self)
715        self.subplot.figure.canvas.resizing = False
716        if self.dimension < 3:
717            self.graph.render(self)
718            self.subplot.figure.canvas.draw()
719        elif FONT_VARIANT:
720            self.subplot.figure.canvas.draw()
721        msg = 'Plotting Completed.'
722        self._status_info(msg, status_type)
723       
724    def onMouseMotion(self, event):
725        """
726        Disable dragging 2D image
727        """
728        pass
729   
730    def onWheel(self, event):
731        """
732        """
733        pass 
734     
735    def onLeftDown(self, event):
736        """
737        Disables LeftDown
738        """
739        pass
740   
741    def onPick(self, event):
742        """
743        Disables OnPick
744        """
745        pass
746   
747    def draw(self):
748        """
749        Draw
750        """
751        # message
752        status_type = 'progress' 
753        msg = 'Plotting...'
754        self._status_info(msg, status_type)
755        status_type = 'stop'           
756       
757        if self.dimension == 3:
758            pass
759        else:
760            self.subplot.figure.canvas.draw_idle() 
761       
762        msg = 'Plotting Completed.'
763        self._status_info(msg, status_type)
764       
765    def onContextMenu(self, event):
766        """
767        Default context menu for a plot panel
768        """
769        # Selective Slicer plot popup menu
770        slicerpop = wx.Menu()
771       
772        id = wx.NewId()
773        slicerpop.Append(id, '&Print Image', 'Print image')
774        wx.EVT_MENU(self, id, self.onPrint)
775
776        id = wx.NewId()
777        slicerpop.Append(id, '&Copy to Clipboard', 'Copy to the clipboard')
778        wx.EVT_MENU(self, id, self.OnCopyFigureMenu)
779       
780        if self.dimension == 1:
781            id = wx.NewId()
782            slicerpop.Append(id, '&Change Scale')
783            wx.EVT_MENU(self, id, self._onProperties)
784        else:
785            slicerpop.AppendSeparator()
786            id_cm = wx.NewId()
787            slicerpop.Append(id_cm, '&Toggle Linear/Log scale')
788            wx.EVT_MENU(self, id_cm, self._onToggleScale)
789               
790        pos = event.GetPosition()
791        pos = self.ScreenToClient(pos)
792        self.PopupMenu(slicerpop, pos)
793       
794    def _status_info(self, msg = '', type = "update"):
795        """
796        Status msg
797        """
798        if type == "stop":
799            label = "Plotting..."
800            able = True
801        else:   
802            label = "Wait..."
803            able = False
804        if self.parent.parent.parent != None:
805                wx.PostEvent(self.parent.parent.parent, 
806                             StatusEvent(status = msg, type = type ))
807
808class ViewerFrame(wx.Frame):
809    """
810    Add comment
811    """
812    def __init__(self, parent, id, title):
813        """
814        comment
815        :param parent: parent panel/container
816        """
817        # Initialize the Frame object
818        wx.Frame.__init__(self, parent, id, title,
819                          wx.DefaultPosition, wx.Size(950, 850))
820        # Panel for 1D plot
821        self.plotpanel = Maskplotpanel(self, -1, style=wx.RAISED_BORDER)
822
823class ViewApp(wx.App):
824    def OnInit(self):
825        frame = ViewerFrame(None, -1, 'testView')   
826        frame.Show(True)
827        #self.SetTopWindow(frame)
828       
829        return True
830               
831if __name__ == "__main__": 
832    app = ViewApp(0)
833    app.MainLoop()     
Note: See TracBrowser for help on using the repository browser.