1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288 |
- """
- @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 getpass
- from core import globalvar
- import wx
- import wx.lib.mixins.listctrl as listmix
- from grass.script import core as grass
- from core.gcmd import GMessage, GError, DecodeString, RunCommand
- from core.utils import GetListOfLocations, GetListOfMapsets
- from location_wizard.dialogs import RegionDef
- from gui_core.dialogs import TextEntryDialog
- from gui_core.widgets import GenericValidator, StaticWrapText
- from gui_core.wrap import (
- Button,
- ListCtrl,
- StaticText,
- StaticBox,
- TextCtrl,
- BitmapFromImage,
- )
- class GRASSStartup(wx.Frame):
- exit_success = 0
- # 2 is file not found from python interpreter
- exit_user_requested = 5
- """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)
- # scroll panel was used here but not properly and is probably not need
- # as long as it is not high too much
- self.panel = wx.Panel(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, 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("dev"):
- grassRevisionStr = " (%s)" % grassRevision
- else:
- grassRevisionStr = ""
- except ValueError:
- grassVersion = versionLine
- grassRevisionStr = ""
- self.gisdbase_box = StaticBox(
- parent=self.panel,
- id=wx.ID_ANY,
- label=" %s " % _("1. Select GRASS GIS database directory"),
- )
- self.location_box = StaticBox(
- parent=self.panel,
- id=wx.ID_ANY,
- label=" %s " % _("2. Select GRASS Location"),
- )
- self.mapset_box = StaticBox(
- parent=self.panel, id=wx.ID_ANY, label=" %s " % _("3. Select GRASS Mapset")
- )
- self.lmessage = StaticText(parent=self.panel)
- # 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.gisdbase_panel = wx.Panel(parent=self.panel)
- self.location_panel = wx.Panel(parent=self.panel)
- self.mapset_panel = wx.Panel(parent=self.panel)
- self.ldbase = StaticText(
- parent=self.gisdbase_panel,
- id=wx.ID_ANY,
- label=_("GRASS GIS database directory contains Locations."),
- )
- self.llocation = StaticWrapText(
- parent=self.location_panel,
- id=wx.ID_ANY,
- label=_(
- "All data in one Location is in the same "
- " coordinate reference system (projection)."
- " One Location can be one project."
- " Location contains Mapsets."
- ),
- style=wx.ALIGN_LEFT,
- )
- self.lmapset = StaticWrapText(
- parent=self.mapset_panel,
- id=wx.ID_ANY,
- label=_(
- "Mapset contains GIS data related"
- " to one project, task within one project,"
- " subregion or user."
- ),
- style=wx.ALIGN_LEFT,
- )
- try:
- for label in [self.ldbase, self.llocation, self.lmapset]:
- label.SetForegroundColour(
- wx.SystemSettings.GetColour(wx.SYS_COLOUR_GRAYTEXT)
- )
- except AttributeError:
- # for explanation of try-except see above
- pass
- # buttons
- self.bstart = Button(
- parent=self.panel, id=wx.ID_ANY, label=_("Start &GRASS session")
- )
- self.bstart.SetDefault()
- self.bexit = Button(parent=self.panel, id=wx.ID_EXIT)
- self.bstart.SetMinSize((180, self.bexit.GetSize()[1]))
- self.bhelp = Button(parent=self.panel, id=wx.ID_HELP)
- self.bbrowse = Button(
- parent=self.gisdbase_panel, id=wx.ID_ANY, label=_("&Browse")
- )
- self.bmapset = Button(
- parent=self.mapset_panel,
- id=wx.ID_ANY,
- # GTC New mapset
- label=_("&New"),
- )
- self.bmapset.SetToolTip(_("Create a new Mapset in selected Location"))
- self.bwizard = Button(
- parent=self.location_panel,
- id=wx.ID_ANY,
- # GTC New location
- label=_("N&ew"),
- )
- self.bwizard.SetToolTip(
- _(
- "Create a new location using location wizard."
- " After location is created successfully,"
- " GRASS session is started."
- )
- )
- self.rename_location_button = Button(
- parent=self.location_panel,
- id=wx.ID_ANY,
- # GTC Rename location
- label=_("Ren&ame"),
- )
- self.rename_location_button.SetToolTip(_("Rename selected location"))
- self.delete_location_button = Button(
- parent=self.location_panel,
- id=wx.ID_ANY,
- # GTC Delete location
- label=_("De&lete"),
- )
- self.delete_location_button.SetToolTip(_("Delete selected location"))
- self.rename_mapset_button = Button(
- parent=self.mapset_panel,
- id=wx.ID_ANY,
- # GTC Rename mapset
- label=_("&Rename"),
- )
- self.rename_mapset_button.SetToolTip(_("Rename selected mapset"))
- self.delete_mapset_button = Button(
- parent=self.mapset_panel,
- id=wx.ID_ANY,
- # GTC Delete mapset
- label=_("&Delete"),
- )
- self.delete_mapset_button.SetToolTip(_("Delete selected mapset"))
- # textinputs
- self.tgisdbase = TextCtrl(
- parent=self.gisdbase_panel,
- id=wx.ID_ANY,
- value="",
- size=(300, -1),
- style=wx.TE_PROCESS_ENTER,
- )
- # Locations
- self.lblocations = GListBox(
- parent=self.location_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.mapset_panel,
- id=wx.ID_ANY,
- size=(180, 200),
- choices=self.listOfMapsets,
- )
- self.lbmapsets.SetColumnWidth(0, 180)
- # layout & properties, first do layout so everything is created
- self._do_layout()
- self._set_properties(grassVersion, grassRevisionStr)
- # 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.rename_location_button.Bind(wx.EVT_BUTTON, self.RenameLocation)
- self.delete_location_button.Bind(wx.EVT_BUTTON, self.DeleteLocation)
- self.rename_mapset_button.Bind(wx.EVT_BUTTON, self.RenameMapset)
- self.delete_mapset_button.Bind(wx.EVT_BUTTON, self.DeleteMapset)
- 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, version, revision):
- """Set frame properties"""
- self.SetTitle(_("GRASS GIS %s startup%s") % (version, revision))
- self.SetIcon(
- wx.Icon(os.path.join(globalvar.ICONDIR, "grass.ico"), wx.BITMAP_TYPE_ICO)
- )
- self.bstart.SetToolTip(_("Enter GRASS session"))
- self.bstart.Enable(False)
- self.bmapset.Enable(False)
- # this all was originally a choice, perhaps just mapset needed
- self.rename_location_button.Enable(False)
- self.delete_location_button.Enable(False)
- self.rename_mapset_button.Enable(False)
- self.delete_mapset_button.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)
- self.sizer = sizer # for the layout call after changing message
- dbase_sizer = wx.BoxSizer(wx.HORIZONTAL)
- location_mapset_sizer = wx.BoxSizer(wx.HORIZONTAL)
- gisdbase_panel_sizer = wx.BoxSizer(wx.VERTICAL)
- gisdbase_boxsizer = wx.StaticBoxSizer(self.gisdbase_box, wx.VERTICAL)
- btns_sizer = wx.BoxSizer(wx.HORIZONTAL)
- self.gisdbase_panel.SetSizer(gisdbase_panel_sizer)
- # gis data directory
- gisdbase_boxsizer.Add(
- self.gisdbase_panel, proportion=1, flag=wx.EXPAND | wx.ALL, border=1
- )
- gisdbase_panel_sizer.Add(
- dbase_sizer, proportion=1, flag=wx.EXPAND | wx.ALL, border=1
- )
- gisdbase_panel_sizer.Add(
- self.ldbase, proportion=0, flag=wx.EXPAND | wx.ALL, border=1
- )
- dbase_sizer.Add(
- self.tgisdbase,
- proportion=1,
- flag=wx.ALIGN_CENTER_VERTICAL | wx.ALL,
- border=1,
- )
- dbase_sizer.Add(
- self.bbrowse, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL | wx.ALL, border=1
- )
- gisdbase_panel_sizer.Fit(self.gisdbase_panel)
- # location and mapset lists
- def layout_list_box(box, panel, list_box, buttons, description):
- panel_sizer = wx.BoxSizer(wx.VERTICAL)
- main_sizer = wx.BoxSizer(wx.HORIZONTAL)
- box_sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
- buttons_sizer = wx.BoxSizer(wx.VERTICAL)
- panel.SetSizer(panel_sizer)
- panel_sizer.Fit(panel)
- main_sizer.Add(list_box, proportion=1, flag=wx.EXPAND | wx.ALL, border=1)
- main_sizer.Add(
- buttons_sizer,
- proportion=0,
- flag=wx.ALIGN_CENTER_HORIZONTAL | wx.ALL,
- border=1,
- )
- for button in buttons:
- buttons_sizer.Add(
- button,
- proportion=0,
- flag=wx.EXPAND | wx.LEFT | wx.RIGHT | wx.BOTTOM,
- border=3,
- )
- box_sizer.Add(panel, proportion=1, flag=wx.EXPAND | wx.ALL, border=1)
- panel_sizer.Add(main_sizer, proportion=1, flag=wx.EXPAND | wx.ALL, border=1)
- panel_sizer.Add(
- description, proportion=0, flag=wx.EXPAND | wx.ALL, border=1
- )
- return box_sizer
- location_boxsizer = layout_list_box(
- box=self.location_box,
- panel=self.location_panel,
- list_box=self.lblocations,
- buttons=[
- self.bwizard,
- self.rename_location_button,
- self.delete_location_button,
- ],
- description=self.llocation,
- )
- mapset_boxsizer = layout_list_box(
- box=self.mapset_box,
- panel=self.mapset_panel,
- list_box=self.lbmapsets,
- buttons=[
- self.bmapset,
- self.rename_mapset_button,
- self.delete_mapset_button,
- ],
- description=self.lmapset,
- )
- # location and mapset sizer
- location_mapset_sizer.Add(
- location_boxsizer,
- proportion=1,
- flag=wx.LEFT | wx.RIGHT | wx.EXPAND,
- border=3,
- )
- location_mapset_sizer.Add(
- mapset_boxsizer, proportion=1, flag=wx.RIGHT | wx.EXPAND, border=3
- )
- # buttons
- btns_sizer.Add(
- self.bstart,
- proportion=0,
- flag=wx.ALIGN_CENTER_HORIZONTAL | wx.ALIGN_CENTER_VERTICAL | wx.ALL,
- border=5,
- )
- btns_sizer.Add(
- self.bexit,
- proportion=0,
- flag=wx.ALIGN_CENTER_HORIZONTAL | wx.ALIGN_CENTER_VERTICAL | wx.ALL,
- border=5,
- )
- btns_sizer.Add(
- self.bhelp,
- proportion=0,
- flag=wx.ALIGN_CENTER_HORIZONTAL | wx.ALIGN_CENTER_VERTICAL | wx.ALL,
- border=5,
- )
- # main sizer
- sizer.Add(
- self.hbitmap,
- proportion=0,
- flag=wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_CENTER_HORIZONTAL | wx.ALL,
- border=3,
- ) # image
- sizer.Add(
- gisdbase_boxsizer,
- proportion=0,
- flag=wx.ALIGN_CENTER_HORIZONTAL | wx.RIGHT | wx.LEFT | wx.TOP | wx.EXPAND,
- border=3,
- ) # GISDBASE setting
- # warning/error message
- sizer.Add(
- self.lmessage,
- proportion=0,
- flag=wx.ALIGN_CENTER_VERTICAL | wx.ALIGN_LEFT | wx.ALL | wx.EXPAND,
- border=5,
- )
- sizer.Add(
- location_mapset_sizer,
- proportion=1,
- flag=wx.RIGHT | wx.LEFT | wx.EXPAND,
- border=1,
- )
- sizer.Add(
- btns_sizer,
- proportion=0,
- flag=wx.ALIGN_CENTER_VERTICAL
- | wx.ALIGN_CENTER_HORIZONTAL
- | wx.RIGHT
- | wx.LEFT,
- border=3,
- )
- self.panel.SetAutoLayout(True)
- self.panel.SetSizer(sizer)
- sizer.Fit(self.panel)
- sizer.SetSizeHints(self)
- self.Layout()
- def _readGisRC(self):
- """Read variables from $HOME/.grass8/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 as 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, hint or info message to the user.
- This function can be used for all kinds of messages except for
- error messages.
- .. note::
- There is no cleaning procedure. You should call _hideMessage when
- you know that there is everything correct now.
- """
- self.lmessage.SetLabel(text)
- self.lmessage.Wrap(self.GetClientSize()[0])
- self.sizer.Layout()
- def _showError(self, text):
- """Displays a error message to the user.
- This function should be used only when something serious and unexpected
- happens, otherwise _showWarning should be used.
- .. note::
- There is no cleaning procedure. You should call _hideMessage when
- you know that there is everything correct now.
- """
- self.lmessage.SetLabel(_("Error: {text}").format(text=text))
- self.lmessage.Wrap(self.GetClientSize()[0])
- self.sizer.Layout()
- 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("")
- self.sizer.Layout()
- def GetRCValue(self, value):
- """Return GRASS variable (read from GISRC)"""
- if value in self.grassrc:
- 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 is not 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?"
- ) % {"name": gWizard.georeffile}
- dlg = wx.MessageDialog(
- parent=self,
- message=message,
- caption=_("Import data?"),
- style=wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION,
- )
- dlg.CenterOnParent()
- if dlg.ShowModal() == wx.ID_YES:
- self.ImportFile(gWizard.georeffile)
- dlg.Destroy()
- if gWizard.default_region:
- defineRegion = RegionDef(self, location=gWizard.location)
- defineRegion.CenterOnParent()
- defineRegion.ShowModal()
- defineRegion.Destroy()
- if gWizard.user_mapset:
- dlg = TextEntryDialog(
- parent=self,
- message=_("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 ImportFile(self, filePath):
- """Tries to import file as vector or raster.
- If successful sets default region from imported map.
- """
- RunCommand("db.connect", flags="c")
- mapName = os.path.splitext(os.path.basename(filePath))[0]
- vectors = RunCommand("v.in.ogr", input=filePath, flags="l", read=True)
- wx.BeginBusyCursor()
- wx.GetApp().Yield()
- if mapName in vectors:
- # vector detected
- returncode, error = RunCommand(
- "v.in.ogr", input=filePath, output=mapName, flags="e", getErrorMsg=True
- )
- else:
- returncode, error = RunCommand(
- "r.in.gdal", input=filePath, output=mapName, flags="e", 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. "
- "The location's default region was set from this imported map."
- )
- % {"name": filePath},
- parent=self,
- )
- # the event can be refactored out by using lambda in bind
- def RenameMapset(self, event):
- """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 Exception as 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, event):
- """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 Exception as 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, event):
- """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, event):
- """
- 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, UnicodeDecodeError) as e:
- GError(
- parent=self,
- message=_(
- "Unicode error detected. "
- "Check your locale settings. Details: {0}"
- ).format(e),
- showTraceback=False,
- )
- 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'."
- " Create a new Location or choose different"
- " GRASS database directory."
- )
- % 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.bstart.SetFocus()
- self.bmapset.Enable()
- # replacing disabled choice, perhaps just mapset needed
- self.rename_location_button.Enable()
- self.delete_location_button.Enable()
- self.rename_mapset_button.Enable()
- self.delete_mapset_button.Enable()
- else:
- self.lbmapsets.SetSelection(wx.NOT_FOUND)
- self.bstart.Enable(False)
- self.bmapset.Enable(False)
- # this all was originally a choice, perhaps just mapset needed
- self.rename_location_button.Enable(False)
- self.delete_location_button.Enable(False)
- self.rename_mapset_button.Enable(False)
- self.delete_mapset_button.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:
- dlg1.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 Exception as 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 as 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()
- # raise error if not ascii (not valid mapset name)
- defaultName.encode("ascii")
- except: # whatever might go wrong
- defaultName = "user"
- return defaultName
- def ExitSuccessfully(self):
- self.Destroy()
- sys.exit(self.exit_success)
- def OnExit(self, event):
- """'Exit' button clicked"""
- self.Destroy()
- sys.exit(self.exit_user_requested)
- 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(self.exit_user_requested)
- 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(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=[]):
- 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 indices of non-selectable items
- """
- idx = 0
- count = self.GetItemCount()
- for item in choices:
- index = self.InsertItem(count + idx, item)
- self.SetItem(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):
- StartUp = GRASSStartup()
- StartUp.CenterOnScreen()
- self.SetTopWindow(StartUp)
- StartUp.Show()
- if StartUp.GetRCValue("LOCATION_NAME") == "<UNKNOWN>":
- # TODO: This is not ideal, either it should be checked elsewhere
- # where other checks are performed or it should use some public
- # API. There is no reason for not exposing it.
- # TODO: another question is what should be warning, hint or message
- StartUp._showWarning(
- _(
- "GRASS needs a directory (GRASS database) "
- "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. "
- "Press Browse button to select the directory."
- )
- )
- 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()
|