123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068 |
- """!
- @package gis_set
- GRASS start-up screen.
- Initialization module for wxPython GRASS GUI.
- Location/mapset management (selection, creation, etc.).
- Classes:
- - gis_set::GRASSStartup
- - gis_set::GListBox
- - gis_set::StartUp
- (C) 2006-2014 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 and Jachym Cepicky (original author)
- @author Martin Landa <landa.martin gmail.com> (various updates)
- """
- import os
- import sys
- import shutil
- import copy
- import platform
- import codecs
- import getpass
- from core import globalvar
- from core.utils import _
- import wx
- import wx.lib.mixins.listctrl as listmix
- import wx.lib.scrolledpanel as scrolled
- from grass.script import core as grass
- from gui_core.ghelp import HelpFrame
- from core.gcmd import GMessage, GError, DecodeString, RunCommand, GWarning
- from core.utils import GetListOfLocations, GetListOfMapsets, _
- from location_wizard.dialogs import RegionDef
- from gui_core.dialogs import TextEntryDialog
- from gui_core.widgets import GenericValidator
- sys.stderr = codecs.getwriter('utf8')(sys.stderr)
- class GRASSStartup(wx.Frame):
- """!GRASS start-up screen"""
- def __init__(self, parent = None, id = wx.ID_ANY, style = wx.DEFAULT_FRAME_STYLE):
- #
- # GRASS variables
- #
- self.gisbase = os.getenv("GISBASE")
- self.grassrc = self._readGisRC()
- self.gisdbase = self.GetRCValue("GISDBASE")
- #
- # list of locations/mapsets
- #
- self.listOfLocations = []
- self.listOfMapsets = []
- self.listOfMapsetsSelectable = []
-
- wx.Frame.__init__(self, parent = parent, id = id, style = style)
-
- self.locale = wx.Locale(language = wx.LANGUAGE_DEFAULT)
-
- self.panel = scrolled.ScrolledPanel(parent = self, id = wx.ID_ANY)
-
- # i18N
-
- #
- # graphical elements
- #
- # image
- try:
- if os.getenv('ISISROOT'):
- name = os.path.join(globalvar.GUIDIR, "images", "startup_banner_isis.png")
- else:
- name = os.path.join(globalvar.GUIDIR, "images", "startup_banner.png")
- self.hbitmap = wx.StaticBitmap(self.panel, wx.ID_ANY,
- wx.Bitmap(name = name,
- type = wx.BITMAP_TYPE_PNG))
- except:
- self.hbitmap = wx.StaticBitmap(self.panel, wx.ID_ANY, wx.BitmapFromImage(wx.EmptyImage(530,150)))
- # labels
- ### crashes when LOCATION doesn't exist
- # get version & revision
- versionFile = open(os.path.join(globalvar.ETCDIR, "VERSIONNUMBER"))
- versionLine = versionFile.readline().rstrip('\n')
- versionFile.close()
- try:
- grassVersion, grassRevision = versionLine.split(' ', 1)
- if grassVersion.endswith('svn'):
- grassRevisionStr = ' (%s)' % grassRevision
- else:
- grassRevisionStr = ''
- except ValueError:
- grassVersion = versionLine
- grassRevisionStr = ''
-
- self.select_box = wx.StaticBox (parent = self.panel, id = wx.ID_ANY,
- label = " %s " % _("Choose project location and mapset"))
- self.manage_box = wx.StaticBox (parent = self.panel, id = wx.ID_ANY,
- label = " %s " % _("Manage"))
- self.lwelcome = wx.StaticText(parent = self.panel, id = wx.ID_ANY,
- label = _("Welcome to GRASS GIS %s%s\n"
- "The world's leading open source GIS") % (grassVersion, grassRevisionStr),
- style = wx.ALIGN_CENTRE)
- self.ltitle = wx.StaticText(parent = self.panel, id = wx.ID_ANY,
- label = _("Select an existing project location and mapset\n"
- "or define a new location"),
- style = wx.ALIGN_CENTRE)
- self.ldbase = wx.StaticText(parent = self.panel, id = wx.ID_ANY,
- label = _("GIS Data Directory:"))
- # no message at the beginning
- self.lmessage = wx.StaticText(parent=self.panel, id=wx.ID_ANY,
- label=_(""))
- # It is not clear if all wx versions supports color, so try-except.
- # The color itself may not be correct for all platforms/system settings
- # but in http://xoomer.virgilio.it/infinity77/wxPython/Widgets/wx.SystemSettings.html
- # there is no 'warning' color.
- try:
- self.lmessage.SetForegroundColour(wx.Colour(255, 0, 0))
- except AttributeError:
- pass
- self.llocation = wx.StaticText(parent = self.panel, id = wx.ID_ANY,
- label = _("Project location\n(projection/coordinate system)"),
- style = wx.ALIGN_CENTRE)
- self.lmapset = wx.StaticText(parent = self.panel, id = wx.ID_ANY,
- label = _("Accessible mapsets\n(directories of GIS files)"),
- style = wx.ALIGN_CENTRE)
- self.lcreate = wx.StaticText(parent = self.panel, id = wx.ID_ANY,
- label = _("Create new mapset\nin selected location"),
- style = wx.ALIGN_CENTRE)
- self.ldefine = wx.StaticText(parent = self.panel, id = wx.ID_ANY,
- label = _("Define new location"),
- style = wx.ALIGN_CENTRE)
- self.lmanageloc = wx.StaticText(parent = self.panel, id = wx.ID_ANY,
- label = _("Rename/delete selected\nmapset or location"),
- style = wx.ALIGN_CENTRE)
- # buttons
- self.bstart = wx.Button(parent = self.panel, id = wx.ID_ANY,
- label = _("Start &GRASS"))
- self.bstart.SetDefault()
- self.bexit = wx.Button(parent = self.panel, id = wx.ID_EXIT)
- self.bstart.SetMinSize((180, self.bexit.GetSize()[1]))
- self.bhelp = wx.Button(parent = self.panel, id = wx.ID_HELP)
- self.bbrowse = wx.Button(parent = self.panel, id = wx.ID_ANY,
- label = _("&Browse"))
- self.bmapset = wx.Button(parent = self.panel, id = wx.ID_ANY,
- label = _("&Create mapset"))
- self.bwizard = wx.Button(parent = self.panel, id = wx.ID_ANY,
- label = _("&Location wizard"))
- self.bwizard.SetToolTipString(_("Start location wizard."
- " After location is created successfully,"
- " GRASS session is started."))
- self.manageloc = wx.Choice(parent = self.panel, id = wx.ID_ANY,
- choices = [_('Rename mapset'), _('Rename location'),
- _('Delete mapset'), _('Delete location')])
- self.manageloc.SetSelection(0)
- # textinputs
- self.tgisdbase = wx.TextCtrl(parent = self.panel, id = wx.ID_ANY, value = "", size = (300, -1),
- style = wx.TE_PROCESS_ENTER)
- # Locations
- self.lblocations = GListBox(parent = self.panel,
- id = wx.ID_ANY, size = (180, 200),
- choices = self.listOfLocations)
-
- self.lblocations.SetColumnWidth(0, 180)
- # TODO: sort; but keep PERMANENT on top of list
- # Mapsets
- self.lbmapsets = GListBox(parent = self.panel,
- id = wx.ID_ANY, size = (180, 200),
- choices = self.listOfMapsets)
-
- self.lbmapsets.SetColumnWidth(0, 180)
- # layout & properties
- self._set_properties()
- self._do_layout()
- # events
- self.bbrowse.Bind(wx.EVT_BUTTON, self.OnBrowse)
- self.bstart.Bind(wx.EVT_BUTTON, self.OnStart)
- self.bexit.Bind(wx.EVT_BUTTON, self.OnExit)
- self.bhelp.Bind(wx.EVT_BUTTON, self.OnHelp)
- self.bmapset.Bind(wx.EVT_BUTTON, self.OnCreateMapset)
- self.bwizard.Bind(wx.EVT_BUTTON, self.OnWizard)
- self.manageloc.Bind(wx.EVT_CHOICE, self.OnManageLoc)
- self.lblocations.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnSelectLocation)
- self.lbmapsets.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnSelectMapset)
- self.lbmapsets.Bind(wx.EVT_LIST_ITEM_ACTIVATED, self.OnStart)
- self.tgisdbase.Bind(wx.EVT_TEXT_ENTER, self.OnSetDatabase)
- self.Bind(wx.EVT_CLOSE, self.OnCloseWindow)
-
- def _set_properties(self):
- """!Set frame properties"""
- self.SetTitle(_("Welcome to GRASS GIS"))
- self.SetIcon(wx.Icon(os.path.join(globalvar.ICONDIR, "grass.ico"),
- wx.BITMAP_TYPE_ICO))
- self.lwelcome.SetForegroundColour(wx.Colour(35, 142, 35))
- self.lwelcome.SetFont(wx.Font(13, wx.DEFAULT, wx.NORMAL, wx.BOLD, 0, ""))
- self.bstart.SetForegroundColour(wx.Colour(35, 142, 35))
- self.bstart.SetToolTipString(_("Enter GRASS session"))
- self.bstart.Enable(False)
- self.bmapset.Enable(False)
- self.manageloc.Enable(False)
- # set database
- if not self.gisdbase:
- # sets an initial path for gisdbase if nothing in GISRC
- if os.path.isdir(os.getenv("HOME")):
- self.gisdbase = os.getenv("HOME")
- else:
- self.gisdbase = os.getcwd()
- try:
- self.tgisdbase.SetValue(self.gisdbase)
- except UnicodeDecodeError:
- wx.MessageBox(parent = self, caption = _("Error"),
- message = _("Unable to set GRASS database. "
- "Check your locale settings."),
- style = wx.OK | wx.ICON_ERROR | wx.CENTRE)
-
- self.OnSetDatabase(None)
- location = self.GetRCValue("LOCATION_NAME")
- if location == "<UNKNOWN>":
- return
- if not os.path.isdir(os.path.join(self.gisdbase, location)):
- location = None
-
- # list of locations
- self.UpdateLocations(self.gisdbase)
- try:
- self.lblocations.SetSelection(self.listOfLocations.index(location),
- force = True)
- self.lblocations.EnsureVisible(self.listOfLocations.index(location))
- except ValueError:
- sys.stderr.write(_("ERROR: Location <%s> not found\n") % self.GetRCValue("LOCATION_NAME"))
- if len(self.listOfLocations) > 0:
- self.lblocations.SetSelection(0, force = True)
- self.lblocations.EnsureVisible(0)
- location = self.listOfLocations[0]
- else:
- return
-
- # list of mapsets
- self.UpdateMapsets(os.path.join(self.gisdbase, location))
- mapset = self.GetRCValue("MAPSET")
- if mapset:
- try:
- self.lbmapsets.SetSelection(self.listOfMapsets.index(mapset),
- force = True)
- self.lbmapsets.EnsureVisible(self.listOfMapsets.index(mapset))
- except ValueError:
- sys.stderr.write(_("ERROR: Mapset <%s> not found\n") % mapset)
- self.lbmapsets.SetSelection(0, force = True)
- self.lbmapsets.EnsureVisible(0)
-
- def _do_layout(self):
- sizer = wx.BoxSizer(wx.VERTICAL)
- dbase_sizer = wx.BoxSizer(wx.HORIZONTAL)
- location_sizer = wx.BoxSizer(wx.HORIZONTAL)
- select_boxsizer = wx.StaticBoxSizer(self.select_box, wx.VERTICAL)
- select_sizer = wx.FlexGridSizer(rows = 2, cols = 2, vgap = 4, hgap = 4)
- select_sizer.AddGrowableRow(1)
- select_sizer.AddGrowableCol(0)
- select_sizer.AddGrowableCol(1)
- manage_sizer = wx.StaticBoxSizer(self.manage_box, wx.VERTICAL)
- btns_sizer = wx.BoxSizer(wx.HORIZONTAL)
-
- # gis data directory
- dbase_sizer.Add(item = self.ldbase, proportion = 0,
- flag = wx.ALIGN_CENTER_VERTICAL |
- wx.ALIGN_CENTER_HORIZONTAL | wx.ALL,
- border = 3)
- dbase_sizer.Add(item = self.tgisdbase, proportion = 1,
- flag = wx.ALIGN_CENTER_VERTICAL | wx.ALL,
- border = 3)
- dbase_sizer.Add(item = self.bbrowse, proportion = 0,
- flag = wx.ALIGN_CENTER_VERTICAL | wx.ALL,
- border = 3)
-
- # select sizer
- select_sizer.Add(item = self.llocation, proportion = 0,
- flag = wx.ALIGN_CENTER_HORIZONTAL | wx.ALL,
- border = 3)
- select_sizer.Add(item = self.lmapset, proportion = 0,
- flag = wx.ALIGN_CENTER_HORIZONTAL | wx.ALL,
- border = 3)
- select_sizer.Add(item = self.lblocations, proportion = 1,
- flag = wx.EXPAND)
- select_sizer.Add(item = self.lbmapsets, proportion = 1,
- flag = wx.EXPAND)
-
- select_boxsizer.Add(item = select_sizer, proportion = 1,
- flag = wx.EXPAND)
-
- # define new location and mapset
- manage_sizer.Add(item = self.ldefine, proportion = 0,
- flag = wx.ALIGN_CENTER_HORIZONTAL | wx.ALL,
- border = 3)
- manage_sizer.Add(item = self.bwizard, proportion = 0,
- flag = wx.ALIGN_CENTER_HORIZONTAL | wx.BOTTOM,
- border = 5)
- manage_sizer.Add(item = self.lcreate, proportion = 0,
- flag = wx.ALIGN_CENTER_HORIZONTAL | wx.ALL,
- border = 3)
- manage_sizer.Add(item = self.bmapset, proportion = 0,
- flag = wx.ALIGN_CENTER_HORIZONTAL | wx.BOTTOM,
- border = 5)
- manage_sizer.Add(item = self.lmanageloc, proportion = 0,
- flag = wx.ALIGN_CENTER_HORIZONTAL | wx.ALL,
- border = 3)
- manage_sizer.Add(item = self.manageloc, proportion = 0,
- flag = wx.ALIGN_CENTER_HORIZONTAL | wx.BOTTOM,
- border = 5)
-
- # location sizer
- location_sizer.Add(item = select_boxsizer, proportion = 1,
- flag = wx.LEFT | wx.RIGHT | wx.EXPAND,
- border = 3)
- location_sizer.Add(item = manage_sizer, proportion = 0,
- flag = wx.RIGHT | wx.EXPAND,
- border = 3)
-
- # buttons
- btns_sizer.Add(item = self.bstart, proportion = 0,
- flag = wx.ALIGN_CENTER_HORIZONTAL |
- wx.ALIGN_CENTER_VERTICAL |
- wx.ALL,
- border = 5)
- btns_sizer.Add(item = self.bexit, proportion = 0,
- flag = wx.ALIGN_CENTER_HORIZONTAL |
- wx.ALIGN_CENTER_VERTICAL |
- wx.ALL,
- border = 5)
- btns_sizer.Add(item = self.bhelp, proportion = 0,
- flag = wx.ALIGN_CENTER_HORIZONTAL |
- wx.ALIGN_CENTER_VERTICAL |
- wx.ALL,
- border = 5)
-
- # main sizer
- sizer.Add(item = self.hbitmap,
- proportion = 0,
- flag = wx.ALIGN_CENTER_VERTICAL |
- wx.ALIGN_CENTER_HORIZONTAL |
- wx.ALL,
- border = 3) # image
- sizer.Add(item = self.lwelcome, # welcome message
- proportion = 0,
- flag = wx.ALIGN_CENTER_VERTICAL |
- wx.ALIGN_CENTER_HORIZONTAL |
- wx.BOTTOM,
- border = 5)
- sizer.Add(item = self.ltitle, # title
- proportion = 0,
- flag = wx.ALIGN_CENTER_VERTICAL |
- wx.ALIGN_CENTER_HORIZONTAL)
- sizer.Add(item = dbase_sizer, proportion = 0,
- flag = wx.ALIGN_CENTER_HORIZONTAL |
- wx.RIGHT | wx.LEFT | wx.EXPAND,
- border = 20) # GISDBASE setting
- # warning/error message
- sizer.Add(item=self.lmessage,
- proportion=0,
- flag=wx.ALIGN_CENTER_VERTICAL |
- wx.ALIGN_LEFT| wx.LEFT | wx.RIGHT | wx.BOTTOM, border=8)
- sizer.Add(item = location_sizer, proportion = 1,
- flag = wx.RIGHT | wx.LEFT | wx.EXPAND,
- border = 1)
- sizer.Add(item = btns_sizer, proportion = 0,
- flag = wx.ALIGN_CENTER_VERTICAL |
- wx.ALIGN_CENTER_HORIZONTAL |
- wx.RIGHT | wx.LEFT,
- border = 1)
-
- self.panel.SetAutoLayout(True)
- self.panel.SetSizer(sizer)
- sizer.Fit(self.panel)
- sizer.SetSizeHints(self)
-
- self.Layout()
- def _readGisRC(self):
- """!Read variables from $HOME/.grass7/rc file
- """
- grassrc = {}
-
- gisrc = os.getenv("GISRC")
-
- if gisrc and os.path.isfile(gisrc):
- try:
- rc = open(gisrc, "r")
- for line in rc.readlines():
- try:
- key, val = line.split(":", 1)
- except ValueError, e:
- sys.stderr.write(_('Invalid line in GISRC file (%s):%s\n' % \
- (e, line)))
- grassrc[key.strip()] = DecodeString(val.strip())
- finally:
- rc.close()
-
- return grassrc
- def _showWarning(self, text):
- """Displays a warning message to the user.
- There is no cleaning procedure. You should call _hideMessage when
- you know that there is everything correct now.
- """
- self.lmessage.SetLabel(_("Warning: ") + text)
- def _showError(self, text):
- """Displays a error message to the user.
- There is no cleaning procedure. You should call _hideMessage when
- you know that there is everything correct now.
- """
- self.lmessage.SetLabel(_("Error: ") + text)
- def _hideMessage(self):
- """Clears/hides the error message."""
- # we do no hide widget
- # because we do not want the dialog to change the size
- self.lmessage.SetLabel("")
- def GetRCValue(self, value):
- """!Return GRASS variable (read from GISRC)
- """
- if self.grassrc.has_key(value):
- return self.grassrc[value]
- else:
- return None
-
- def OnWizard(self, event):
- """!Location wizard started"""
- from location_wizard.wizard import LocationWizard
- gWizard = LocationWizard(parent = self,
- grassdatabase = self.tgisdbase.GetValue())
- if gWizard.location != None:
- self.tgisdbase.SetValue(gWizard.grassdatabase)
- self.OnSetDatabase(None)
- self.UpdateMapsets(os.path.join(self.gisdbase, gWizard.location))
- self.lblocations.SetSelection(self.listOfLocations.index(gWizard.location))
- self.lbmapsets.SetSelection(0)
- self.SetLocation(self.gisdbase, gWizard.location, 'PERMANENT')
- if gWizard.georeffile:
- message = _("Do you want to import <%(name)s> to the newly created location? "
- "The location's default region will be set from this imported "
- "map.") % {'name': gWizard.georeffile}
- dlg = wx.MessageDialog(parent = self,
- message = message,
- caption = _("Import data?"),
- style = wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION)
- dlg.CenterOnScreen()
- if dlg.ShowModal() == wx.ID_YES:
- self.ImportFile(gWizard.georeffile)
- else:
- self.SetDefaultRegion(location = gWizard.location)
- dlg.Destroy()
- else:
- self.SetDefaultRegion(location = gWizard.location)
- dlg = TextEntryDialog(parent=self,
- message=_("Do you want to create new mapset?"),
- caption=_("Create new mapset"),
- defaultValue=self._getDefaultMapsetName(),
- validator=GenericValidator(grass.legal_name, self._nameValidationFailed),
- style=wx.OK | wx.CANCEL | wx.HELP)
- help = dlg.FindWindowById(wx.ID_HELP)
- help.Bind(wx.EVT_BUTTON, self.OnHelp)
- if dlg.ShowModal() == wx.ID_OK:
- mapsetName = dlg.GetValue()
- self.CreateNewMapset(mapsetName)
- def SetDefaultRegion(self, location):
- """!Asks to set default region."""
- caption = _("Location <%s> created") % location
- message = _("Do you want to set the default "
- "region extents and resolution now?")
- dlg = wx.MessageDialog(parent = self,
- message = "%(caption)s.\n\n%(extent)s" % ({'caption': caption,
- 'extent': message}),
- caption = caption,
- style = wx.YES_NO | wx.NO_DEFAULT | wx.ICON_QUESTION)
- dlg.CenterOnScreen()
- if dlg.ShowModal() == wx.ID_YES:
- dlg.Destroy()
- defineRegion = RegionDef(self, location = location)
- defineRegion.CenterOnScreen()
- defineRegion.ShowModal()
- defineRegion.Destroy()
- else:
- dlg.Destroy()
- def ImportFile(self, filePath):
- """!Tries to import file as vector or raster.
- If successfull sets default region from imported map.
- """
- mapName = os.path.splitext(os.path.basename(filePath))[0]
- vectors = RunCommand('v.in.ogr', dsn = filePath, flags = 'l',
- read = True)
-
- wx.BeginBusyCursor()
- wx.Yield()
- if mapName in vectors:
- # vector detected
- returncode, error = RunCommand('v.in.ogr', dsn = filePath, output = mapName,
- getErrorMsg = True)
- else:
- returncode, error = RunCommand('r.in.gdal', input = filePath, output = mapName,
- getErrorMsg = True)
- wx.EndBusyCursor()
- if returncode != 0:
- GError(parent = self,
- message = _("Import of <%(name)s> failed.\n"
- "Reason: %(msg)s") % ({'name': filePath, 'msg': error}))
- else:
- GMessage(message = _("Data file <%(name)s> imported successfully.") % {'name': filePath},
- parent = self)
- if not grass.find_file(element = 'cell', name = mapName)['fullname'] and \
- not grass.find_file(element = 'vector', name = mapName)['fullname']:
- GError(parent = self,
- message = _("Map <%s> not found.") % mapName)
- else:
- if mapName in vectors:
- args = {'vect' : mapName}
- else:
- args = {'rast' : mapName}
- RunCommand('g.region', flags = 's', parent = self, **args)
-
- def OnManageLoc(self, event):
- """!Location management choice control handler
- """
- sel = event.GetSelection()
- if sel == 0:
- self.RenameMapset()
- elif sel == 1:
- self.RenameLocation()
- elif sel == 2:
- self.DeleteMapset()
- elif sel == 3:
- self.DeleteLocation()
-
- event.Skip()
-
- def RenameMapset(self):
- """!Rename selected mapset
- """
- location = self.listOfLocations[self.lblocations.GetSelection()]
- mapset = self.listOfMapsets[self.lbmapsets.GetSelection()]
- if mapset == 'PERMANENT':
- GMessage(parent = self,
- message = _('Mapset <PERMANENT> is required for valid GRASS location.\n\n'
- 'This mapset cannot be renamed.'))
- return
-
- dlg = TextEntryDialog(parent = self,
- message = _('Current name: %s\n\nEnter new name:') % mapset,
- caption = _('Rename selected mapset'),
- validator = GenericValidator(grass.legal_name, self._nameValidationFailed))
-
- if dlg.ShowModal() == wx.ID_OK:
- newmapset = dlg.GetValue()
- if newmapset == mapset:
- dlg.Destroy()
- return
-
- if newmapset in self.listOfMapsets:
- wx.MessageBox(parent = self,
- caption = _('Message'),
- message = _('Unable to rename mapset.\n\n'
- 'Mapset <%s> already exists in location.') % newmapset,
- style = wx.OK | wx.ICON_INFORMATION | wx.CENTRE)
- else:
- try:
- os.rename(os.path.join(self.gisdbase, location, mapset),
- os.path.join(self.gisdbase, location, newmapset))
- self.OnSelectLocation(None)
- self.lbmapsets.SetSelection(self.listOfMapsets.index(newmapset))
- except StandardError, e:
- wx.MessageBox(parent = self,
- caption = _('Error'),
- message = _('Unable to rename mapset.\n\n%s') % e,
- style = wx.OK | wx.ICON_ERROR | wx.CENTRE)
-
- dlg.Destroy()
- def RenameLocation(self):
- """!Rename selected location
- """
- location = self.listOfLocations[self.lblocations.GetSelection()]
- dlg = TextEntryDialog(parent = self,
- message = _('Current name: %s\n\nEnter new name:') % location,
- caption = _('Rename selected location'),
- validator = GenericValidator(grass.legal_name, self._nameValidationFailed))
- if dlg.ShowModal() == wx.ID_OK:
- newlocation = dlg.GetValue()
- if newlocation == location:
- dlg.Destroy()
- return
- if newlocation in self.listOfLocations:
- wx.MessageBox(parent = self,
- caption = _('Message'),
- message = _('Unable to rename location.\n\n'
- 'Location <%s> already exists in GRASS database.') % newlocation,
- style = wx.OK | wx.ICON_INFORMATION | wx.CENTRE)
- else:
- try:
- os.rename(os.path.join(self.gisdbase, location),
- os.path.join(self.gisdbase, newlocation))
- self.UpdateLocations(self.gisdbase)
- self.lblocations.SetSelection(self.listOfLocations.index(newlocation))
- self.UpdateMapsets(newlocation)
- except StandardError, e:
- wx.MessageBox(parent = self,
- caption = _('Error'),
- message = _('Unable to rename location.\n\n%s') % e,
- style = wx.OK | wx.ICON_ERROR | wx.CENTRE)
-
- dlg.Destroy()
- def DeleteMapset(self):
- """!Delete selected mapset
- """
- location = self.listOfLocations[self.lblocations.GetSelection()]
- mapset = self.listOfMapsets[self.lbmapsets.GetSelection()]
- if mapset == 'PERMANENT':
- GMessage(parent = self,
- message = _('Mapset <PERMANENT> is required for valid GRASS location.\n\n'
- 'This mapset cannot be deleted.'))
- return
-
- dlg = wx.MessageDialog(parent = self, message = _("Do you want to continue with deleting mapset <%(mapset)s> "
- "from location <%(location)s>?\n\n"
- "ALL MAPS included in this mapset will be "
- "PERMANENTLY DELETED!") % {'mapset' : mapset,
- 'location' : location},
- caption = _("Delete selected mapset"),
- style = wx.YES_NO | wx.NO_DEFAULT | wx.ICON_QUESTION)
- if dlg.ShowModal() == wx.ID_YES:
- try:
- shutil.rmtree(os.path.join(self.gisdbase, location, mapset))
- self.OnSelectLocation(None)
- self.lbmapsets.SetSelection(0)
- except:
- wx.MessageBox(message = _('Unable to delete mapset'))
- dlg.Destroy()
- def DeleteLocation(self):
- """
- Delete selected location
- """
- location = self.listOfLocations[self.lblocations.GetSelection()]
- dlg = wx.MessageDialog(parent = self, message = _("Do you want to continue with deleting "
- "location <%s>?\n\n"
- "ALL MAPS included in this location will be "
- "PERMANENTLY DELETED!") % (location),
- caption = _("Delete selected location"),
- style = wx.YES_NO | wx.NO_DEFAULT | wx.ICON_QUESTION)
- if dlg.ShowModal() == wx.ID_YES:
- try:
- shutil.rmtree(os.path.join(self.gisdbase, location))
- self.UpdateLocations(self.gisdbase)
- self.lblocations.SetSelection(0)
- self.OnSelectLocation(None)
- self.lbmapsets.SetSelection(0)
- except:
- wx.MessageBox(message = _('Unable to delete location'))
- dlg.Destroy()
- def UpdateLocations(self, dbase):
- """!Update list of locations"""
- try:
- self.listOfLocations = GetListOfLocations(dbase)
- except UnicodeEncodeError:
- GError(parent = self,
- message = _("Unable to set GRASS database. "
- "Check your locale settings."))
-
- self.lblocations.Clear()
- self.lblocations.InsertItems(self.listOfLocations, 0)
- if len(self.listOfLocations) > 0:
- self._hideMessage()
- self.lblocations.SetSelection(0)
- else:
- self.lblocations.SetSelection(wx.NOT_FOUND)
- self._showWarning(_("No GRASS location found in '%s'.")
- % self.gisdbase)
- return self.listOfLocations
- def UpdateMapsets(self, location):
- """!Update list of mapsets"""
- self.FormerMapsetSelection = wx.NOT_FOUND # for non-selectable item
-
- self.listOfMapsetsSelectable = list()
- self.listOfMapsets = GetListOfMapsets(self.gisdbase, location)
-
- self.lbmapsets.Clear()
-
- # disable mapset with denied permission
- locationName = os.path.basename(location)
-
- ret = RunCommand('g.mapset',
- read = True,
- flags = 'l',
- location = locationName,
- gisdbase = self.gisdbase)
-
- if ret:
- for line in ret.splitlines():
- self.listOfMapsetsSelectable += line.split(' ')
- else:
- RunCommand("g.gisenv",
- set = "GISDBASE=%s" % self.gisdbase)
- RunCommand("g.gisenv",
- set = "LOCATION_NAME=%s" % locationName)
- RunCommand("g.gisenv",
- set = "MAPSET=PERMANENT")
- # first run only
- self.listOfMapsetsSelectable = copy.copy(self.listOfMapsets)
-
- disabled = []
- idx = 0
- for mapset in self.listOfMapsets:
- if mapset not in self.listOfMapsetsSelectable or \
- os.path.isfile(os.path.join(self.gisdbase,
- locationName,
- mapset, ".gislock")):
- disabled.append(idx)
- idx += 1
-
- self.lbmapsets.InsertItems(self.listOfMapsets, 0, disabled = disabled)
-
- return self.listOfMapsets
- def OnSelectLocation(self, event):
- """!Location selected"""
- if event:
- self.lblocations.SetSelection(event.GetIndex())
-
- if self.lblocations.GetSelection() != wx.NOT_FOUND:
- self.UpdateMapsets(os.path.join(self.gisdbase,
- self.listOfLocations[self.lblocations.GetSelection()]))
- else:
- self.listOfMapsets = []
-
- disabled = []
- idx = 0
- try:
- locationName = self.listOfLocations[self.lblocations.GetSelection()]
- except IndexError:
- locationName = ''
-
- for mapset in self.listOfMapsets:
- if mapset not in self.listOfMapsetsSelectable or \
- os.path.isfile(os.path.join(self.gisdbase,
- locationName,
- mapset, ".gislock")):
- disabled.append(idx)
- idx += 1
- self.lbmapsets.Clear()
- self.lbmapsets.InsertItems(self.listOfMapsets, 0, disabled = disabled)
- if len(self.listOfMapsets) > 0:
- self.lbmapsets.SetSelection(0)
- if locationName:
- # enable start button when location and mapset is selected
- self.bstart.Enable()
- self.bmapset.Enable()
- self.manageloc.Enable()
- else:
- self.lbmapsets.SetSelection(wx.NOT_FOUND)
- self.bstart.Enable(False)
- self.bmapset.Enable(False)
- self.manageloc.Enable(False)
-
- def OnSelectMapset(self, event):
- """!Mapset selected"""
- self.lbmapsets.SetSelection(event.GetIndex())
- if event.GetText() not in self.listOfMapsetsSelectable:
- self.lbmapsets.SetSelection(self.FormerMapsetSelection)
- else:
- self.FormerMapsetSelection = event.GetIndex()
- event.Skip()
- def OnSetDatabase(self, event):
- """!Database set"""
- gisdbase = self.tgisdbase.GetValue()
- self._hideMessage()
- if not os.path.exists(gisdbase):
- self._showError(_("Path '%s' doesn't exist.") % gisdbase)
- return
-
- self.gisdbase = self.tgisdbase.GetValue()
- self.UpdateLocations(self.gisdbase)
-
- self.OnSelectLocation(None)
- def OnBrowse(self, event):
- """'Browse' button clicked"""
- if not event:
- defaultPath = os.getenv('HOME')
- else:
- defaultPath = ""
-
- dlg = wx.DirDialog(parent = self, message = _("Choose GIS Data Directory"),
- defaultPath = defaultPath, style = wx.DD_DEFAULT_STYLE)
-
- if dlg.ShowModal() == wx.ID_OK:
- self.gisdbase = dlg.GetPath()
- self.tgisdbase.SetValue(self.gisdbase)
- self.OnSetDatabase(event)
-
- dlg.Destroy()
- def OnCreateMapset(self, event):
- """!Create new mapset"""
- dlg = TextEntryDialog(parent = self,
- message = _('Enter name for new mapset:'),
- caption = _('Create new mapset'),
- defaultValue = self._getDefaultMapsetName(),
- validator = GenericValidator(grass.legal_name, self._nameValidationFailed))
- if dlg.ShowModal() == wx.ID_OK:
- mapset = dlg.GetValue()
- return self.CreateNewMapset(mapset = mapset)
- else:
- return False
- def CreateNewMapset(self, mapset):
- if mapset in self.listOfMapsets:
- GMessage(parent = self,
- message = _("Mapset <%s> already exists.") % mapset)
- return False
- if mapset.lower() == 'ogr':
- dlg1 = wx.MessageDialog(parent = self,
- message = _("Mapset <%s> is reserved for direct "
- "read access to OGR layers. Please consider to use "
- "another name for your mapset.\n\n"
- "Are you really sure that you want to create this mapset?") % mapset,
- caption = _("Reserved mapset name"),
- style = wx.YES_NO | wx.NO_DEFAULT | wx.ICON_QUESTION)
- ret = dlg1.ShowModal()
- dlg1.Destroy()
- if ret == wx.ID_NO:
- dlg.Destroy()
- return False
-
- try:
- self.gisdbase = self.tgisdbase.GetValue()
- location = self.listOfLocations[self.lblocations.GetSelection()]
- os.mkdir(os.path.join(self.gisdbase, location, mapset))
- # copy WIND file and its permissions from PERMANENT and set permissions to u+rw,go+r
- shutil.copy(os.path.join(self.gisdbase, location, 'PERMANENT', 'WIND'),
- os.path.join(self.gisdbase, location, mapset))
- # os.chmod(os.path.join(database,location,mapset,'WIND'), 0644)
- self.OnSelectLocation(None)
- self.lbmapsets.SetSelection(self.listOfMapsets.index(mapset))
- self.bstart.SetFocus()
- return True
- except StandardError, e:
- GError(parent = self,
- message = _("Unable to create new mapset: %s") % e,
- showTraceback = False)
- return False
- def OnStart(self, event):
- """'Start GRASS' button clicked"""
- dbase = self.tgisdbase.GetValue()
- location = self.listOfLocations[self.lblocations.GetSelection()]
- mapset = self.listOfMapsets[self.lbmapsets.GetSelection()]
-
- lockfile = os.path.join(dbase, location, mapset, '.gislock')
- if os.path.isfile(lockfile):
- dlg = wx.MessageDialog(parent = self,
- message = _("GRASS is already running in selected mapset <%(mapset)s>\n"
- "(file %(lock)s found).\n\n"
- "Concurrent use not allowed.\n\n"
- "Do you want to try to remove .gislock (note that you "
- "need permission for this operation) and continue?") %
- { 'mapset' : mapset, 'lock' : lockfile },
- caption = _("Lock file found"),
- style = wx.YES_NO | wx.NO_DEFAULT |
- wx.ICON_QUESTION | wx.CENTRE)
-
- ret = dlg.ShowModal()
- dlg.Destroy()
- if ret == wx.ID_YES:
- dlg1 = wx.MessageDialog(parent = self,
- message = _("ARE YOU REALLY SURE?\n\n"
- "If you really are running another GRASS session doing this "
- "could corrupt your data. Have another look in the processor "
- "manager just to be sure..."),
- caption = _("Lock file found"),
- style = wx.YES_NO | wx.NO_DEFAULT |
- wx.ICON_QUESTION | wx.CENTRE)
-
- ret = dlg1.ShowModal()
- dlg1.Destroy()
-
- if ret == wx.ID_YES:
- try:
- os.remove(lockfile)
- except IOError, e:
- GError(_("Unable to remove '%(lock)s'.\n\n"
- "Details: %(reason)s") % { 'lock' : lockfile, 'reason' : e})
- else:
- return
- else:
- return
- self.SetLocation(dbase, location, mapset)
- self.ExitSuccessfully()
- def SetLocation(self, dbase, location, mapset):
- RunCommand("g.gisenv",
- set = "GISDBASE=%s" % dbase)
- RunCommand("g.gisenv",
- set = "LOCATION_NAME=%s" % location)
- RunCommand("g.gisenv",
- set = "MAPSET=%s" % mapset)
- def _getDefaultMapsetName(self):
- """!Returns default name for mapset."""
- try:
- defaultName = getpass.getuser()
- defaultName.encode('ascii') # raise error if not ascii (not valid mapset name)
- except: # whatever might go wrong
- defaultName = 'user'
- return defaultName
- def ExitSuccessfully(self):
- self.Destroy()
- sys.exit(0)
- def OnExit(self, event):
- """'Exit' button clicked"""
- self.Destroy()
- sys.exit(2)
- def OnHelp(self, event):
- """'Help' button clicked"""
- # help text in lib/init/helptext.html
- RunCommand('g.manual', entry = 'helptext')
- def OnCloseWindow(self, event):
- """!Close window event"""
- event.Skip()
- sys.exit(2)
- def _nameValidationFailed(self, ctrl):
- message = _("Name <%(name)s> is not a valid name for location or mapset. "
- "Please use only ASCII characters excluding %(chars)s "
- "and space.") % {'name': ctrl.GetValue(), 'chars': '/"\'@,=*~'}
- GError(parent=self, message=message, caption=_("Invalid name"))
- class GListBox(wx.ListCtrl, listmix.ListCtrlAutoWidthMixin):
- """!Use wx.ListCtrl instead of wx.ListBox, different style for
- non-selectable items (e.g. mapsets with denied permission)"""
- def __init__(self, parent, id, size,
- choices, disabled = []):
- wx.ListCtrl.__init__(self, parent, id, size = size,
- style = wx.LC_REPORT | wx.LC_NO_HEADER | wx.LC_SINGLE_SEL |
- wx.BORDER_SUNKEN)
-
- listmix.ListCtrlAutoWidthMixin.__init__(self)
-
- self.InsertColumn(0, '')
-
- self.selected = wx.NOT_FOUND
-
- self._LoadData(choices, disabled)
-
- def _LoadData(self, choices, disabled = []):
- """!Load data into list
-
- @param choices list of item
- @param disabled list of indeces of non-selectable items
- """
- idx = 0
- for item in choices:
- index = self.InsertStringItem(sys.maxint, item)
- self.SetStringItem(index, 0, item)
-
- if idx in disabled:
- self.SetItemTextColour(idx, wx.Colour(150, 150, 150))
- idx += 1
-
- def Clear(self):
- self.DeleteAllItems()
-
- def InsertItems(self, choices, pos, disabled = []):
- self._LoadData(choices, disabled)
-
- def SetSelection(self, item, force = False):
- if item != wx.NOT_FOUND and \
- (platform.system() != 'Windows' or force):
- ### Windows -> FIXME
- self.SetItemState(item, wx.LIST_STATE_SELECTED, wx.LIST_STATE_SELECTED)
-
- self.selected = item
-
- def GetSelection(self):
- return self.selected
-
- class StartUp(wx.App):
- """!Start-up application"""
- def OnInit(self):
- if not globalvar.CheckWxVersion([2, 9]):
- wx.InitAllImageHandlers()
- StartUp = GRASSStartup()
- StartUp.CenterOnScreen()
- self.SetTopWindow(StartUp)
- StartUp.Show()
-
- if StartUp.GetRCValue("LOCATION_NAME") == "<UNKNOWN>":
- wx.MessageBox(parent = StartUp,
- caption = _('Starting GRASS for the first time'),
- message = _('GRASS needs a directory in which to store its data. '
- 'Create one now if you have not already done so. '
- 'A popular choice is "grassdata", located in '
- 'your home directory.'),
- style = wx.OK | wx.ICON_ERROR | wx.CENTRE)
-
- StartUp.OnBrowse(None)
-
- return 1
- if __name__ == "__main__":
- if os.getenv("GISBASE") is None:
- sys.exit("Failed to start GUI, GRASS GIS is not running.")
-
-
- GRASSStartUp = StartUp(0)
- GRASSStartUp.MainLoop()
|