gselect.py 20 KB

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