123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530 |
- """!
- @package wxplot.base
- @brief Base classes for iinteractive plotting using PyPlot
- Classes:
- - BasePlotFrame
- (C) 2011 by the GRASS Development Team
- This program is free software under the GNU General Public License
- (>=v2). Read the file COPYING that comes with GRASS for details.
- @author Michael Barton, Arizona State University
- """
- import os
- import sys
- import wx
- import wx.lib.plot as plot
- from core.globalvar import ETCICONDIR
- from core.settings import UserSettings
- from wxplot.dialogs import TextDialog, OptDialog
- import grass.script as grass
- class BasePlotFrame(wx.Frame):
- """!Abstract PyPlot display frame class"""
- def __init__(self, parent = None, id = wx.ID_ANY, size = (700, 300),
- style = wx.DEFAULT_FRAME_STYLE, rasterList = [], **kwargs):
- wx.Frame.__init__(self, parent, id, size = size, style = style, **kwargs)
-
- self.parent = parent # MapFrame
- self.mapwin = self.parent.MapWindow
- self.Map = Map() # instance of render.Map to be associated with display
- self.rasterList = rasterList #list of rasters to plot
- self.raster = {} # dictionary of raster maps and their plotting parameters
- self.plottype = ''
-
- self.linestyledict = { 'solid' : wx.SOLID,
- 'dot' : wx.DOT,
- 'long-dash' : wx.LONG_DASH,
- 'short-dash' : wx.SHORT_DASH,
- 'dot-dash' : wx.DOT_DASH }
- self.ptfilldict = { 'transparent' : wx.TRANSPARENT,
- 'solid' : wx.SOLID }
- #
- # Icon
- #
- self.SetIcon(wx.Icon(os.path.join(ETCICONDIR, 'grass.ico'), wx.BITMAP_TYPE_ICO))
-
- #
- # Add statusbar
- #
- self.statusbar = self.CreateStatusBar(number = 2, style = 0)
- self.statusbar.SetStatusWidths([-2, -1])
- #
- # Define canvas and settings
- #
- #
- self.client = plot.PlotCanvas(self)
- #define the function for drawing pointLabels
- self.client.SetPointLabelFunc(self.DrawPointLabel)
- # Create mouse event for showing cursor coords in status bar
- self.client.canvas.Bind(wx.EVT_LEFT_DOWN, self.OnMouseLeftDown)
- # Show closest point when enabled
- self.client.canvas.Bind(wx.EVT_MOTION, self.OnMotion)
- self.plotlist = [] # list of things to plot
- self.plot = None # plot draw object
- self.ptitle = "" # title of window
- self.xlabel = "" # default X-axis label
- self.ylabel = "" # default Y-axis label
- #
- # Bind various events
- #
- self.Bind(wx.EVT_CLOSE, self.OnCloseWindow)
-
- self.CentreOnScreen()
-
- self._createColorDict()
- def _createColorDict(self):
- """!Create color dictionary to return wx.Color tuples
- for assigning colors to images in imagery groups"""
-
- self.colorDict = {}
- for clr in grass.named_colors.iterkeys():
- if clr == 'white' or clr == 'black': continue
- r = grass.named_colors[clr][0] * 255
- g = grass.named_colors[clr][1] * 255
- b = grass.named_colors[clr][2] * 255
- self.colorDict[clr] = (r,g,b,255)
- def InitPlotOpts(self, plottype):
- """!Initialize options for entire plot
- """
-
- self.plottype = plottype # histogram, profile, or scatter
- self.properties = {} # plot properties
- self.properties['font'] = {}
- self.properties['font']['prop'] = UserSettings.Get(group = self.plottype, key = 'font')
- self.properties['font']['wxfont'] = wx.Font(11, wx.FONTFAMILY_SWISS,
- wx.FONTSTYLE_NORMAL,
- wx.FONTWEIGHT_NORMAL)
-
- if self.plottype == 'profile':
- self.properties['marker'] = UserSettings.Get(group = self.plottype, key = 'marker')
- # changing color string to tuple for markers/points
- colstr = str(self.properties['marker']['color'])
- self.properties['marker']['color'] = tuple(int(colval) for colval in colstr.strip('()').split(','))
- self.properties['grid'] = UserSettings.Get(group = self.plottype, key = 'grid')
- colstr = str(self.properties['grid']['color']) # changing color string to tuple
- self.properties['grid']['color'] = tuple(int(colval) for colval in colstr.strip('()').split(','))
-
- self.properties['x-axis'] = {}
- self.properties['x-axis']['prop'] = UserSettings.Get(group = self.plottype, key = 'x-axis')
- self.properties['x-axis']['axis'] = None
- self.properties['y-axis'] = {}
- self.properties['y-axis']['prop'] = UserSettings.Get(group = self.plottype, key = 'y-axis')
- self.properties['y-axis']['axis'] = None
-
- self.properties['legend'] = UserSettings.Get(group = self.plottype, key = 'legend')
- self.zoom = False # zooming disabled
- self.drag = False # draging disabled
- self.client.SetShowScrollbars(True) # vertical and horizontal scrollbars
- # x and y axis set to normal (non-log)
- self.client.setLogScale((False, False))
- if self.properties['x-axis']['prop']['type']:
- self.client.SetXSpec(self.properties['x-axis']['prop']['type'])
- else:
- self.client.SetXSpec('auto')
-
- if self.properties['y-axis']['prop']['type']:
- self.client.SetYSpec(self.properties['y-axis']['prop']['type'])
- else:
- self.client.SetYSpec('auto')
-
- def InitRasterOpts(self, rasterList, plottype):
- """!Initialize or update raster dictionary for plotting
- """
- rdict = {} # initialize a dictionary
- for r in rasterList:
- idx = rasterList.index(r)
-
- try:
- ret = grass.raster_info(r)
- except:
- continue
- # if r.info cannot parse map, skip it
-
- self.raster[r] = UserSettings.Get(group = plottype, key = 'raster') # some default settings
- rdict[r] = {} # initialize sub-dictionaries for each raster in the list
- if ret['units'] == '(none)' or ret['units'] == '' or ret['units'] == None:
- rdict[r]['units'] = ''
- else:
- self.raster[r]['units'] = ret['units']
- rdict[r]['plegend'] = r.split('@')[0]
- rdict[r]['datalist'] = [] # list of cell value,frequency pairs for plotting histogram
- rdict[r]['pline'] = None
- rdict[r]['datatype'] = ret['datatype']
- rdict[r]['pwidth'] = 1
- rdict[r]['pstyle'] = 'solid'
-
- if idx <= len(self.colorList):
- rdict[r]['pcolor'] = self.colorDict[self.colorList[idx]]
- else:
- r = randint(0, 255)
- b = randint(0, 255)
- g = randint(0, 255)
- rdict[r]['pcolor'] = ((r,g,b,255))
-
-
- return rdict
-
- def InitRasterPairs(self, rasterList, plottype):
- """!Initialize or update raster dictionary with raster pairs for
- bivariate scatterplots
- """
-
- if len(rasterList) == 0: return
- rdict = {} # initialize a dictionary
- for rpair in rasterList:
- idx = rasterList.index(rpair)
-
- try:
- ret0 = grass.raster_info(rpair[0])
- ret1 = grass.raster_info(rpair[1])
- except:
- continue
- # if r.info cannot parse map, skip it
- self.raster[rpair] = UserSettings.Get(group = plottype, key = 'rasters') # some default settings
- rdict[rpair] = {} # initialize sub-dictionaries for each raster in the list
- rdict[rpair][0] = {}
- rdict[rpair][1] = {}
- if ret0['units'] == '(none)' or ret['units'] == '' or ret['units'] == None:
- rdict[rpair][0]['units'] = ''
- else:
- self.raster[rpair][0]['units'] = ret0['units']
- if ret1['units'] == '(none)' or ret['units'] == '' or ret['units'] == None:
- rdict[rpair][1]['units'] = ''
- else:
- self.raster[rpair][1]['units'] = ret1['units']
- rdict[rpair]['plegend'] = rpair[0].split('@')[0] + ' vs ' + rpair[1].split('@')[0]
- rdict[rpair]['datalist'] = [] # list of cell value,frequency pairs for plotting histogram
- rdict[rpair]['ptype'] = 'dot'
- rdict[rpair][0]['datatype'] = ret0['datatype']
- rdict[rpair][1]['datatype'] = ret1['datatype']
- rdict[rpair]['psize'] = 1
- rdict[rpair]['pfill'] = 'solid'
-
- if idx <= len(self.colorList):
- rdict[rpair]['pcolor'] = self.colorDict[self.colorList[idx]]
- else:
- r = randint(0, 255)
- b = randint(0, 255)
- g = randint(0, 255)
- rdict[rpair]['pcolor'] = ((r,g,b,255))
-
- return rdict
- def SetGraphStyle(self):
- """!Set plot and text options
- """
- self.client.SetFont(self.properties['font']['wxfont'])
- self.client.SetFontSizeTitle(self.properties['font']['prop']['titleSize'])
- self.client.SetFontSizeAxis(self.properties['font']['prop']['axisSize'])
- self.client.SetEnableZoom(self.zoom)
- self.client.SetEnableDrag(self.drag)
-
- #
- # axis settings
- #
- if self.properties['x-axis']['prop']['type'] == 'custom':
- self.client.SetXSpec('min')
- else:
- self.client.SetXSpec(self.properties['x-axis']['prop']['type'])
- if self.properties['y-axis']['prop']['type'] == 'custom':
- self.client.SetYSpec('min')
- else:
- self.client.SetYSpec(self.properties['y-axis']['prop'])
- if self.properties['x-axis']['prop']['type'] == 'custom' and \
- self.properties['x-axis']['prop']['min'] < self.properties['x-axis']['prop']['max']:
- self.properties['x-axis']['axis'] = (self.properties['x-axis']['prop']['min'],
- self.properties['x-axis']['prop']['max'])
- else:
- self.properties['x-axis']['axis'] = None
- if self.properties['y-axis']['prop']['type'] == 'custom' and \
- self.properties['y-axis']['prop']['min'] < self.properties['y-axis']['prop']['max']:
- self.properties['y-axis']['axis'] = (self.properties['y-axis']['prop']['min'],
- self.properties['y-axis']['prop']['max'])
- else:
- self.properties['y-axis']['axis'] = None
- self.client.SetEnableGrid(self.properties['grid']['enabled'])
-
- self.client.SetGridColour(wx.Color(self.properties['grid']['color'][0],
- self.properties['grid']['color'][1],
- self.properties['grid']['color'][2],
- 255))
- self.client.SetFontSizeLegend(self.properties['font']['prop']['legendSize'])
- self.client.SetEnableLegend(self.properties['legend']['enabled'])
- if self.properties['x-axis']['prop']['log'] == True:
- self.properties['x-axis']['axis'] = None
- self.client.SetXSpec('min')
- if self.properties['y-axis']['prop']['log'] == True:
- self.properties['y-axis']['axis'] = None
- self.client.SetYSpec('min')
-
- self.client.setLogScale((self.properties['x-axis']['prop']['log'],
- self.properties['y-axis']['prop']['log']))
- def DrawPlot(self, plotlist):
- """!Draw line and point plot from list plot elements.
- """
- self.plot = plot.PlotGraphics(plotlist,
- self.ptitle,
- self.xlabel,
- self.ylabel)
- if self.properties['x-axis']['prop']['type'] == 'custom':
- self.client.SetXSpec('min')
- else:
- self.client.SetXSpec(self.properties['x-axis']['prop']['type'])
- if self.properties['y-axis']['prop']['type'] == 'custom':
- self.client.SetYSpec('min')
- else:
- self.client.SetYSpec(self.properties['y-axis']['prop']['type'])
- self.client.Draw(self.plot, self.properties['x-axis']['axis'],
- self.properties['y-axis']['axis'])
-
- def DrawPointLabel(self, dc, mDataDict):
- """!This is the fuction that defines how the pointLabels are
- plotted dc - DC that will be passed mDataDict - Dictionary
- of data that you want to use for the pointLabel
- As an example I have decided I want a box at the curve
- point with some text information about the curve plotted
- below. Any wxDC method can be used.
- """
- dc.SetPen(wx.Pen(wx.BLACK))
- dc.SetBrush(wx.Brush( wx.BLACK, wx.SOLID ) )
- sx, sy = mDataDict["scaledXY"] #scaled x,y of closest point
- dc.DrawRectangle( sx-5,sy-5, 10, 10) #10by10 square centered on point
- px,py = mDataDict["pointXY"]
- cNum = mDataDict["curveNum"]
- pntIn = mDataDict["pIndex"]
- legend = mDataDict["legend"]
- #make a string to display
- s = "Crv# %i, '%s', Pt. (%.2f,%.2f), PtInd %i" %(cNum, legend, px, py, pntIn)
- dc.DrawText(s, sx , sy+1)
- def OnZoom(self, event):
- """!Enable zooming and disable dragging
- """
- self.zoom = True
- self.drag = False
- self.client.SetEnableZoom(self.zoom)
- self.client.SetEnableDrag(self.drag)
- def OnDrag(self, event):
- """!Enable dragging and disable zooming
- """
- self.zoom = False
- self.drag = True
- self.client.SetEnableDrag(self.drag)
- self.client.SetEnableZoom(self.zoom)
- def OnRedraw(self, event):
- """!Redraw the plot window. Unzoom to original size
- """
- self.client.Reset()
- self.client.Redraw()
-
- def OnErase(self, event):
- """!Erase the plot window
- """
- self.client.Clear()
- self.mapwin.ClearLines(self.mapwin.pdc)
- self.mapwin.ClearLines(self.mapwin.pdcTmp)
- self.mapwin.polycoords = []
- self.mapwin.Refresh()
- def SaveToFile(self, event):
- """!Save plot to graphics file
- """
- self.client.SaveFile()
- def OnMouseLeftDown(self,event):
- self.SetStatusText(_("Left Mouse Down at Point: (%.4f, %.4f)") % \
- self.client._getXY(event))
- event.Skip() # allows plotCanvas OnMouseLeftDown to be called
- def OnMotion(self, event):
- """!Indicate when mouse is outside the plot area
- """
- if self.client.OnLeave(event): print 'out of area'
- #show closest point (when enbled)
- if self.client.GetEnablePointLabel() == True:
- #make up dict with info for the pointLabel
- #I've decided to mark the closest point on the closest curve
- dlst = self.client.GetClosetPoint( self.client._getXY(event), pointScaled = True)
- if dlst != []: #returns [] if none
- curveNum, legend, pIndex, pointXY, scaledXY, distance = dlst
- #make up dictionary to pass to my user function (see DrawPointLabel)
- mDataDict = {"curveNum":curveNum, "legend":legend, "pIndex":pIndex,\
- "pointXY":pointXY, "scaledXY":scaledXY}
- #pass dict to update the pointLabel
- self.client.UpdatePointLabel(mDataDict)
- event.Skip() #go to next handler
-
-
- def PlotOptionsMenu(self, event):
- """!Popup menu for plot and text options
- """
- point = wx.GetMousePosition()
- popt = wx.Menu()
- # Add items to the menu
- settext = wx.MenuItem(popt, wx.ID_ANY, _('Text settings'))
- popt.AppendItem(settext)
- self.Bind(wx.EVT_MENU, self.PlotText, settext)
- setgrid = wx.MenuItem(popt, wx.ID_ANY, _('Plot settings'))
- popt.AppendItem(setgrid)
- self.Bind(wx.EVT_MENU, self.PlotOptions, setgrid)
- # Popup the menu. If an item is selected then its handler
- # will be called before PopupMenu returns.
- self.PopupMenu(popt)
- popt.Destroy()
- def NotFunctional(self):
- """!Creates a 'not functional' message dialog
- """
- dlg = wx.MessageDialog(parent = self,
- message = _('This feature is not yet functional'),
- caption = _('Under Construction'),
- style = wx.OK | wx.ICON_INFORMATION)
- dlg.ShowModal()
- dlg.Destroy()
- def OnPlotText(self, dlg):
- """!Custom text settings for histogram plot.
- """
- self.ptitle = dlg.ptitle
- self.xlabel = dlg.xlabel
- self.ylabel = dlg.ylabel
- dlg.UpdateSettings()
- self.client.SetFont(self.properties['font']['wxfont'])
- self.client.SetFontSizeTitle(self.properties['font']['prop']['titleSize'])
- self.client.SetFontSizeAxis(self.properties['font']['prop']['axisSize'])
- if self.plot:
- self.plot.setTitle(dlg.ptitle)
- self.plot.setXLabel(dlg.xlabel)
- self.plot.setYLabel(dlg.ylabel)
-
- self.OnRedraw(event = None)
-
- def PlotText(self, event):
- """!Set custom text values for profile title and axis labels.
- """
- dlg = TextDialog(parent = self, id = wx.ID_ANY,
- plottype = self.plottype,
- title = _('Histogram text settings'))
- if dlg.ShowModal() == wx.ID_OK:
- self.OnPlotText(dlg)
- dlg.Destroy()
- def PlotOptions(self, event):
- """!Set various profile options, including: line width, color,
- style; marker size, color, fill, and style; grid and legend
- options. Calls OptDialog class.
- """
- dlg = OptDialog(parent = self, id = wx.ID_ANY,
- plottype = self.plottype,
- title = _('Plot settings'))
- btnval = dlg.ShowModal()
- if btnval == wx.ID_SAVE:
- dlg.UpdateSettings()
- self.SetGraphStyle()
- dlg.Destroy()
- elif btnval == wx.ID_CANCEL:
- dlg.Destroy()
- def PrintMenu(self, event):
- """!Print options and output menu
- """
- point = wx.GetMousePosition()
- printmenu = wx.Menu()
- for title, handler in ((_("Page setup"), self.OnPageSetup),
- (_("Print preview"), self.OnPrintPreview),
- (_("Print display"), self.OnDoPrint)):
- item = wx.MenuItem(printmenu, wx.ID_ANY, title)
- printmenu.AppendItem(item)
- self.Bind(wx.EVT_MENU, handler, item)
-
- # Popup the menu. If an item is selected then its handler
- # will be called before PopupMenu returns.
- self.PopupMenu(printmenu)
- printmenu.Destroy()
- def OnPageSetup(self, event):
- self.client.PageSetup()
- def OnPrintPreview(self, event):
- self.client.PrintPreview()
- def OnDoPrint(self, event):
- self.client.Printout()
- def OnQuit(self, event):
- self.Close(True)
- def OnCloseWindow(self, event):
- """!Close plot window and clean up
- """
- try:
- self.mapwin.ClearLines()
- self.mapwin.mouse['begin'] = self.mapwin.mouse['end'] = (0.0, 0.0)
- self.mapwin.mouse['use'] = 'pointer'
- self.mapwin.mouse['box'] = 'point'
- self.mapwin.polycoords = []
- self.mapwin.UpdateMap(render = False, renderVector = False)
- except:
- pass
-
- self.mapwin.SetCursor(self.Parent.cursors["default"])
- self.Destroy()
-
|