1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060 |
- """!
- @package mapdisp_statusbar.py
- @brief Classes for statusbar management
- Classes:
- - SbException
- - SbManager
- - SbItem
- - SbRender
- - SbShowRegion
- - SbAlignExtent
- - SbResolution
- - SbMapScale
- - SbGoTo
- - SbProjection
- - SbMask
- - SbTextItem
- - SbDisplayGeometry
- - SbCoordinates
- - SbRegionExtent
- - SbCompRegionExtent
- - SbProgress
- (C) 2006-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 Vaclav Petras <wenzeslaus gmail.com>
- @author Anna Kratochvilova <kratochanna gmail.com>
- """
- import wx
- import utils
- import gcmd
- from grass.script import core as grass
- from preferences import globalSettings as UserSettings
- class SbException:
- """! Exception class used in SbManager and SbItems"""
- def __init__(self, message):
- self.message = message
- def __str__(self):
- return self.message
- class SbManager:
- """!Statusbar manager for wx.Statusbar and SbItems.
-
- Statusbar manager manages items added by AddStatusbarItem method.
- Provides progress bar (SbProgress) and choice (wx.Choice).
- Items with position 0 are shown according to choice selection.
- Only one item of the same class is supposed to be in statusbar.
- Manager user have to create statusbar on his own, add items to manager
- and call Update method to show particular widgets.
- User settings (group = 'display', key = 'statusbarMode', subkey = 'selection')
- are taken into account.
-
- @todo generalize access to UserSettings (specify group, etc.)
- @todo add GetMode method using name instead of index
- """
- def __init__(self, mapframe, statusbar):
- """!Connects manager to statusbar
-
- Creates choice and progress bar.
- """
- self.mapFrame = mapframe
- self.statusbar = statusbar
-
- self.choice = wx.Choice(self.statusbar, wx.ID_ANY)
-
- self.choice.Bind(wx.EVT_CHOICE, self.OnToggleStatus)
-
- self.statusbarItems = dict()
-
- self._postInitialized = False
-
- self.progressbar = SbProgress(self.mapFrame, self.statusbar)
-
- self._hiddenItems = {}
-
- def SetProperty(self, name, value):
- """!Sets property represented by one of contained SbItems
-
- @param name name of SbItem (from name attribute)
- @param value value to be set
- """
- self.statusbarItems[name].SetValue(value)
-
- def GetProperty(self, name):
- """!Returns property represented by one of contained SbItems
-
- @param name name of SbItem (from name attribute)
- """
- return self.statusbarItems[name].GetValue()
-
- def HasProperty(self, name):
- """!Checks whether property is represented by one of contained SbItems
-
- @param name name of SbItem (from name attribute)
-
- @returns True if particular SbItem is contained, False otherwise
- """
- if name in self.statusbarItems:
- return True
- return False
-
- def AddStatusbarItem(self, item):
- """!Adds item to statusbar
-
- If item position is 0, item is managed by choice.
-
- @see AddStatusbarItemsByClass
- """
- self.statusbarItems[item.name] = item
- if item.GetPosition() == 0:
- self.choice.Append(item.label, clientData = item) #attrError?
-
- def AddStatusbarItemsByClass(self, itemClasses, **kwargs):
- """!Adds items to statusbar
- @param itemClasses list of classes of items to be add
- @param kwargs SbItem constructor parameters
-
- @see AddStatusbarItem
- """
- for Item in itemClasses:
- item = Item(**kwargs)
- self.AddStatusbarItem(item)
-
- def HideStatusbarChoiceItemsByClass(self, itemClasses):
- """!Hides items showed in choice
-
- Hides items with position 0 (items showed in choice) by removing
- them from choice.
-
- @param itemClasses list of classes of items to be hided
-
- @see ShowStatusbarChoiceItemsByClass
- @todo consider adding similar function which would take item names
- """
- index = []
- for itemClass in itemClasses:
- for i in range(0, self.choice.GetCount() - 1):
- item = self.choice.GetClientData(i)
- if item.__class__ == itemClass:
- index.append(i)
- self._hiddenItems[i] = item
- # must be sorted in reverse order to be removed correctly
- for i in sorted(index, reverse = True):
- self.choice.Delete(i)
-
- def ShowStatusbarChoiceItemsByClass(self, itemClasses):
- """!Shows items showed in choice
-
- Shows items with position 0 (items showed in choice) by adding
- them to choice.
- Items are restored in their old positions.
-
- @param itemClasses list of classes of items to be showed
-
- @see HideStatusbarChoiceItemsByClass
- """
- # must be sorted to be inserted correctly
- for pos in sorted(self._hiddenItems.keys()):
- item = self._hiddenItems[pos]
- if item.__class__ in itemClasses:
- self.choice.Insert(item.label, pos, item)
-
- def ShowItem(self, itemName):
- """!Invokes showing of particular item
-
- @see Update
- """
- self.statusbarItems[itemName].Show()
-
- def _postInit(self):
- """!Post-initialization method
-
- It sets internal user settings,
- set choice's selection (from user settings) and does reposition.
- It needs choice filled by items.
- it is called automatically.
- """
- UserSettings.Set(group = 'display',
- key = 'statusbarMode',
- subkey = 'choices',
- value = self.choice.GetItems(),
- internal = True)
-
- self.choice.SetSelection(UserSettings.Get(group = 'display',
- key = 'statusbarMode',
- subkey = 'selection'))
- self.Reposition()
-
- self._postInitialized = True
-
- def Update(self):
- """!Updates statusbar
- It always updates mask.
- """
- if not self._postInitialized:
- self._postInit()
-
- for item in self.statusbarItems.values():
- if item.GetPosition() == 0:
- item.Hide()
- else:
- item.Update() # mask, render
-
- if self.choice.GetCount() > 0:
- item = self.choice.GetClientData(self.choice.GetSelection())
- item.Update()
-
- def Reposition(self):
- """!Reposition items in statusbar
-
- Set positions to all items managed by statusbar manager.
- It should not be necessary to call it manually.
- """
-
- widgets = []
- for item in self.statusbarItems.values():
- widgets.append((item.GetPosition(), item.GetWidget()))
-
- widgets.append((1, self.choice))
- widgets.append((0, self.progressbar.GetWidget()))
-
- for idx, win in widgets:
- if not win:
- continue
- rect = self.statusbar.GetFieldRect(idx)
- if idx == 0: # show region / mapscale / process bar
- # -> size
- wWin, hWin = win.GetBestSize()
- if win == self.progressbar.GetWidget():
- wWin = rect.width - 6
- # -> position
- # if win == self.statusbarWin['region']:
- # x, y = rect.x + rect.width - wWin, rect.y - 1
- # align left
- # else:
- x, y = rect.x + 3, rect.y - 1
- w, h = wWin, rect.height + 2
- else: # choice || auto-rendering
- x, y = rect.x, rect.y - 1
- w, h = rect.width, rect.height + 2
- if idx == 2: # mask
- x += 5
- y += 4
- elif idx == 3: # render
- x += 5
- win.SetPosition((x, y))
- win.SetSize((w, h))
-
- def GetProgressBar(self):
- """!Returns progress bar"""
- return self.progressbar
-
- def OnToggleStatus(self, event):
- """!Toggle status text
- """
- self.Update()
-
- def SetMode(self, modeIndex):
- """!Sets current mode
-
- Mode is usually driven by user through choice.
- """
- self.choice.SetSelection(modeIndex)
-
- def GetMode(self):
- """!Returns current mode"""
- return self.choice.GetSelection()
- class SbItem:
- """!Base class for statusbar items.
-
- Each item represents functionality (or action) controlled by statusbar
- and related to MapFrame.
- One item is usually connected with one widget but it is not necessary.
- Item can represent property (depends on manager).
- Items are not widgets but can provide interface to them.
- Items usually has requirements to MapFrame instance
- (specified as MapFrame.methodname or MapWindow.methodname).
-
- @todo consider externalizing position (see SbProgress use in SbManager)
- """
- def __init__(self, mapframe, statusbar, position = 0):
- """!
-
- @param mapframe instance of class with MapFrame interface
- @param statusbar statusbar instance (wx.Statusbar)
- @param position item position in statusbar
-
- @todo rewrite Update also in derived classes to take in account item position
- """
- self.mapFrame = mapframe
- self.statusbar = statusbar
- self.position = position
-
- def Show(self):
- """!Invokes showing of underlying widget.
-
- In derived classes it can do what is appropriate for it,
- e.g. showing text on statusbar (only).
- """
- self.widget.Show()
-
- def Hide(self):
- self.widget.Hide()
-
- def SetValue(self, value):
- self.widget.SetValue(value)
-
- def GetValue(self):
- return self.widget.GetValue()
-
- def GetPosition(self):
- return self.position
-
- def GetWidget(self):
- """!Returns underlaying winget.
-
- @return widget or None if doesn't exist
- """
- return self.widget
-
- def _update(self, longHelp):
- """!Default implementation for Update method.
-
- @param longHelp True to enable long help (help from toolbars)
- """
- self.statusbar.SetStatusText("", 0)
- self.Show()
- self.mapFrame.StatusbarEnableLongHelp(longHelp)
-
- def Update(self):
- """!Called when statusbar action is activated (e.g. through wx.Choice).
- """
- self._update(longHelp = False)
- class SbRender(SbItem):
- """!Checkbox to enable and disable auto-rendering.
-
- Requires MapFrame.OnRender method.
- """
- def __init__(self, mapframe, statusbar, position = 0):
- SbItem.__init__(self, mapframe, statusbar, position)
- self.name = 'render'
-
- self.widget = wx.CheckBox(parent = self.statusbar, id = wx.ID_ANY,
- label = _("Render"))
-
- self.widget.SetValue(UserSettings.Get(group = 'display',
- key = 'autoRendering',
- subkey = 'enabled'))
- self.widget.Hide()
- self.widget.SetToolTip(wx.ToolTip (_("Enable/disable auto-rendering")))
-
- self.widget.Bind(wx.EVT_CHECKBOX, self.OnToggleRender)
-
- def OnToggleRender(self, event):
- # (other items should call self.mapFrame.IsAutoRendered())
- if self.GetValue():
- self.mapFrame.OnRender(None)
- def Update(self):
- self.Show()
-
- class SbShowRegion(SbItem):
- """!Checkbox to enable and disable showing of computational region.
-
- Requires MapFrame.OnRender, MapFrame.IsAutoRendered, MapFrame.GetWindow.
- Expects that instance returned by MapFrame.GetWindow will handle
- regionCoords attribute.
- """
- def __init__(self, mapframe, statusbar, position = 0):
- SbItem.__init__(self, mapframe, statusbar, position)
- self.name = 'region'
- self.label = _("Show comp. extent")
-
- self.widget = wx.CheckBox(parent = self.statusbar, id = wx.ID_ANY,
- label = _("Show computational extent"))
-
- self.widget.SetValue(False)
- self.widget.Hide()
- self.widget.SetToolTip(wx.ToolTip (_("Show/hide computational "
- "region extent (set with g.region). "
- "Display region drawn as a blue box inside the "
- "computational region, "
- "computational region inside a display region "
- "as a red box).")))
-
- self.widget.Bind(wx.EVT_CHECKBOX, self.OnToggleShowRegion)
-
- def OnToggleShowRegion(self, event):
- """!Shows/Hides extent (comp. region) in map canvas.
-
- Shows or hides according to checkbox value.
- """
- if self.widget.GetValue():
- # show extent
- self.mapFrame.GetWindow().regionCoords = []
- elif hasattr(self.mapFrame.GetWindow(), 'regionCoords'):
- del self.mapFrame.GetWindow().regionCoords
- # redraw map if auto-rendering is enabled
- if self.mapFrame.IsAutoRendered():
- self.mapFrame.OnRender(None)
- def SetValue(self, value):
- SbItem.SetValue(self, value)
- if value:
- self.mapFrame.GetWindow().regionCoords = []
- elif hasattr(self.mapFrame.GetWindow(), 'regionCoords'):
- del self.mapFrame.GetWindow().regionCoords
-
- class SbAlignExtent(SbItem):
- """!Checkbox to select zoom behavior.
-
- Used by BufferedWindow (through MapFrame property).
- See tooltip for explanation.
- """
- def __init__(self, mapframe, statusbar, position = 0):
- SbItem.__init__(self, mapframe, statusbar, position)
- self.name = 'alignExtent'
- self.label = _("Display mode")
-
- self.widget = wx.CheckBox(parent = self.statusbar, id = wx.ID_ANY,
- label = _("Align region extent based on display size"))
-
- self.widget.SetValue(UserSettings.Get(group = 'display', key = 'alignExtent', subkey = 'enabled'))
- self.widget.Hide()
- self.widget.SetToolTip(wx.ToolTip (_("Align region extent based on display "
- "size from center point. "
- "Default value for new map displays can "
- "be set up in 'User GUI settings' dialog.")))
-
- class SbResolution(SbItem):
- """!Checkbox to select used display resolution.
-
- Requires MapFrame.OnRender method.
- """
- def __init__(self, mapframe, statusbar, position = 0):
- SbItem.__init__(self, mapframe, statusbar, position)
- self.name = 'resolution'
- self.label = _("Display resolution")
-
- self.widget = wx.CheckBox(parent = self.statusbar, id = wx.ID_ANY,
- label = _("Constrain display resolution to computational settings"))
-
- self.widget.SetValue(UserSettings.Get(group = 'display', key = 'compResolution', subkey = 'enabled'))
- self.widget.Hide()
- self.widget.SetToolTip(wx.ToolTip (_("Constrain display resolution "
- "to computational region settings. "
- "Default value for new map displays can "
- "be set up in 'User GUI settings' dialog.")))
-
- self.widget.Bind(wx.EVT_CHECKBOX, self.OnToggleUpdateMap)
-
- def OnToggleUpdateMap(self, event):
- """!Update display when toggle display mode
- """
- # redraw map if auto-rendering is enabled
- if self.mapFrame.IsAutoRendered():
- self.mapFrame.OnRender(None)
- class SbMapScale(SbItem):
- """!Editable combobox to get/set current map scale.
-
- Requires MapFrame.GetMapScale, MapFrame.SetMapScale
- and MapFrame.GetWindow (and GetWindow().UpdateMap()).
- """
- def __init__(self, mapframe, statusbar, position = 0):
- SbItem.__init__(self, mapframe, statusbar, position)
- self.name = 'mapscale'
- self.label = _("Map scale")
-
- self.widget = wx.ComboBox(parent = self.statusbar, id = wx.ID_ANY,
- style = wx.TE_PROCESS_ENTER,
- size = (150, -1))
-
- self.widget.SetItems(['1:1000',
- '1:5000',
- '1:10000',
- '1:25000',
- '1:50000',
- '1:100000',
- '1:1000000'])
- self.widget.Hide()
- self.widget.SetToolTip(wx.ToolTip (_("As everyone's monitors and resolutions "
- "are set differently these values are not "
- "true map scales, but should get you into "
- "the right neighborhood.")))
-
- self.widget.Bind(wx.EVT_TEXT_ENTER, self.OnChangeMapScale)
- self.widget.Bind(wx.EVT_COMBOBOX, self.OnChangeMapScale)
-
- self.lastMapScale = None
- def Update(self):
- scale = self.mapFrame.GetMapScale()
- self.statusbar.SetStatusText("")
- try:
- self.SetValue("1:%ld" % (scale + 0.5))
- except TypeError:
- pass # FIXME, why this should happen?
-
- self.lastMapScale = scale
- self.Show()
- # disable long help
- self.mapFrame.StatusbarEnableLongHelp(False)
- def OnChangeMapScale(self, event):
- """!Map scale changed by user
- """
- scale = event.GetString()
- try:
- if scale[:2] != '1:':
- raise ValueError
- value = int(scale[2:])
- except ValueError:
- self.SetValue('1:%ld' % int(self.lastMapScale))
- return
-
- self.mapFrame.SetMapScale(value)
-
- # redraw a map
- self.mapFrame.GetWindow().UpdateMap()
- self.GetWidget().SetFocus()
-
-
- class SbGoTo(SbItem):
- """!Textctrl to set coordinates which to focus on.
-
- Requires MapFrame.GetWindow, MapWindow.GoTo method.
- """
-
- def __init__(self, mapframe, statusbar, position = 0):
- SbItem.__init__(self, mapframe, statusbar, position)
- self.name = 'goto'
- self.label = _("Go to")
-
- self.widget = wx.TextCtrl(parent = self.statusbar, id = wx.ID_ANY,
- value = "", style = wx.TE_PROCESS_ENTER,
- size = (300, -1))
-
- self.widget.Hide()
-
- self.widget.Bind(wx.EVT_TEXT_ENTER, self.OnGoTo)
-
- def ReprojectENToMap(self, e, n, useDefinedProjection):
- """!Reproject east, north from user defined projection
-
- @param e,n coordinate (for DMS string, else float or string)
- @param useDefinedProjection projection defined by user in settings dialog
-
- @throws SbException if useDefinedProjection is True and projection is not defined in UserSettings
- """
- if useDefinedProjection:
- settings = UserSettings.Get(group = 'projection', key = 'statusbar', subkey = 'proj4')
- if not settings:
- raise SbException(_("Projection not defined (check the settings)"))
- else:
- # reproject values
- projIn = settings
- projOut = gcmd.RunCommand('g.proj',
- flags = 'jf',
- read = True)
- proj = projIn.split(' ')[0].split('=')[1]
- if proj in ('ll', 'latlong', 'longlat'):
- e, n = utils.DMS2Deg(e, n)
- proj, coord1 = utils.ReprojectCoordinates(coord = (e, n),
- projIn = projIn,
- projOut = projOut, flags = 'd')
- e, n = coord1
- else:
- e, n = float(e), float(n)
- proj, coord1 = utils.ReprojectCoordinates(coord = (e, n),
- projIn = projIn,
- projOut = projOut, flags = 'd')
- e, n = coord1
- elif self.mapFrame.GetMap().projinfo['proj'] == 'll':
- e, n = utils.DMS2Deg(e, n)
- else:
- e, n = float(e), float(n)
- return e, n
- def OnGoTo(self, event):
- """!Go to position
- """
- try:
- e, n = self.GetValue().split(';')
- e, n = self.ReprojectENToMap(e, n, self.mapFrame.GetProperty('projection'))
- self.mapFrame.GetWindow().GoTo(e, n)
- self.widget.SetFocus()
- except ValueError:
- # FIXME: move this code to MapWindow/BufferedWindow/MapFrame
- region = self.mapFrame.GetMap().GetCurrentRegion()
- precision = int(UserSettings.Get(group = 'projection', key = 'format',
- subkey = 'precision'))
- format = UserSettings.Get(group = 'projection', key = 'format',
- subkey = 'll')
- if self.mapFrame.GetMap().projinfo['proj'] == 'll' and format == 'DMS':
- self.SetValue("%s" % utils.Deg2DMS(region['center_easting'],
- region['center_northing'],
- precision = precision))
- else:
- self.SetValue("%.*f; %.*f" % \
- (precision, region['center_easting'],
- precision, region['center_northing']))
- except SbException, e:
- # FIXME: this may be useless since statusbar update checks user defined projection and this exception raises when user def proj does not exists
- self.statusbar.SetStatusText(str(e), 0)
- def GetCenterString(self, map):
- """!Get current map center in appropriate format"""
- region = map.GetCurrentRegion()
- precision = int(UserSettings.Get(group = 'projection', key = 'format',
- subkey = 'precision'))
- format = UserSettings.Get(group = 'projection', key = 'format',
- subkey = 'll')
- projection = UserSettings.Get(group='projection', key='statusbar', subkey='proj4')
-
- if self.mapFrame.GetProperty('projection'):
- if not projection:
- raise SbException(_("Projection not defined (check the settings)"))
- else:
- proj, coord = utils.ReprojectCoordinates(coord = (region['center_easting'],
- region['center_northing']),
- projOut = projection,
- flags = 'd')
- if coord:
- if proj in ('ll', 'latlong', 'longlat') and format == 'DMS':
- return "%s" % utils.Deg2DMS(coord[0],
- coord[1],
- precision = precision)
- else:
- return "%.*f; %.*f" % (precision, coord[0], precision, coord[1])
- else:
- raise SbException(_("Error in projection (check the settings)"))
- else:
- if self.mapFrame.GetMap().projinfo['proj'] == 'll' and format == 'DMS':
- return "%s" % utils.Deg2DMS(region['center_easting'], region['center_northing'],
- precision = precision)
- else:
- return "%.*f; %.*f" % (precision, region['center_easting'], precision, region['center_northing'])
- def SetCenter(self):
- """!Set current map center as item value"""
- center = self.GetCenterString(self.mapFrame.GetMap())
- self.SetValue(center)
-
- def Update(self):
- self.statusbar.SetStatusText("")
-
- try:
- self.SetCenter()
- self.Show()
- except SbException, e:
- self.statusbar.SetStatusText(str(e), 0)
-
- # disable long help
- self.mapFrame.StatusbarEnableLongHelp(False)
-
- class SbProjection(SbItem):
- """!Checkbox to enable user defined projection (can be set in settings)"""
- def __init__(self, mapframe, statusbar, position = 0):
- SbItem.__init__(self, mapframe, statusbar, position)
- self.name = 'projection'
- self.label = _("Projection")
-
- self.defaultLabel = _("Use defined projection")
-
- self.widget = wx.CheckBox(parent = self.statusbar, id = wx.ID_ANY,
- label = self.defaultLabel)
-
- self.widget.SetValue(False)
-
- # necessary?
- size = self.widget.GetSize()
- self.widget.SetMinSize((size[0] + 150, size[1]))
-
- self.widget.Hide()
- self.widget.SetToolTip(wx.ToolTip (_("Reproject coordinates displayed "
- "in the statusbar. Projection can be "
- "defined in GUI preferences dialog "
- "(tab 'Projection')")))
-
- def Update(self):
- self.statusbar.SetStatusText("")
- epsg = UserSettings.Get(group = 'projection', key = 'statusbar', subkey = 'epsg')
- if epsg:
- label = '%s (EPSG: %s)' % (self.defaultLabel, epsg)
- self.widget.SetLabel(label)
- else:
- self.widget.SetLabel(self.defaultLabel)
- self.Show()
-
- # disable long help
- self.mapFrame.StatusbarEnableLongHelp(False)
-
- class SbMask(SbItem):
- """!StaticText to show whether mask is activated."""
- def __init__(self, mapframe, statusbar, position = 0):
- SbItem.__init__(self, mapframe, statusbar, position)
- self.name = 'mask'
-
- self.widget = wx.StaticText(parent = self.statusbar, id = wx.ID_ANY, label = _('MASK'))
- self.widget.SetForegroundColour(wx.Colour(255, 0, 0))
- self.widget.Hide()
-
- def Update(self):
- if grass.find_file(name = 'MASK', element = 'cell')['name']:
- self.Show()
- else:
- self.Hide()
-
- class SbTextItem(SbItem):
- """!Base class for items without widgets.
-
- Only sets statusbar text.
- """
- def __init__(self, mapframe, statusbar, position = 0):
- SbItem.__init__(self, mapframe, statusbar, position)
-
- self.text = None
-
- def Show(self):
- self.statusbar.SetStatusText(self.GetValue(), self.position)
-
- def Hide(self):
- self.statusbar.SetStatusText("", self.position)
-
- def SetValue(self, value):
- self.text = value
-
- def GetValue(self):
- return self.text
-
- def GetWidget(self):
- return None
-
- def Update(self):
- self._update(longHelp = True)
- class SbDisplayGeometry(SbTextItem):
- """!Show current display resolution."""
- def __init__(self, mapframe, statusbar, position = 0):
- SbTextItem.__init__(self, mapframe, statusbar, position)
- self.name = 'displayGeometry'
- self.label = _("Display geometry")
-
- def Show(self):
- region = self.mapFrame.GetMap().GetCurrentRegion()
- self.SetValue("rows=%d; cols=%d; nsres=%.2f; ewres=%.2f" %
- (region["rows"], region["cols"],
- region["nsres"], region["ewres"]))
- SbTextItem.Show(self)
- class SbCoordinates(SbTextItem):
- """!Show map coordinates when mouse moves.
-
- Requires MapWindow.GetLastEN method."""
- def __init__(self, mapframe, statusbar, position = 0):
- SbTextItem.__init__(self, mapframe, statusbar, position)
- self.name = 'coordinates'
- self.label = _("Coordinates")
-
- def Show(self):
- precision = int(UserSettings.Get(group = 'projection', key = 'format',
- subkey = 'precision'))
- format = UserSettings.Get(group = 'projection', key = 'format',
- subkey = 'll')
- projection = self.mapFrame.GetProperty('projection')
- try:
- e, n = self.mapFrame.GetWindow().GetLastEN()
- self.SetValue(self.ReprojectENFromMap(e, n, projection, precision, format))
- except SbException, e:
- self.SetValue(e)
- except TypeError, e:
- self.SetValue("")
- except AttributeError:
- self.SetValue("") # during initialization MapFrame has no MapWindow
- SbTextItem.Show(self)
-
- def ReprojectENFromMap(self, e, n, useDefinedProjection, precision, format):
- """!Reproject east, north to user defined projection.
-
- @param e,n coordinate
-
- @throws SbException if useDefinedProjection is True and projection is not defined in UserSettings
- """
- if useDefinedProjection:
- settings = UserSettings.Get(group = 'projection', key = 'statusbar', subkey = 'proj4')
- if not settings:
- raise SbException(_("Projection not defined (check the settings)"))
- else:
- # reproject values
- proj, coord = utils.ReprojectCoordinates(coord = (e, n),
- projOut = settings,
- flags = 'd')
- if coord:
- e, n = coord
- if proj in ('ll', 'latlong', 'longlat') and format == 'DMS':
- return utils.Deg2DMS(e, n, precision = precision)
- else:
- return "%.*f; %.*f" % (precision, e, precision, n)
- else:
- raise SbException(_("Error in projection (check the settings)"))
- else:
- if self.mapFrame.GetMap().projinfo['proj'] == 'll' and format == 'DMS':
- return utils.Deg2DMS(e, n, precision = precision)
- else:
- return "%.*f; %.*f" % (precision, e, precision, n)
-
- class SbRegionExtent(SbTextItem):
- """!Shows current display region"""
- def __init__(self, mapframe, statusbar, position = 0):
- SbTextItem.__init__(self, mapframe, statusbar, position)
- self.name = 'displayRegion'
- self.label = _("Extent")
-
- def Show(self):
- precision = int(UserSettings.Get(group = 'projection', key = 'format',
- subkey = 'precision'))
- format = UserSettings.Get(group = 'projection', key = 'format',
- subkey = 'll')
- projection = self.mapFrame.GetProperty('projection')
- region = self._getRegion()
- try:
- regionReprojected = self.ReprojectRegionFromMap(region, projection, precision, format)
- self.SetValue(regionReprojected)
- except SbException, e:
- self.SetValue(e)
- SbTextItem.Show(self)
-
- def _getRegion(self):
- """!Get current display region"""
- return self.mapFrame.GetMap().GetCurrentRegion() # display region
-
- def _formatRegion(self, w, e, s, n, nsres, ewres, precision = None):
- """!Format display region string for statusbar
- @param nsres,ewres unused
- """
- if precision is not None:
- return "%.*f - %.*f, %.*f - %.*f" % (precision, w, precision, e,
- precision, s, precision, n)
- else:
- return "%s - %s, %s - %s" % (w, e, s, n)
-
-
- def ReprojectRegionFromMap(self, region, useDefinedProjection, precision, format):
- """!Reproject region values
-
- @todo reorganize this method to remove code useful only for derived class SbCompRegionExtent
- """
- if useDefinedProjection:
- settings = UserSettings.Get(group = 'projection', key = 'statusbar', subkey = 'proj4')
-
- if not settings:
- raise SbException(_("Projection not defined (check the settings)"))
- else:
- projOut = settings
- proj, coord1 = utils.ReprojectCoordinates(coord = (region["w"], region["s"]),
- projOut = projOut, flags = 'd')
- proj, coord2 = utils.ReprojectCoordinates(coord = (region["e"], region["n"]),
- projOut = projOut, flags = 'd')
- # useless, used in derived class
- proj, coord3 = utils.ReprojectCoordinates(coord = (0.0, 0.0),
- projOut = projOut, flags = 'd')
- proj, coord4 = utils.ReprojectCoordinates(coord = (region["ewres"], region["nsres"]),
- projOut = projOut, flags = 'd')
- if coord1 and coord2:
- if proj in ('ll', 'latlong', 'longlat') and format == 'DMS':
- w, s = utils.Deg2DMS(coord1[0], coord1[1], string = False,
- precision = precision)
- e, n = utils.Deg2DMS(coord2[0], coord2[1], string = False,
- precision = precision)
- ewres, nsres = utils.Deg2DMS(abs(coord3[0]) - abs(coord4[0]),
- abs(coord3[1]) - abs(coord4[1]),
- string = False, hemisphere = False,
- precision = precision)
- return self._formatRegion(w = w, s = s, e = e, n = n, ewres = ewres, nsres = nsres)
- else:
- w, s = coord1
- e, n = coord2
- ewres, nsres = coord3
- return self._formatRegion(w = w, s = s, e = e, n = n, ewres = ewres,
- nsres = nsres, precision = precision)
- else:
- raise SbException(_("Error in projection (check the settings)"))
-
- else:
- if self.mapFrame.GetMap().projinfo['proj'] == 'll' and format == 'DMS':
- w, s = utils.Deg2DMS(region["w"], region["s"],
- string = False, precision = precision)
- e, n = utils.Deg2DMS(region["e"], region["n"],
- string = False, precision = precision)
- ewres, nsres = utils.Deg2DMS(region['ewres'], region['nsres'],
- string = False, precision = precision)
- return self._formatRegion(w = w, s = s, e = e, n = n, ewres = ewres, nsres = nsres)
- else:
- w, s = region["w"], region["s"]
- e, n = region["e"], region["n"]
- ewres, nsres = region['ewres'], region['nsres']
- return self._formatRegion(w = w, s = s, e = e, n = n, ewres = ewres,
- nsres = nsres, precision = precision)
-
-
- class SbCompRegionExtent(SbRegionExtent):
- """!Shows computational region."""
- def __init__(self, mapframe, statusbar, position = 0):
- SbRegionExtent.__init__(self, mapframe, statusbar, position)
- self.name = 'computationalRegion'
- self.label = _("Comp. region")
-
- def _formatRegion(self, w, e, s, n, ewres, nsres, precision = None):
- """!Format computational region string for statusbar"""
- if precision is not None:
- return "%.*f - %.*f, %.*f - %.*f (%.*f, %.*f)" % (precision, w, precision, e,
- precision, s, precision, n,
- precision, ewres, precision, nsres)
- else:
- return "%s - %s, %s - %s (%s, %s)" % (w, e, s, n, ewres, nsres)
-
- def _getRegion(self):
- """!Returns computational region."""
- return self.mapFrame.GetMap().GetRegion() # computational region
-
-
- class SbProgress(SbItem):
- """!General progress bar to show progress.
-
- Underlaying widget is wx.Gauge.
- """
- def __init__(self, mapframe, statusbar, position = 0):
- SbItem.__init__(self, mapframe, statusbar, position)
- self.name = 'progress'
- # on-render gauge
- self.widget = wx.Gauge(parent = self.statusbar, id = wx.ID_ANY,
- range = 0, style = wx.GA_HORIZONTAL)
- self.widget.Hide()
-
- def GetRange(self):
- """!Returns progress range."""
- return self.widget.GetRange()
-
- def SetRange(self, range):
- """!Sets progress range."""
- self.widget.SetRange(range)
-
- class SbGoToGCP(SbItem):
- """!SpinCtrl to select GCP to focus on
-
- Requires MapFrame.GetSrcWindow, MapFrame.GetTgtWindow, MapFrame.GetListCtrl,
- MapFrame.GetMapCoordList.
- """
-
- def __init__(self, mapframe, statusbar, position = 0):
- SbItem.__init__(self, mapframe, statusbar, position)
- self.name = 'gotoGCP'
- self.label = _("Go to GCP No.")
- self.widget = wx.SpinCtrl(parent = self.statusbar, id = wx.ID_ANY,
- value = "", min = 0)
- self.widget.Hide()
-
- self.widget.Bind(wx.EVT_TEXT_ENTER, self.OnGoToGCP)
- self.widget.Bind(wx.EVT_SPINCTRL, self.OnGoToGCP)
-
- def OnGoToGCP(self, event):
- """!Zooms to given GCP."""
- GCPNo = self.GetValue()
- mapCoords = self.mapFrame.GetMapCoordList()
-
- if GCPNo < 0 or GCPNo > len(mapCoords): # always false, spin checks it
- gcmd.GMessage(parent=self,
- message="%s 1 - %s." % (_("Valid Range:"),
- len(mapCoords)))
- return
- if GCPNo == 0:
- return
-
- listCtrl = self.mapFrame.GetListCtrl()
-
- listCtrl.selectedkey = GCPNo
- listCtrl.selected = listCtrl.FindItemData(-1, GCPNo)
- listCtrl.render = False
- listCtrl.SetItemState(listCtrl.selected,
- wx.LIST_STATE_SELECTED,
- wx.LIST_STATE_SELECTED)
- listCtrl.render = True
-
- srcWin = self.mapFrame.GetSrcWindow()
- tgtWin = self.mapFrame.GetTgtWindow()
-
- # Source MapWindow:
- begin = (mapCoords[GCPNo][1], mapCoords[GCPNo][2])
- begin = srcWin.Cell2Pixel(begin)
- end = begin
- srcWin.Zoom(begin, end, 0)
- # redraw map
- srcWin.UpdateMap()
- if self.mapFrame.GetShowTarget():
- # Target MapWindow:
- begin = (mapCoords[GCPNo][3], mapCoords[GCPNo][4])
- begin = tgtWin.Cell2Pixel(begin)
- end = begin
- tgtWin.Zoom(begin, end, 0)
- # redraw map
- tgtWin.UpdateMap()
- self.GetWidget().SetFocus()
-
- def Update(self):
- self.statusbar.SetStatusText("")
- max = self.mapFrame.GetListCtrl().GetItemCount()
- if max < 1:
- max = 1
- self.widget.SetRange(0, max)
- self.Show()
-
- # disable long help
- self.mapFrame.StatusbarEnableLongHelp(False)
-
- class SbRMSError(SbTextItem):
- """!Shows RMS error.
-
- Requires MapFrame.GetFwdError, MapFrame.GetBkwError.
- """
- def __init__(self, mapframe, statusbar, position = 0):
- SbTextItem.__init__(self, mapframe, statusbar, position)
- self.name = 'RMSError'
- self.label = _("RMS error")
-
- def Show(self):
- self.SetValue(_("Forward: %(forw)s, Backward: %(back)s") %
- { 'forw' : self.mapFrame.GetFwdError(),
- 'back' : self.mapFrame.GetBkwError() })
- SbTextItem.Show(self)
|