gselect.py 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673
  1. """!
  2. @package gselect
  3. @brief Custom control that selects elements
  4. Classes:
  5. - Select
  6. - VectorSelect
  7. - TreeCrtlComboPopup
  8. - VectorDBInfo
  9. - LayerSelect
  10. - DriverSelect
  11. - DatabaseSelect
  12. - ColumnSelect
  13. (C) 2007-2009 by the GRASS Development Team This program is free
  14. software under the GNU General Public License (>=v2). Read the file
  15. COPYING that comes with GRASS for details.
  16. @author Michael Barton
  17. @author Martin Landa <landa.martin gmail.com>
  18. """
  19. import os
  20. import sys
  21. import wx
  22. import wx.combo
  23. import grass.script as grass
  24. import globalvar
  25. import gcmd
  26. import utils
  27. from preferences import globalSettings as UserSettings
  28. class Select(wx.combo.ComboCtrl):
  29. def __init__(self, parent, id, size = globalvar.DIALOG_GSELECT_SIZE,
  30. type = None, multiple = False, mapsets = None, exclude = [],
  31. updateOnPopup = True):
  32. """!Custom control to create a ComboBox with a tree control
  33. to display and select GIS elements within acessible mapsets.
  34. Elements can be selected with mouse. Can allow multiple selections, when
  35. argument multiple=True. Multiple selections are separated by commas.
  36. """
  37. wx.combo.ComboCtrl.__init__(self, parent=parent, id=id, size=size)
  38. self.GetChildren()[0].SetName("Select")
  39. self.GetChildren()[0].type = type
  40. self.tcp = TreeCtrlComboPopup()
  41. self.SetPopupControl(self.tcp)
  42. self.SetPopupExtents(0,100)
  43. if type:
  44. self.tcp.SetData(type = type, mapsets = mapsets,
  45. exclude = exclude, multiple = multiple,
  46. updateOnPopup = updateOnPopup)
  47. def SetElementList(self, type, mapsets = None, exclude = []):
  48. """!Set element list
  49. @param type GIS element type
  50. @param mapsets list of acceptable mapsets (None for all in search path)
  51. @param exclude list of GIS elements to be excluded
  52. """
  53. self.tcp.SetData(type = type, mapsets = mapsets,
  54. exclude = exclude)
  55. def GetElementList(self):
  56. """!Load elements"""
  57. self.tcp.GetElementList()
  58. class VectorSelect(Select):
  59. def __init__(self, parent, ftype, **kwargs):
  60. """!Custom to create a ComboBox with a tree control to display and
  61. select vector maps. Control allows to filter vector maps. If you
  62. don't need this feature use Select class instead
  63. @ftype filter vector maps based on feature type
  64. """
  65. Select.__init__(self, parent = parent, id = wx.ID_ANY,
  66. type = 'vector', **kwargs)
  67. self.ftype = ftype
  68. # remove vector maps which do not contain given feature type
  69. self.tcp.SetFilter(self.__filterElements)
  70. def __filterElements(self, parentItem):
  71. """!Filter vector maps in given mapset"""
  72. root = self.tcp.seltree.GetRootItem()
  73. item, cookie = self.tcp.seltree.GetFirstChild(parentItem)
  74. while item:
  75. if self.tcp.seltree.GetItemParent(item) != root:
  76. # skip Mapset items
  77. vectorName = self.tcp.seltree.GetItemText(item)
  78. try:
  79. if int(grass.vector_info_topo(vectorName)[self.ftype]) < 1:
  80. self.tcp.seltree.Delete(item)
  81. except KeyError:
  82. self.tcp.seltree.Delete(item)
  83. if not item:
  84. item, cookie = self.tcp.seltree.GetNextChild(parentItem, cookie)
  85. continue
  86. if self.tcp.seltree.ItemHasChildren(item):
  87. item = self.__filterElements(item)
  88. item, cookie = self.tcp.seltree.GetNextChild(parentItem, cookie)
  89. class TreeCtrlComboPopup(wx.combo.ComboPopup):
  90. """!Create a tree ComboBox for selecting maps and other GIS elements
  91. in accessible mapsets within the current location
  92. """
  93. # overridden ComboPopup methods
  94. def Init(self):
  95. self.value = [] # for multiple is False -> len(self.value) in [0,1]
  96. self.curitem = None
  97. self.multiple = False
  98. self.type = None
  99. self.mapsets = []
  100. self.exclude = []
  101. self.SetFilter(None)
  102. def Create(self, parent):
  103. self.seltree = wx.TreeCtrl(parent, style=wx.TR_HIDE_ROOT
  104. |wx.TR_HAS_BUTTONS
  105. |wx.TR_SINGLE
  106. |wx.TR_LINES_AT_ROOT
  107. |wx.SIMPLE_BORDER
  108. |wx.TR_FULL_ROW_HIGHLIGHT)
  109. self.seltree.Bind(wx.EVT_MOTION, self.OnMotion)
  110. self.seltree.Bind(wx.EVT_LEFT_DOWN, self.OnLeftDown)
  111. self.seltree.Bind(wx.EVT_TREE_ITEM_EXPANDING, self.mapsetExpanded)
  112. self.seltree.Bind(wx.EVT_TREE_ITEM_COLLAPSED, self.mapsetCollapsed)
  113. self.seltree.Bind(wx.EVT_TREE_ITEM_ACTIVATED, self.mapsetActivated)
  114. self.seltree.Bind(wx.EVT_TREE_SEL_CHANGED, self.mapsetSelected)
  115. self.seltree.Bind(wx.EVT_TREE_DELETE_ITEM, lambda x: None)
  116. # the following dummy handler are needed to keep tree events from propagating up to
  117. # the parent GIS Manager layer tree
  118. def mapsetExpanded(self, event):
  119. pass
  120. def mapsetCollapsed(self, event):
  121. pass
  122. def mapsetActivated(self, event):
  123. pass
  124. def mapsetSelected(self, event):
  125. pass
  126. # end of dummy events
  127. def GetControl(self):
  128. return self.seltree
  129. def GetStringValue(self):
  130. str = ""
  131. for value in self.value:
  132. str += value + ","
  133. str = str.rstrip(',')
  134. return str
  135. def SetFilter(self, filter):
  136. """!Set filter for GIS elements, see e.g. VectorSelect"""
  137. self.filterElements = filter
  138. def OnPopup(self, force = False):
  139. """!Limited only for first selected"""
  140. if not force and not self.updateOnPopup:
  141. return
  142. self.GetElementList()
  143. def GetElementList(self):
  144. """!Get filtered list of GIS elements in accessible mapsets
  145. and display as tree with all relevant elements displayed
  146. beneath each mapset branch
  147. """
  148. # update list
  149. self.seltree.DeleteAllItems()
  150. self._getElementList(self.type, self.mapsets, self.exclude)
  151. if self.filterElements:
  152. self.filterElements(self.seltree.GetRootItem())
  153. if len(self.value) > 0:
  154. root = self.seltree.GetRootItem()
  155. if not root:
  156. return
  157. item = self.FindItem(root, self.value[0])
  158. try:
  159. self.seltree.EnsureVisible(item)
  160. self.seltree.SelectItem(item)
  161. except:
  162. pass
  163. def SetStringValue(self, value):
  164. # this assumes that item strings are unique...
  165. root = self.seltree.GetRootItem()
  166. if not root:
  167. return
  168. found = self.FindItem(root, value)
  169. if found:
  170. self.value.append(found)
  171. self.seltree.SelectItem(found)
  172. def GetAdjustedSize(self, minWidth, prefHeight, maxHeight):
  173. return wx.Size(minWidth, min(200, maxHeight))
  174. def _getElementList(self, element, mapsets=None, exclude=[]):
  175. """!Get list of GIS elements in accessible mapsets and display as tree
  176. with all relevant elements displayed beneath each mapset branch
  177. @param element GIS element
  178. @param mapsets list of acceptable mapsets (None for all mapsets in search path)
  179. @param exclude list of GIS elements to be excluded
  180. """
  181. # get current mapset
  182. curr_mapset = grass.gisenv()['MAPSET']
  183. # list of mapsets in current location
  184. if mapsets is None:
  185. mapsets = utils.ListOfMapsets()
  186. # map element types to g.mlist types
  187. elementdict = {'cell':'rast',
  188. 'raster':'rast',
  189. 'rast':'rast',
  190. 'raster files':'rast',
  191. 'grid3':'rast3d',
  192. 'rast3d':'rast3d',
  193. 'raster3D':'rast3d',
  194. 'raster3D files':'rast3d',
  195. 'vector':'vect',
  196. 'vect':'vect',
  197. 'binary vector files':'vect',
  198. 'dig':'oldvect',
  199. 'oldvect':'oldvect',
  200. 'old vector':'oldvect',
  201. 'dig_ascii':'asciivect',
  202. 'asciivect':'asciivect',
  203. 'asciivector':'asciivect',
  204. 'ascii vector files':'asciivect',
  205. 'icons':'icon',
  206. 'icon':'icon',
  207. 'paint icon files':'icon',
  208. 'paint/labels':'labels',
  209. 'labels':'labels',
  210. 'label':'labels',
  211. 'paint label files':'labels',
  212. 'site_lists':'sites',
  213. 'sites':'sites',
  214. 'site list':'sites',
  215. 'site list files':'sites',
  216. 'windows':'region',
  217. 'region':'region',
  218. 'region definition':'region',
  219. 'region definition files':'region',
  220. 'windows3d':'region3d',
  221. 'region3d':'region3d',
  222. 'region3D definition':'region3d',
  223. 'region3D definition files':'region3d',
  224. 'group':'group',
  225. 'imagery group':'group',
  226. 'imagery group files':'group',
  227. '3d.view':'3dview',
  228. '3dview':'3dview',
  229. '3D viewing parameters':'3dview',
  230. '3D view parameters':'3dview'}
  231. if element not in elementdict:
  232. self.AddItem(_('Not selectable element'))
  233. return
  234. # get directory tree nodes
  235. # reorder mapsets based on search path (TODO)
  236. for i in range(len(mapsets)):
  237. if i > 0 and mapsets[i] == curr_mapset:
  238. mapsets[i] = mapsets[0]
  239. mapsets[0] = curr_mapset
  240. if globalvar.have_mlist:
  241. filesdict = grass.mlist_grouped(elementdict[element])
  242. else:
  243. filesdict = grass.list_grouped(elementdict[element])
  244. first_dir = None
  245. for dir in mapsets:
  246. dir_node = self.AddItem('Mapset: ' + dir)
  247. if not first_dir:
  248. first_dir = dir_node
  249. self.seltree.SetItemTextColour(dir_node, wx.Colour(50, 50, 200))
  250. try:
  251. elem_list = filesdict[dir]
  252. elem_list.sort(key=str.lower)
  253. for elem in elem_list:
  254. if elem != '':
  255. fullqElem = elem + '@' + dir
  256. if len(exclude) > 0 and fullqElem in exclude:
  257. continue
  258. self.AddItem(fullqElem, parent=dir_node)
  259. except:
  260. continue
  261. if self.seltree.ItemHasChildren(dir_node):
  262. sel = UserSettings.Get(group='general', key='elementListExpand',
  263. subkey='selection')
  264. collapse = True
  265. if sel == 0: # collapse all except PERMANENT and current
  266. if dir in ('PERMANENT', curr_mapset):
  267. collapse = False
  268. elif sel == 1: # collapse all except PERMANENT
  269. if dir == 'PERMANENT':
  270. collapse = False
  271. elif sel == 2: # collapse all except current
  272. if dir == curr_mapset:
  273. collapse = False
  274. elif sel == 3: # collapse all
  275. pass
  276. elif sel == 4: # expand all
  277. collapse = False
  278. if collapse:
  279. self.seltree.Collapse(dir_node)
  280. else:
  281. self.seltree.Expand(dir_node)
  282. if first_dir:
  283. # select first mapset (MSW hack)
  284. self.seltree.SelectItem(first_dir)
  285. # helpers
  286. def FindItem(self, parentItem, text):
  287. item, cookie = self.seltree.GetFirstChild(parentItem)
  288. while item:
  289. if self.seltree.GetItemText(item) == text:
  290. return item
  291. if self.seltree.ItemHasChildren(item):
  292. item = self.FindItem(item, text)
  293. item, cookie = self.seltree.GetNextChild(parentItem, cookie)
  294. return wx.TreeItemId()
  295. def AddItem(self, value, parent=None):
  296. if not parent:
  297. root = self.seltree.GetRootItem()
  298. if not root:
  299. root = self.seltree.AddRoot("<hidden root>")
  300. parent = root
  301. item = self.seltree.AppendItem(parent, text=value)
  302. return item
  303. def OnMotion(self, evt):
  304. # have the selection follow the mouse, like in a real combobox
  305. item, flags = self.seltree.HitTest(evt.GetPosition())
  306. if item and flags & wx.TREE_HITTEST_ONITEMLABEL:
  307. self.seltree.SelectItem(item)
  308. self.curitem = item
  309. evt.Skip()
  310. def OnLeftDown(self, evt):
  311. # do the combobox selection
  312. item, flags = self.seltree.HitTest(evt.GetPosition())
  313. if item and flags & wx.TREE_HITTEST_ONITEMLABEL:
  314. self.curitem = item
  315. if self.seltree.GetRootItem() == self.seltree.GetItemParent(item):
  316. self.value = [] # cannot select mapset item
  317. else:
  318. if self.multiple is True:
  319. # text item should be unique
  320. self.value.append(self.seltree.GetItemText(item))
  321. else:
  322. self.value = [self.seltree.GetItemText(item), ]
  323. self.Dismiss()
  324. evt.Skip()
  325. def SetData(self, **kargs):
  326. """!Set object properties"""
  327. if kargs.has_key('type'):
  328. self.type = kargs['type']
  329. if kargs.has_key('mapsets'):
  330. self.mapsets = kargs['mapsets']
  331. if kargs.has_key('exclude'):
  332. self.exclude = kargs['exclude']
  333. if kargs.has_key('multiple'):
  334. self.multiple = kargs['multiple']
  335. if kargs.has_key('updateOnPopup'):
  336. self.updateOnPopup = kargs['updateOnPopup']
  337. class VectorDBInfo:
  338. """!Class providing information about attribute tables
  339. linked to a vector map"""
  340. def __init__(self, map):
  341. self.map = map
  342. # dictionary of layer number and associated (driver, database, table)
  343. self.layers = {}
  344. # dictionary of table and associated columns (type, length, values, ids)
  345. self.tables = {}
  346. if not self.__CheckDBConnection(): # -> self.layers
  347. return
  348. self.__DescribeTables() # -> self.tables
  349. def __CheckDBConnection(self):
  350. """!Check DB connection"""
  351. nuldev = file(os.devnull, 'w+')
  352. self.layers = grass.vector_db(map=self.map, stderr=nuldev)
  353. nuldev.close()
  354. if (len(self.layers.keys()) == 0):
  355. return False
  356. return True
  357. def __DescribeTables(self):
  358. """!Describe linked tables"""
  359. for layer in self.layers.keys():
  360. # determine column names and types
  361. table = self.layers[layer]["table"]
  362. columns = {} # {name: {type, length, [values], [ids]}}
  363. i = 0
  364. for item in grass.db_describe(table = self.layers[layer]["table"],
  365. driver = self.layers[layer]["driver"],
  366. database = self.layers[layer]["database"])['cols']:
  367. name, type, length = item
  368. # FIXME: support more datatypes
  369. if type.lower() == "integer":
  370. ctype = int
  371. elif type.lower() == "double precision":
  372. ctype = float
  373. else:
  374. ctype = str
  375. columns[name.strip()] = { 'index' : i,
  376. 'type' : type.lower(),
  377. 'ctype' : ctype,
  378. 'length' : int(length),
  379. 'values' : [],
  380. 'ids' : []}
  381. i += 1
  382. # check for key column
  383. # v.db.connect -g/p returns always key column name lowercase
  384. if self.layers[layer]["key"] not in columns.keys():
  385. for col in columns.keys():
  386. if col.lower() == self.layers[layer]["key"]:
  387. self.layers[layer]["key"] = col.upper()
  388. break
  389. self.tables[table] = columns
  390. return True
  391. def Reset(self):
  392. """!Reset"""
  393. for layer in self.layers:
  394. table = self.layers[layer]["table"] # get table desc
  395. columns = self.tables[table]
  396. for name in self.tables[table].keys():
  397. self.tables[table][name]['values'] = []
  398. self.tables[table][name]['ids'] = []
  399. def GetName(self):
  400. """!Get vector name"""
  401. return self.map
  402. def GetKeyColumn(self, layer):
  403. """!Get key column of given layer
  404. @param layer vector layer number
  405. """
  406. return self.layers[layer]['key']
  407. def GetTable(self, layer):
  408. """!Get table name of given layer
  409. @param layer vector layer number
  410. """
  411. return self.layers[layer]['table']
  412. def GetTableDesc(self, table):
  413. """!Get table columns
  414. @param table table name
  415. """
  416. return self.tables[table]
  417. class LayerSelect(wx.Choice):
  418. """!Creates combo box for selecting data layers defined for vector.
  419. The 'layer' terminology is likely to change for GRASS 7
  420. """
  421. def __init__(self, parent,
  422. id=wx.ID_ANY, pos=wx.DefaultPosition,
  423. size=globalvar.DIALOG_LAYER_SIZE,
  424. vector=None, choices=[], all=False, default=None):
  425. super(LayerSelect, self).__init__(parent, id, pos=pos, size=size,
  426. choices=choices)
  427. self.all = all
  428. self.SetName("LayerSelect")
  429. # default value
  430. self.default = default
  431. if len(choices) > 1:
  432. return
  433. if vector:
  434. self.InsertLayers(vector)
  435. else:
  436. if all:
  437. self.SetItems(['-1', '1'])
  438. else:
  439. self.SetItems(['1'])
  440. self.SetStringSelection('1')
  441. def InsertLayers(self, vector):
  442. """!Insert layers for a vector into the layer combobox"""
  443. layerchoices = utils.GetVectorNumberOfLayers(vector)
  444. if self.all:
  445. layerchoices.insert(0, '-1')
  446. if len(layerchoices) > 1:
  447. self.SetItems(layerchoices)
  448. self.SetStringSelection('1')
  449. else:
  450. self.SetItems(['1'])
  451. self.SetStringSelection('1')
  452. if self.default:
  453. self.SetStringSelection(str(self.default))
  454. class DriverSelect(wx.ComboBox):
  455. """!Creates combo box for selecting database driver.
  456. """
  457. def __init__(self, parent, choices, value,
  458. id=wx.ID_ANY, pos=wx.DefaultPosition,
  459. size=globalvar.DIALOG_LAYER_SIZE, **kargs):
  460. super(DriverSelect, self).__init__(parent, id, value, pos, size,
  461. choices, style=wx.CB_READONLY)
  462. self.SetName("DriverSelect")
  463. self.SetStringSelection(value)
  464. class DatabaseSelect(wx.TextCtrl):
  465. """!Creates combo box for selecting database driver.
  466. """
  467. def __init__(self, parent, value='',
  468. id=wx.ID_ANY, pos=wx.DefaultPosition,
  469. size=globalvar.DIALOG_TEXTCTRL_SIZE, **kargs):
  470. super(DatabaseSelect, self).__init__(parent, id, value, pos, size)
  471. self.SetName("DatabaseSelect")
  472. class TableSelect(wx.ComboBox):
  473. """!Creates combo box for selecting attribute tables from the database
  474. """
  475. def __init__(self, parent,
  476. id=wx.ID_ANY, value='', pos=wx.DefaultPosition,
  477. size=globalvar.DIALOG_COMBOBOX_SIZE,
  478. choices=[]):
  479. super(TableSelect, self).__init__(parent, id, value, pos, size, choices,
  480. style=wx.CB_READONLY)
  481. self.SetName("TableSelect")
  482. if not choices:
  483. self.InsertTables()
  484. def InsertTables(self, driver=None, database=None):
  485. """!Insert attribute tables into combobox"""
  486. items = []
  487. if not driver or not database:
  488. connect = grass.db_connection()
  489. driver = connect['driver']
  490. database = connect['database']
  491. ret = gcmd.RunCommand('db.tables',
  492. flags = 'p',
  493. read = True,
  494. driver = driver,
  495. database = database)
  496. if ret:
  497. for table in ret.splitlines():
  498. items.append(table)
  499. self.SetItems(items)
  500. self.SetValue('')
  501. class ColumnSelect(wx.ComboBox):
  502. """
  503. Creates combo box for selecting columns in the attribute table for a vector.
  504. The 'layer' terminology is likely to change for GRASS 7
  505. """
  506. def __init__(self, parent,
  507. id=wx.ID_ANY, value='', pos=wx.DefaultPosition,
  508. size=globalvar.DIALOG_COMBOBOX_SIZE, vector=None,
  509. layer=1, choices=[]):
  510. super(ColumnSelect, self).__init__(parent, id, value, pos, size, choices,
  511. style=wx.CB_READONLY)
  512. self.SetName("ColumnSelect")
  513. if vector:
  514. self.InsertColumns(vector, layer)
  515. def InsertColumns(self, vector, layer, excludeKey = False, type = None):
  516. """!Insert columns for a vector attribute table into the columns combobox
  517. @param vector vector name
  518. @param layer vector layer number
  519. @param excludeKey exclude key column from the list?
  520. @param type only columns of given type (given as list)
  521. """
  522. dbInfo = VectorDBInfo(vector)
  523. try:
  524. table = dbInfo.GetTable(int(layer))
  525. columnchoices = dbInfo.GetTableDesc(table)
  526. keyColumn = dbInfo.GetKeyColumn(int(layer))
  527. columns = len(columnchoices.keys()) * ['']
  528. for key, val in columnchoices.iteritems():
  529. columns[val['index']] = key
  530. if excludeKey: # exclude key column
  531. columns.remove(keyColumn)
  532. if type: # only selected column types
  533. for key, value in columnchoices.iteritems():
  534. if value['type'] not in type:
  535. columns.remove(key)
  536. except (KeyError, ValueError):
  537. columns = []
  538. self.SetItems(columns)
  539. self.SetValue('')
  540. def InsertTableColumns(self, table, driver=None, database=None):
  541. """!Insert table columns"""
  542. columns = []
  543. ret = gcmd.RunCommand('db.columns',
  544. read = True,
  545. driver = driver,
  546. database = database,
  547. table = table)
  548. if ret:
  549. columns = ret.splitlines()
  550. self.SetItems(columns)