preferences.py 69 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804
  1. """
  2. @package preferences
  3. @brief User preferences dialog
  4. Sets default display font, etc.
  5. Classes:
  6. - PreferencesDialog
  7. - DefaultFontDialog
  8. - MapsetAccess
  9. (C) 2007-2008 by the GRASS Development Team
  10. This program is free software under the GNU General Public
  11. License (>=v2). Read the file COPYING that comes with GRASS
  12. for details.
  13. @author Michael Barton (Arizona State University)
  14. Martin Landa <landa.martin gmail.com>
  15. """
  16. import os
  17. import sys
  18. import copy
  19. import stat
  20. if os.name in ('posix', 'mac'):
  21. import pwd
  22. import wx
  23. import wx.lib.filebrowsebutton as filebrowse
  24. import wx.lib.colourselect as csel
  25. import wx.lib.mixins.listctrl as listmix
  26. from wx.lib.wordwrap import wordwrap
  27. import grass
  28. import gcmd
  29. import utils
  30. import globalvar
  31. from debug import Debug as Debug
  32. class Settings:
  33. """Generic class where to store settings"""
  34. def __init__(self):
  35. #
  36. # settings filename
  37. #
  38. self.fileName = ".grasswx7"
  39. self.filePath = None
  40. #
  41. # key/value separator
  42. #
  43. self.sep = ';'
  44. #
  45. # default settings
  46. #
  47. self.defaultSettings = {
  48. #
  49. # general
  50. #
  51. 'general': {
  52. # use default window layout (layer manager, displays, ...)
  53. 'defWindowPos' : {
  54. 'enabled' : False,
  55. 'dim' : ''
  56. },
  57. # expand/collapse element list
  58. 'elementListExpand' : {
  59. 'selection' : 0
  60. },
  61. },
  62. 'manager' : {
  63. # show opacity level widget
  64. 'changeOpacityLevel' : {
  65. 'enabled' : False
  66. },
  67. # ask when removing layer from layer tree
  68. 'askOnRemoveLayer' : {
  69. 'enabled' : True
  70. },
  71. },
  72. #
  73. # display
  74. #
  75. 'display': {
  76. 'font' : {
  77. 'type' : '',
  78. 'encoding': 'ISO-8859-1',
  79. },
  80. 'driver': {
  81. 'type': 'cairo'
  82. },
  83. 'compResolution' : {
  84. 'enabled' : False
  85. },
  86. 'autoRendering': {
  87. 'enabled' : False
  88. },
  89. 'statusbarMode': {
  90. 'selection' : 0
  91. },
  92. 'bgcolor': {
  93. 'color' : (255, 255, 255, 255),
  94. }
  95. },
  96. #
  97. # advanced
  98. #
  99. 'advanced' : {
  100. 'settingsFile' : {
  101. 'type' : 'home'
  102. }, # home, gisdbase, location, mapset
  103. 'iconTheme' : {
  104. 'type' : 'grass'
  105. }, # grass, silk
  106. },
  107. #
  108. # Attribute Table Manager
  109. #
  110. 'atm' : {
  111. 'highlight' : {
  112. 'color' : (255, 255, 0, 255),
  113. 'width' : 2
  114. },
  115. 'leftDbClick' : {
  116. 'selection' : 1 # draw selected
  117. },
  118. 'askOnDeleteRec' : {
  119. 'enabled' : True
  120. },
  121. 'keycolumn' : {
  122. 'value' : 'cat'
  123. },
  124. 'encoding' : {
  125. 'value' : '',
  126. }
  127. },
  128. #
  129. # Command
  130. #
  131. 'cmd': {
  132. 'overwrite' : {
  133. 'enabled' : False
  134. },
  135. 'closeDlg' : {
  136. 'enabled' : False
  137. },
  138. 'verbosity' : {
  139. 'selection' : 'grassenv'
  140. },
  141. # d.rast
  142. 'rasterOpaque' : {
  143. 'enabled' : False
  144. },
  145. # d.vect
  146. 'showType': {
  147. 'point' : {
  148. 'enabled' : True
  149. },
  150. 'line' : {
  151. 'enabled' : True
  152. },
  153. 'centroid' : {
  154. 'enabled' : True
  155. },
  156. 'boundary' : {
  157. 'enabled' : True
  158. },
  159. 'area' : {
  160. 'enabled' : True
  161. },
  162. 'face' : {
  163. 'enabled' : True
  164. },
  165. },
  166. 'addNewLayer' : {
  167. 'enabled' : False
  168. },
  169. },
  170. #
  171. # Workspace
  172. #
  173. 'workspace' : {
  174. 'posDisplay' : {
  175. 'enabled' : False
  176. },
  177. 'posManager' : {
  178. 'enabled' : False
  179. },
  180. },
  181. #
  182. # vdigit
  183. #
  184. 'vdigit' : {
  185. # symbology
  186. 'symbol' : {
  187. 'highlight' : {
  188. 'enabled' : None,
  189. 'color' : (255, 255, 0, 255)
  190. }, # yellow
  191. 'highlightDupl' : {
  192. 'enabled' : None,
  193. 'color' : (255, 72, 0, 255)
  194. }, # red
  195. 'point' : {
  196. 'enabled' : True,
  197. 'color' : (0, 0, 0, 255)
  198. }, # black
  199. 'line' : {
  200. 'enabled' : True,
  201. 'color' : (0, 0, 0, 255)
  202. }, # black
  203. 'boundaryNo' : {
  204. 'enabled' : True,
  205. 'color' : (126, 126, 126, 255)
  206. }, # grey
  207. 'boundaryOne' : {
  208. 'enabled' : True,
  209. 'color' : (0, 255, 0, 255)
  210. }, # green
  211. 'boundaryTwo' : {
  212. 'enabled' : True,
  213. 'color' : (255, 135, 0, 255)
  214. }, # orange
  215. 'centroidIn' : {
  216. 'enabled' : True,
  217. 'color' : (0, 0, 255, 255)
  218. }, # blue
  219. 'centroidOut' : {
  220. 'enabled' : True,
  221. 'color' : (165, 42, 42, 255)
  222. }, # brown
  223. 'centroidDup' : {
  224. 'enabled' : True,
  225. 'color' : (156, 62, 206, 255)
  226. }, # violet
  227. 'nodeOne' : {
  228. 'enabled' : True,
  229. 'color' : (255, 0, 0, 255)
  230. }, # red
  231. 'nodeTwo' : {
  232. 'enabled' : True,
  233. 'color' : (0, 86, 45, 255)
  234. }, # dark green
  235. 'vertex' : {
  236. 'enabled' : False,
  237. 'color' : (255, 20, 147, 255)
  238. }, # deep pink
  239. 'area' : {
  240. 'enabled' : False,
  241. 'color' : (217, 255, 217, 255)
  242. }, # green
  243. 'direction' : {
  244. 'enabled' : False,
  245. 'color' : (255, 0, 0, 255)
  246. }, # red
  247. },
  248. # display
  249. 'lineWidth' : {
  250. 'value' : 2,
  251. 'units' : 'screen pixels'
  252. },
  253. # snapping
  254. 'snapping' : {
  255. 'value' : 10,
  256. 'units' : 'screen pixels'
  257. },
  258. 'snapToVertex' : {
  259. 'enabled' : False
  260. },
  261. # digitize new record
  262. 'addRecord' : {
  263. 'enabled' : True
  264. },
  265. 'layer' :{
  266. 'value' : 1
  267. },
  268. 'category' : {
  269. 'value' : 1
  270. },
  271. 'categoryMode' : {
  272. 'selection' : 0
  273. },
  274. # delete existing feature(s)
  275. 'delRecord' : {
  276. 'enabled' : True
  277. },
  278. # query tool
  279. 'query' : {
  280. 'selection' : 0,
  281. 'box' : True
  282. },
  283. 'queryLength' : {
  284. 'than-selection' : 0,
  285. 'thresh' : 0
  286. },
  287. 'queryDangle' : {
  288. 'than-selection' : 0,
  289. 'thresh' : 0
  290. },
  291. # select feature (point, line, centroid, boundary)
  292. 'selectType': {
  293. 'point' : {
  294. 'enabled' : True
  295. },
  296. 'line' : {
  297. 'enabled' : True
  298. },
  299. 'centroid' : {
  300. 'enabled' : True
  301. },
  302. 'boundary' : {
  303. 'enabled' : True
  304. },
  305. },
  306. 'selectThresh' : {
  307. 'value' : 10,
  308. 'units' : 'screen pixels'
  309. },
  310. 'checkForDupl' : {
  311. 'enabled' : False
  312. },
  313. 'selectInside' : {
  314. 'enabled' : False
  315. },
  316. # exit
  317. 'saveOnExit' : {
  318. 'enabled' : False,
  319. },
  320. # break lines on intersection
  321. 'breakLines' : {
  322. 'enabled' : False,
  323. },
  324. },
  325. 'profile': {
  326. 'raster0' : {
  327. 'pcolor' : (0, 0, 255, 255), # profile line color
  328. 'pwidth' : 1, # profile line width
  329. 'pstyle' : 'solid', # profile line pen style
  330. },
  331. 'raster1' : {
  332. 'pcolor' : (255, 0, 0, 255),
  333. 'pwidth' : 1,
  334. 'pstyle' : 'solid',
  335. },
  336. 'raster2' : {
  337. 'pcolor' : (0, 255, 0, 255),
  338. 'pwidth' : 1,
  339. 'pstyle' : 'solid',
  340. },
  341. 'font' : {
  342. 'titleSize' : 12,
  343. 'axisSize' : 11,
  344. 'legendSize' : 10,
  345. },
  346. 'marker' : {
  347. 'color' : (0, 0, 0, 255),
  348. 'fill' : 'transparent',
  349. 'size' : 2,
  350. 'type' : 'triangle',
  351. 'legend' : _('Segment break'),
  352. },
  353. 'grid' : {
  354. 'color' : (200, 200, 200, 255),
  355. 'enabled' : True,
  356. },
  357. 'x-axis' : {
  358. 'type' : 'auto', # axis format
  359. 'min' : 0, # axis min for custom axis range
  360. 'max': 0, # axis max for custom axis range
  361. 'log' : False,
  362. },
  363. 'y-axis' : {
  364. 'type' : 'auto', # axis format
  365. 'min' : 0, # axis min for custom axis range
  366. 'max': 0, # axis max for custom axis range
  367. 'log' : False,
  368. },
  369. 'legend' : {
  370. 'enabled' : True
  371. },
  372. },
  373. 'georect' : {
  374. 'symbol' : {
  375. 'color' : (0, 0, 255, 255),
  376. 'width' : 2,
  377. },
  378. },
  379. 'nviz' : {
  380. 'view' : {
  381. 'persp' : {
  382. 'value' : 40,
  383. 'step' : 5,
  384. },
  385. 'pos' : {
  386. 'x' : 0.85,
  387. 'y' : 0.85,
  388. },
  389. 'height' : {
  390. 'step' : 100,
  391. },
  392. 'twist' : {
  393. 'value' : 0,
  394. 'step' : 5,
  395. },
  396. 'z-exag' : {
  397. 'value': 1,
  398. 'step' : 1,
  399. },
  400. },
  401. 'surface' : {
  402. 'shine': {
  403. 'map' : False,
  404. 'value' : 60.0,
  405. },
  406. 'color' : {
  407. 'map' : True,
  408. 'value' : (0, 0, 0, 255), # constant: black
  409. },
  410. 'draw' : {
  411. 'wire-color' : (136, 136, 136, 255),
  412. 'mode' : 1, # fine
  413. 'style' : 1, # surface
  414. 'shading' : 1, # gouraud
  415. 'res-fine' : 6,
  416. 'res-coarse' : 9,
  417. },
  418. 'position' : {
  419. 'x' : 0,
  420. 'y' : 0,
  421. 'z' : 0,
  422. },
  423. },
  424. 'vector' : {
  425. 'lines' : {
  426. 'show' : False,
  427. 'width' : 2,
  428. 'color' : (0, 0, 255, 255), # blue
  429. 'flat' : False,
  430. 'height' : 0,
  431. },
  432. 'points' : {
  433. 'show' : False,
  434. 'size' : 100,
  435. 'width' : 2,
  436. 'marker' : 2,
  437. 'color' : (0, 0, 255, 255), # blue
  438. 'height' : 0,
  439. }
  440. },
  441. 'volume' : {
  442. 'color' : {
  443. 'map' : True,
  444. 'value' : (0, 0, 0, 255), # constant: black
  445. },
  446. 'draw' : {
  447. 'mode' : 0, # isosurfaces
  448. 'shading' : 1, # gouraud
  449. 'resolution' : 3, # polygon resolution
  450. },
  451. 'shine': {
  452. 'map' : False,
  453. 'value' : 60.0,
  454. },
  455. },
  456. 'settings': {
  457. 'general' : {
  458. 'bgcolor' : (255, 255, 255, 255), # white
  459. },
  460. },
  461. },
  462. }
  463. #
  464. # user settings
  465. #
  466. self.userSettings = copy.deepcopy(self.defaultSettings)
  467. try:
  468. self.ReadSettingsFile()
  469. except gcmd.SettingsError, e:
  470. print >> sys.stderr, e.message
  471. #
  472. # internal settings (based on user settings)
  473. #
  474. self.internalSettings = {}
  475. for group in self.userSettings.keys():
  476. self.internalSettings[group] = {}
  477. for key in self.userSettings[group].keys():
  478. self.internalSettings[group][key] = {}
  479. # self.internalSettings['general']["mapsetPath"]['value'] = self.GetMapsetPath()
  480. self.internalSettings['general']['elementListExpand']['choices'] = (_("Collapse all except PERMANENT and current"),
  481. _("Collapse all except PERMANENT"),
  482. _("Collapse all"),
  483. _("Expand all"))
  484. self.internalSettings['atm']['leftDbClick']['choices'] = (_('Edit selected record'),
  485. _('Display selected'))
  486. self.internalSettings['advanced']['settingsFile']['choices'] = ('home',
  487. 'gisdbase',
  488. 'location',
  489. 'mapset')
  490. self.internalSettings['advanced']['iconTheme']['choices'] = ('grass',
  491. 'silk')
  492. self.internalSettings['cmd']['verbosity']['choices'] = ('grassenv',
  493. 'verbose',
  494. 'quiet')
  495. self.internalSettings['display']['driver']['choices'] = ['cairo', 'png']
  496. self.internalSettings['display']['statusbarMode']['choices'] = globalvar.MAP_DISPLAY_STATUSBAR_MODE
  497. self.internalSettings['nviz']['view'] = {}
  498. self.internalSettings['nviz']['view']['twist'] = {}
  499. self.internalSettings['nviz']['view']['twist']['min'] = -180
  500. self.internalSettings['nviz']['view']['twist']['max'] = 180
  501. self.internalSettings['nviz']['view']['persp'] = {}
  502. self.internalSettings['nviz']['view']['persp']['min'] = 1
  503. self.internalSettings['nviz']['view']['persp']['max'] = 100
  504. self.internalSettings['nviz']['view']['height'] = {}
  505. self.internalSettings['nviz']['view']['height']['value'] = -1
  506. self.internalSettings['nviz']['vector'] = {}
  507. self.internalSettings['nviz']['vector']['points'] = {}
  508. self.internalSettings['nviz']['vector']['points']['marker'] = ("x",
  509. _("box"),
  510. _("sphere"),
  511. _("cube"),
  512. _("diamond"),
  513. _("dtree"),
  514. _("ctree"),
  515. _("aster"),
  516. _("gyro"),
  517. _("histogram"))
  518. self.internalSettings['vdigit']['bgmap'] = {}
  519. self.internalSettings['vdigit']['bgmap']['value'] = ''
  520. def ReadSettingsFile(self, settings=None):
  521. """Reads settings file (mapset, location, gisdbase)"""
  522. if settings is None:
  523. settings = self.userSettings
  524. # look for settings file
  525. # -> mapser
  526. # -> location
  527. # -> gisdbase
  528. gisenv = grass.gisenv()
  529. gisdbase = gisenv['GISDBASE']
  530. location_name = gisenv['LOCATION_NAME']
  531. mapset_name = gisenv['MAPSET']
  532. mapset_file = os.path.join(gisdbase, location_name, mapset_name, self.fileName)
  533. location_file = os.path.join(gisdbase, location_name, self.fileName)
  534. gisdbase_file = os.path.join(gisdbase, self.fileName)
  535. home_file = os.path.join(os.path.expanduser("~"), self.fileName) # MS Windows fix ?
  536. if os.path.isfile(mapset_file):
  537. self.filePath = mapset_file
  538. elif os.path.isfile(location_file):
  539. self.filePath = location_file
  540. elif os.path.isfile(gisdbase_file):
  541. self.filePath = gisdbase_file
  542. elif os.path.isfile(home_file):
  543. self.filePath = home_file
  544. if self.filePath:
  545. self.__ReadFile(self.filePath, settings)
  546. # set environment variables
  547. os.environ["GRASS_FONT"] = self.Get(group='display',
  548. key='font', subkey='type')
  549. os.environ["GRASS_ENCODING"] = self.Get(group='display',
  550. key='font', subkey='encoding')
  551. def __ReadFile(self, filename, settings=None):
  552. """Read settings from file to dict"""
  553. if settings is None:
  554. settings = self.userSettings
  555. try:
  556. file = open(filename, "r")
  557. line = ''
  558. for line in file.readlines():
  559. line = line.rstrip('%s' % os.linesep)
  560. group, key = line.split(self.sep)[0:2]
  561. kv = line.split(self.sep)[2:]
  562. subkeyMaster = None
  563. if len(kv) % 2 != 0: # multiple (e.g. nviz)
  564. subkeyMaster = kv[0]
  565. del kv[0]
  566. idx = 0
  567. while idx < len(kv):
  568. if subkeyMaster:
  569. subkey = [subkeyMaster, kv[idx]]
  570. else:
  571. subkey = kv[idx]
  572. value = kv[idx+1]
  573. value = self.__parseValue(value, read=True)
  574. self.Append(settings, group, key, subkey, value)
  575. idx += 2
  576. except ValueError, e:
  577. print >> sys.stderr, _("Error: Reading settings from file <%(file)s> failed.\n"
  578. " Details: %(detail)s\n"
  579. " Line: '%(line)s'") % { 'file' : filename,
  580. 'detail' : e,
  581. 'line' : line }
  582. file.close()
  583. file.close()
  584. def SaveToFile(self, settings=None):
  585. """Save settings to the file"""
  586. if settings is None:
  587. settings = self.userSettings
  588. loc = self.Get(group='advanced', key='settingsFile', subkey='type')
  589. home = os.path.expanduser("~") # MS Windows fix ?
  590. gisenv = grass.gisenv()
  591. gisdbase = gisenv['GISDBASE']
  592. location_name = gisenv['LOCATION_NAME']
  593. mapset_name = gisenv['MAPSET']
  594. filePath = None
  595. if loc == 'home':
  596. filePath = os.path.join(home, self.fileName)
  597. elif loc == 'gisdbase':
  598. filePath = os.path.join(gisdbase, self.fileName)
  599. elif loc == 'location':
  600. filePath = os.path.join(gisdbase, location_name, self.fileName)
  601. elif loc == 'mapset':
  602. filePath = os.path.join(gisdbase, location_name, mapset_name, self.fileName)
  603. if filePath is None:
  604. raise gcmd.SettingsError(_('Uknown settings file location.'))
  605. try:
  606. file = open(filePath, "w")
  607. for group in settings.keys():
  608. for key in settings[group].keys():
  609. file.write('%s%s%s%s' % (group, self.sep, key, self.sep))
  610. subkeys = settings[group][key].keys()
  611. for idx in range(len(subkeys)):
  612. value = settings[group][key][subkeys[idx]]
  613. if type(value) == type({}):
  614. if idx > 0:
  615. file.write('%s%s%s%s%s' % (os.linesep, group, self.sep, key, self.sep))
  616. file.write('%s%s' % (subkeys[idx], self.sep))
  617. kvalues = settings[group][key][subkeys[idx]].keys()
  618. srange = range(len(kvalues))
  619. for sidx in srange:
  620. svalue = self.__parseValue(settings[group][key][subkeys[idx]][kvalues[sidx]])
  621. file.write('%s%s%s' % (kvalues[sidx], self.sep,
  622. svalue))
  623. if sidx < len(kvalues) - 1:
  624. file.write('%s' % self.sep)
  625. else:
  626. value = self.__parseValue(settings[group][key][subkeys[idx]])
  627. file.write('%s%s%s' % (subkeys[idx], self.sep, value))
  628. if idx < len(subkeys) - 1:
  629. file.write('%s' % self.sep)
  630. file.write('%s' % os.linesep)
  631. except IOError, e:
  632. raise gcmd.SettingsError(message=e)
  633. except StandardError, e:
  634. raise gcmd.SettingsError(message=_('Writing settings to file <%(file)s> failed.'
  635. '\n\nDetails: %(detail)s') % { 'file' : filePath,
  636. 'detail' : e })
  637. file.close()
  638. return filePath
  639. def __parseValue(self, value, read=False):
  640. """Parse value to be store in settings file"""
  641. if read: # -> read settings (cast values)
  642. if value == 'True':
  643. value = True
  644. elif value == 'False':
  645. value = False
  646. elif value == 'None':
  647. value = None
  648. elif ':' in value: # -> color
  649. value = tuple(map(int, value.split(':')))
  650. else:
  651. try:
  652. value = int(value)
  653. except ValueError:
  654. try:
  655. value = float(value)
  656. except ValueError:
  657. pass
  658. else: # -> write settings
  659. if type(value) == type(()): # -> color
  660. value = str(value[0]) + ':' +\
  661. str(value[1]) + ':' + \
  662. str(value[2])
  663. return value
  664. def Get(self, group, key=None, subkey=None, internal=False):
  665. """Get value by key/subkey
  666. Raise KeyError if key is not found
  667. @param group settings group
  668. @param key (value, None)
  669. @param subkey (value, list or None)
  670. @param internal use internal settings instead
  671. @return value
  672. """
  673. if internal is True:
  674. settings = self.internalSettings
  675. else:
  676. settings = self.userSettings
  677. try:
  678. if subkey is None:
  679. if key is None:
  680. return settings[group]
  681. else:
  682. return settings[group][key]
  683. else:
  684. if type(subkey) == type([]) or \
  685. type(subkey) == type(()):
  686. return settings[group][key][subkey[0]][subkey[1]]
  687. else:
  688. return settings[group][key][subkey]
  689. except KeyError:
  690. #raise gcmd.SettingsError("%s %s:%s:%s." % (_("Unable to get value"),
  691. # group, key, subkey))
  692. print >> sys.stderr, "Settings: unable to get value '%s:%s:%s'\n" % \
  693. (group, key, subkey)
  694. def Set(self, group, value, key=None, subkey=None, internal=False):
  695. """Set value of key/subkey
  696. Raise KeyError if group/key is not found
  697. @param group settings group
  698. @param key key (value, None)
  699. @param subkey subkey (value, list or None)
  700. @param value value
  701. @param internal use internal settings instead
  702. """
  703. if internal is True:
  704. settings = self.internalSettings
  705. else:
  706. settings = self.userSettings
  707. try:
  708. if subkey is None:
  709. if key is None:
  710. settings[group] = value
  711. else:
  712. settings[group][key] = value
  713. else:
  714. if type(subkey) == type([]):
  715. settings[group][key][subkey[0]][subkey[1]] = value
  716. else:
  717. settings[group][key][subkey] = value
  718. except KeyError:
  719. raise gcmd.SettingsError("%s '%s:%s:%s'" % (_("Unable to set "), group, key, subkey))
  720. def Append(self, dict, group, key, subkey, value):
  721. """Set value of key/subkey
  722. Create group/key/subkey if not exists
  723. @param dict settings dictionary to use
  724. @param group settings group
  725. @param key key
  726. @param subkey subkey (value or list)
  727. @param value value
  728. """
  729. if not dict.has_key(group):
  730. dict[group] = {}
  731. if not dict[group].has_key(key):
  732. dict[group][key] = {}
  733. if type(subkey) == type([]):
  734. # TODO: len(subkey) > 2
  735. if not dict[group][key].has_key(subkey[0]):
  736. dict[group][key][subkey[0]] = {}
  737. dict[group][key][subkey[0]][subkey[1]] = value
  738. else:
  739. dict[group][key][subkey] = value
  740. def GetDefaultSettings(self):
  741. """Get default user settings"""
  742. return self.defaultSettings
  743. globalSettings = Settings()
  744. class PreferencesDialog(wx.Dialog):
  745. """User preferences dialog"""
  746. def __init__(self, parent, title=_("User GUI settings"),
  747. settings=globalSettings,
  748. style=wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER):
  749. self.parent = parent # GMFrame
  750. self.title = title
  751. wx.Dialog.__init__(self, parent=parent, id=wx.ID_ANY, title=title,
  752. style=style, size=(-1, -1))
  753. self.settings = settings
  754. # notebook
  755. notebook = wx.Notebook(parent=self, id=wx.ID_ANY, style=wx.BK_DEFAULT)
  756. # dict for window ids
  757. self.winId = {}
  758. # create notebook pages
  759. self.__CreateGeneralPage(notebook)
  760. self.__CreateDisplayPage(notebook)
  761. self.__CreateCmdPage(notebook)
  762. self.__CreateAttributeManagerPage(notebook)
  763. self.__CreateWorkspacePage(notebook)
  764. self.__CreateAdvancedPage(notebook)
  765. # buttons
  766. btnDefault = wx.Button(self, wx.ID_ANY, _("Set to default"))
  767. btnSave = wx.Button(self, wx.ID_SAVE)
  768. btnApply = wx.Button(self, wx.ID_APPLY)
  769. btnCancel = wx.Button(self, wx.ID_CANCEL)
  770. btnSave.SetDefault()
  771. # bindigs
  772. btnDefault.Bind(wx.EVT_BUTTON, self.OnDefault)
  773. btnDefault.SetToolTipString(_("Revert settings to default and apply changes"))
  774. btnApply.Bind(wx.EVT_BUTTON, self.OnApply)
  775. btnApply.SetToolTipString(_("Apply changes for the current session"))
  776. btnSave.Bind(wx.EVT_BUTTON, self.OnSave)
  777. btnSave.SetToolTipString(_("Apply and save changes to user settings file (default for next sessions)"))
  778. btnSave.SetDefault()
  779. btnCancel.Bind(wx.EVT_BUTTON, self.OnCancel)
  780. btnCancel.SetToolTipString(_("Close dialog and ignore changes"))
  781. # sizers
  782. btnSizer = wx.BoxSizer(wx.HORIZONTAL)
  783. btnSizer.Add(item=btnDefault, proportion=1,
  784. flag=wx.ALL, border=5)
  785. btnStdSizer = wx.StdDialogButtonSizer()
  786. btnStdSizer.AddButton(btnCancel)
  787. btnStdSizer.AddButton(btnSave)
  788. btnStdSizer.AddButton(btnApply)
  789. btnStdSizer.Realize()
  790. mainSizer = wx.BoxSizer(wx.VERTICAL)
  791. mainSizer.Add(item=notebook, proportion=1, flag=wx.EXPAND | wx.ALL, border=5)
  792. mainSizer.Add(item=btnSizer, proportion=0,
  793. flag=wx.EXPAND, border=0)
  794. mainSizer.Add(item=btnStdSizer, proportion=0,
  795. flag=wx.EXPAND | wx.ALL | wx.ALIGN_RIGHT, border=5)
  796. self.SetSizer(mainSizer)
  797. mainSizer.Fit(self)
  798. self.SetMinSize(self.GetBestSize())
  799. self.SetSize((500, 375))
  800. def __CreateGeneralPage(self, notebook):
  801. """Create notebook page for general settings"""
  802. panel = wx.Panel(parent=notebook, id=wx.ID_ANY)
  803. notebook.AddPage(page=panel, text=_("General"))
  804. border = wx.BoxSizer(wx.VERTICAL)
  805. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("General settings"))
  806. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  807. gridSizer = wx.GridBagSizer (hgap=3, vgap=3)
  808. gridSizer.AddGrowableCol(0)
  809. #
  810. # expand element list
  811. #
  812. row = 0
  813. gridSizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY,
  814. label=_("Element list:")),
  815. flag=wx.ALIGN_LEFT |
  816. wx.ALIGN_CENTER_VERTICAL,
  817. pos=(row, 0))
  818. elementList = wx.Choice(parent=panel, id=wx.ID_ANY,
  819. choices=self.settings.Get(group='general', key='elementListExpand',
  820. subkey='choices', internal=True),
  821. name="GetSelection")
  822. elementList.SetSelection(self.settings.Get(group='general', key='elementListExpand',
  823. subkey='selection'))
  824. self.winId['general:elementListExpand:selection'] = elementList.GetId()
  825. gridSizer.Add(item=elementList,
  826. flag=wx.ALIGN_RIGHT |
  827. wx.ALIGN_CENTER_VERTICAL,
  828. pos=(row, 1))
  829. #
  830. # default window layout
  831. #
  832. row += 1
  833. defaultPos = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  834. label=_("Save current window layout as default"),
  835. name='IsChecked')
  836. defaultPos.SetValue(self.settings.Get(group='general', key='defWindowPos', subkey='enabled'))
  837. defaultPos.SetToolTip(wx.ToolTip (_("Save current position and size of Layer Manager window and opened "
  838. "Map Display window(s) and use as default for next sessions.")))
  839. self.winId['general:defWindowPos:enabled'] = defaultPos.GetId()
  840. gridSizer.Add(item=defaultPos,
  841. pos=(row, 0), span=(1, 2))
  842. sizer.Add(item=gridSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=5)
  843. border.Add(item=sizer, proportion=0, flag=wx.ALL | wx.EXPAND, border=3)
  844. #
  845. # Layer Manager settings
  846. #
  847. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Layer Manager settings"))
  848. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  849. gridSizer = wx.GridBagSizer (hgap=3, vgap=3)
  850. gridSizer.AddGrowableCol(0)
  851. #
  852. # ask when removing map layer from layer tree
  853. #
  854. row = 0
  855. askOnRemoveLayer = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  856. label=_("Ask when removing map layer from layer tree"),
  857. name='IsChecked')
  858. askOnRemoveLayer.SetValue(self.settings.Get(group='manager', key='askOnRemoveLayer', subkey='enabled'))
  859. self.winId['manager:askOnRemoveLayer:enabled'] = askOnRemoveLayer.GetId()
  860. gridSizer.Add(item=askOnRemoveLayer,
  861. pos=(row, 0), span=(1, 2))
  862. sizer.Add(item=gridSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=5)
  863. border.Add(item=sizer, proportion=0, flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border=3)
  864. panel.SetSizer(border)
  865. return panel
  866. def __CreateDisplayPage(self, notebook):
  867. """Create notebook page for display settings"""
  868. panel = wx.Panel(parent=notebook, id=wx.ID_ANY)
  869. notebook.AddPage(page=panel, text=_("Display"))
  870. border = wx.BoxSizer(wx.VERTICAL)
  871. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Font settings"))
  872. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  873. gridSizer = wx.GridBagSizer (hgap=3, vgap=3)
  874. gridSizer.AddGrowableCol(0)
  875. #
  876. # font settings
  877. #
  878. row = 0
  879. gridSizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY,
  880. label=_("Default font for GRASS displays:")),
  881. flag=wx.ALIGN_LEFT |
  882. wx.ALIGN_CENTER_VERTICAL,
  883. pos=(row, 0))
  884. fontButton = wx.Button(parent=panel, id=wx.ID_ANY,
  885. label=_("Set font"), size=(100, -1))
  886. gridSizer.Add(item=fontButton,
  887. flag=wx.ALIGN_RIGHT |
  888. wx.ALIGN_CENTER_VERTICAL,
  889. pos=(row, 1))
  890. sizer.Add(item=gridSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=5)
  891. border.Add(item=sizer, proportion=0, flag=wx.ALL | wx.EXPAND, border=3)
  892. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Default display settings"))
  893. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  894. gridSizer = wx.GridBagSizer (hgap=3, vgap=3)
  895. gridSizer.AddGrowableCol(0)
  896. #
  897. # display driver
  898. #
  899. row = 0
  900. gridSizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY,
  901. label=_("Display driver:")),
  902. flag=wx.ALIGN_LEFT |
  903. wx.ALIGN_CENTER_VERTICAL,
  904. pos=(row, 0))
  905. listOfDrivers = self.settings.Get(group='display', key='driver', subkey='choices', internal=True)
  906. driver = wx.Choice(parent=panel, id=wx.ID_ANY, size=(150, -1),
  907. choices=listOfDrivers,
  908. name="GetStringSelection")
  909. driver.SetStringSelection(self.settings.Get(group='display', key='driver', subkey='type'))
  910. self.winId['display:driver:type'] = driver.GetId()
  911. gridSizer.Add(item=driver,
  912. flag=wx.ALIGN_RIGHT,
  913. pos=(row, 1))
  914. #
  915. # Statusbar mode
  916. #
  917. row += 1
  918. gridSizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY,
  919. label=_("Statusbar mode:")),
  920. flag=wx.ALIGN_LEFT |
  921. wx.ALIGN_CENTER_VERTICAL,
  922. pos=(row, 0))
  923. listOfModes = self.settings.Get(group='display', key='statusbarMode', subkey='choices', internal=True)
  924. statusbarMode = wx.Choice(parent=panel, id=wx.ID_ANY, size=(150, -1),
  925. choices=listOfModes,
  926. name="GetSelection")
  927. statusbarMode.SetSelection(self.settings.Get(group='display', key='statusbarMode', subkey='selection'))
  928. self.winId['display:statusbarMode:selection'] = statusbarMode.GetId()
  929. gridSizer.Add(item=statusbarMode,
  930. flag=wx.ALIGN_RIGHT,
  931. pos=(row, 1))
  932. #
  933. # Background color
  934. #
  935. row += 1
  936. gridSizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY,
  937. label=_("Background color:")),
  938. flag=wx.ALIGN_LEFT |
  939. wx.ALIGN_CENTER_VERTICAL,
  940. pos=(row, 0))
  941. bgColor = csel.ColourSelect(parent=panel, id=wx.ID_ANY,
  942. colour=self.settings.Get(group='display', key='bgcolor', subkey='color'),
  943. size=(35, 35))
  944. bgColor.SetName('GetColour')
  945. self.winId['display:bgcolor:color'] = bgColor.GetId()
  946. gridSizer.Add(item=bgColor,
  947. flag=wx.ALIGN_RIGHT,
  948. pos=(row, 1))
  949. #
  950. # Use computation resolution
  951. #
  952. row += 1
  953. compResolution = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  954. label=_("Constrain display resolution to computational settings"),
  955. name="IsChecked")
  956. compResolution.SetValue(self.settings.Get(group='display', key='compResolution', subkey='enabled'))
  957. self.winId['display:compResolution:enabled'] = compResolution.GetId()
  958. gridSizer.Add(item=compResolution,
  959. pos=(row, 0), span=(1, 2))
  960. #
  961. # auto-rendering
  962. #
  963. row += 1
  964. autoRendering = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  965. label=_("Enable auto-rendering"),
  966. name="IsChecked")
  967. autoRendering.SetValue(self.settings.Get(group='display', key='autoRendering', subkey='enabled'))
  968. self.winId['display:autoRendering:enabled'] = autoRendering.GetId()
  969. gridSizer.Add(item=autoRendering,
  970. pos=(row, 0), span=(1, 2))
  971. sizer.Add(item=gridSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=5)
  972. border.Add(item=sizer, proportion=0, flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border=3)
  973. panel.SetSizer(border)
  974. # bindings
  975. fontButton.Bind(wx.EVT_BUTTON, self.OnSetFont)
  976. return panel
  977. def __CreateCmdPage(self, notebook):
  978. """Create notebook page for commad dialog settings"""
  979. panel = wx.Panel(parent=notebook, id=wx.ID_ANY)
  980. notebook.AddPage(page=panel, text=_("Command"))
  981. border = wx.BoxSizer(wx.VERTICAL)
  982. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Command dialog settings"))
  983. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  984. gridSizer = wx.GridBagSizer (hgap=3, vgap=3)
  985. gridSizer.AddGrowableCol(0)
  986. #
  987. # command dialog settings
  988. #
  989. row = 0
  990. # overwrite
  991. overwrite = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  992. label=_("Allow output files to overwrite existing files"),
  993. name="IsChecked")
  994. overwrite.SetValue(self.settings.Get(group='cmd', key='overwrite', subkey='enabled'))
  995. self.winId['cmd:overwrite:enabled'] = overwrite.GetId()
  996. gridSizer.Add(item=overwrite,
  997. pos=(row, 0), span=(1, 2))
  998. row += 1
  999. # close
  1000. close = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  1001. label=_("Close dialog on finish"),
  1002. name="IsChecked")
  1003. close.SetValue(self.settings.Get(group='cmd', key='closeDlg', subkey='enabled'))
  1004. self.winId['cmd:closeDlg:enabled'] = close.GetId()
  1005. gridSizer.Add(item=close,
  1006. pos=(row, 0), span=(1, 2))
  1007. row += 1
  1008. # add layer
  1009. add = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  1010. label=_("Add created map into layer tree"),
  1011. name="IsChecked")
  1012. add.SetValue(self.settings.Get(group='cmd', key='addNewLayer', subkey='enabled'))
  1013. self.winId['cmd:addNewLayer:enabled'] = add.GetId()
  1014. gridSizer.Add(item=add,
  1015. pos=(row, 0), span=(1, 2))
  1016. row += 1
  1017. # verbosity
  1018. gridSizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY,
  1019. label=_("Verbosity level:")),
  1020. flag=wx.ALIGN_LEFT |
  1021. wx.ALIGN_CENTER_VERTICAL,
  1022. pos=(row, 0))
  1023. verbosity = wx.Choice(parent=panel, id=wx.ID_ANY, size=(200, -1),
  1024. choices=self.settings.Get(group='cmd', key='verbosity', subkey='choices', internal=True),
  1025. name="GetStringSelection")
  1026. verbosity.SetStringSelection(self.settings.Get(group='cmd', key='verbosity', subkey='selection'))
  1027. self.winId['cmd:verbosity:selection'] = verbosity.GetId()
  1028. gridSizer.Add(item=verbosity,
  1029. pos=(row, 1))
  1030. sizer.Add(item=gridSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=5)
  1031. border.Add(item=sizer, proportion=0, flag=wx.ALL | wx.EXPAND, border=3)
  1032. #
  1033. # raster settings
  1034. #
  1035. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Raster settings"))
  1036. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1037. gridSizer = wx.GridBagSizer (hgap=3, vgap=3)
  1038. gridSizer.AddGrowableCol(0)
  1039. #
  1040. # raster overlay
  1041. #
  1042. row = 0
  1043. rasterOpaque = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  1044. label=_("Make null cells opaque"),
  1045. name='IsChecked')
  1046. rasterOpaque.SetValue(self.settings.Get(group='cmd', key='rasterOpaque', subkey='enabled'))
  1047. self.winId['cmd:rasterOpaque:enabled'] = rasterOpaque.GetId()
  1048. gridSizer.Add(item=rasterOpaque,
  1049. pos=(row, 0), span=(1, 2))
  1050. sizer.Add(item=gridSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=5)
  1051. border.Add(item=sizer, proportion=0, flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border=3)
  1052. #
  1053. # vector settings
  1054. #
  1055. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Vector settings"))
  1056. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1057. gridSizer = wx.FlexGridSizer (cols=7, hgap=3, vgap=3)
  1058. gridSizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY,
  1059. label=_("Display:")),
  1060. flag=wx.ALIGN_CENTER_VERTICAL)
  1061. for type in ('point', 'line', 'centroid', 'boundary',
  1062. 'area', 'face'):
  1063. chkbox = wx.CheckBox(parent=panel, label=type)
  1064. checked = self.settings.Get(group='cmd', key='showType',
  1065. subkey=[type, 'enabled'])
  1066. chkbox.SetValue(checked)
  1067. self.winId['cmd:showType:%s:enabled' % type] = chkbox.GetId()
  1068. gridSizer.Add(item=chkbox)
  1069. sizer.Add(item=gridSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=5)
  1070. border.Add(item=sizer, proportion=0, flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND, border=3)
  1071. panel.SetSizer(border)
  1072. return panel
  1073. def __CreateAttributeManagerPage(self, notebook):
  1074. """Create notebook page for 'Attribute Table Manager' settings"""
  1075. panel = wx.Panel(parent=notebook, id=wx.ID_ANY)
  1076. notebook.AddPage(page=panel, text=_("Attributes"))
  1077. pageSizer = wx.BoxSizer(wx.VERTICAL)
  1078. #
  1079. # highlighting
  1080. #
  1081. highlightBox = wx.StaticBox(parent=panel, id=wx.ID_ANY,
  1082. label=" %s " % _("Highlighting"))
  1083. highlightSizer = wx.StaticBoxSizer(highlightBox, wx.VERTICAL)
  1084. flexSizer = wx.FlexGridSizer (cols=2, hgap=5, vgap=5)
  1085. flexSizer.AddGrowableCol(0)
  1086. label = wx.StaticText(parent=panel, id=wx.ID_ANY, label="Color")
  1087. hlColor = csel.ColourSelect(parent=panel, id=wx.ID_ANY,
  1088. colour=self.settings.Get(group='atm', key='highlight', subkey='color'),
  1089. size=(35, 35))
  1090. hlColor.SetName('GetColour')
  1091. self.winId['atm:highlight:color'] = hlColor.GetId()
  1092. flexSizer.Add(label, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1093. flexSizer.Add(hlColor, proportion=0, flag=wx.ALIGN_RIGHT | wx.FIXED_MINSIZE)
  1094. label = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Line width (in pixels)"))
  1095. hlWidth = wx.SpinCtrl(parent=panel, id=wx.ID_ANY, size=(50, -1),
  1096. initial=self.settings.Get(group='atm', key='highlight',subkey='width'),
  1097. min=1, max=1e6)
  1098. self.winId['atm:highlight:width'] = hlWidth.GetId()
  1099. flexSizer.Add(label, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1100. flexSizer.Add(hlWidth, proportion=0, flag=wx.ALIGN_RIGHT | wx.FIXED_MINSIZE)
  1101. highlightSizer.Add(item=flexSizer,
  1102. proportion=0,
  1103. flag=wx.ALL | wx.EXPAND,
  1104. border=5)
  1105. pageSizer.Add(item=highlightSizer,
  1106. proportion=0,
  1107. flag=wx.ALL | wx.EXPAND,
  1108. border=5)
  1109. #
  1110. # data browser related settings
  1111. #
  1112. dataBrowserBox = wx.StaticBox(parent=panel, id=wx.ID_ANY,
  1113. label=" %s " % _("Data browser"))
  1114. dataBrowserSizer = wx.StaticBoxSizer(dataBrowserBox, wx.VERTICAL)
  1115. flexSizer = wx.FlexGridSizer (cols=2, hgap=5, vgap=5)
  1116. flexSizer.AddGrowableCol(0)
  1117. label = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Left mouse double click"))
  1118. leftDbClick = wx.Choice(parent=panel, id=wx.ID_ANY,
  1119. choices=self.settings.Get(group='atm', key='leftDbClick', subkey='choices', internal=True),
  1120. name="GetSelection")
  1121. leftDbClick.SetSelection(self.settings.Get(group='atm', key='leftDbClick', subkey='selection'))
  1122. self.winId['atm:leftDbClick:selection'] = leftDbClick.GetId()
  1123. flexSizer.Add(label, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1124. flexSizer.Add(leftDbClick, proportion=0, flag=wx.ALIGN_RIGHT | wx.FIXED_MINSIZE)
  1125. # encoding
  1126. label = wx.StaticText(parent=panel, id=wx.ID_ANY, label=_("Encoding"))
  1127. encoding = wx.TextCtrl(parent=panel, id=wx.ID_ANY,
  1128. value=self.settings.Get(group='atm', key='encoding', subkey='value'),
  1129. name="GetValue", size=(200, -1))
  1130. self.winId['atm:encoding:value'] = encoding.GetId()
  1131. flexSizer.Add(label, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1132. flexSizer.Add(encoding, proportion=0, flag=wx.ALIGN_RIGHT | wx.FIXED_MINSIZE)
  1133. # ask on delete record
  1134. askOnDeleteRec = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  1135. label=_("Ask when deleting data record(s) from table"),
  1136. name='IsChecked')
  1137. askOnDeleteRec.SetValue(self.settings.Get(group='atm', key='askOnDeleteRec', subkey='enabled'))
  1138. self.winId['atm:askOnDeleteRec:enabled'] = askOnDeleteRec.GetId()
  1139. flexSizer.Add(askOnDeleteRec, proportion=0)
  1140. dataBrowserSizer.Add(item=flexSizer,
  1141. proportion=0,
  1142. flag=wx.ALL | wx.EXPAND,
  1143. border=5)
  1144. pageSizer.Add(item=dataBrowserSizer,
  1145. proportion=0,
  1146. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND,
  1147. border=3)
  1148. #
  1149. # create table
  1150. #
  1151. createTableBox = wx.StaticBox(parent=panel, id=wx.ID_ANY,
  1152. label=" %s " % _("Create table"))
  1153. createTableSizer = wx.StaticBoxSizer(createTableBox, wx.VERTICAL)
  1154. flexSizer = wx.FlexGridSizer (cols=2, hgap=5, vgap=5)
  1155. flexSizer.AddGrowableCol(0)
  1156. label = wx.StaticText(parent=panel, id=wx.ID_ANY,
  1157. label=_("Key column"))
  1158. keyColumn = wx.TextCtrl(parent=panel, id=wx.ID_ANY,
  1159. size=(250, -1))
  1160. keyColumn.SetValue(self.settings.Get(group='atm', key='keycolumn', subkey='value'))
  1161. self.winId['atm:keycolumn:value'] = keyColumn.GetId()
  1162. flexSizer.Add(label, proportion=0, flag=wx.ALIGN_CENTER_VERTICAL)
  1163. flexSizer.Add(keyColumn, proportion=0, flag=wx.ALIGN_RIGHT | wx.FIXED_MINSIZE)
  1164. createTableSizer.Add(item=flexSizer,
  1165. proportion=0,
  1166. flag=wx.ALL | wx.EXPAND,
  1167. border=5)
  1168. pageSizer.Add(item=createTableSizer,
  1169. proportion=0,
  1170. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM | wx.EXPAND,
  1171. border=3)
  1172. panel.SetSizer(pageSizer)
  1173. return panel
  1174. def __CreateWorkspacePage(self, notebook):
  1175. """Create notebook page for workspace settings"""
  1176. panel = wx.Panel(parent=notebook, id=wx.ID_ANY)
  1177. notebook.AddPage(page=panel, text=_("Workspace"))
  1178. border = wx.BoxSizer(wx.VERTICAL)
  1179. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Loading workspace"))
  1180. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1181. gridSizer = wx.GridBagSizer (hgap=3, vgap=3)
  1182. gridSizer.AddGrowableCol(0)
  1183. row = 0
  1184. #
  1185. # positioning
  1186. #
  1187. posDisplay = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  1188. label=_("Suppress positioning Map Display Window(s)"),
  1189. name='IsChecked')
  1190. posDisplay.SetValue(self.settings.Get(group='workspace', key='posDisplay', subkey='enabled'))
  1191. self.winId['workspace:posDisplay:enabled'] = posDisplay.GetId()
  1192. gridSizer.Add(item=posDisplay,
  1193. pos=(row, 0), span=(1, 2))
  1194. row +=1
  1195. posManager = wx.CheckBox(parent=panel, id=wx.ID_ANY,
  1196. label=_("Suppress positioning Layer Manager window"),
  1197. name='IsChecked')
  1198. posManager.SetValue(self.settings.Get(group='workspace', key='posManager', subkey='enabled'))
  1199. self.winId['workspace:posManager:enabled'] = posManager.GetId()
  1200. gridSizer.Add(item=posManager,
  1201. pos=(row, 0), span=(1, 2))
  1202. sizer.Add(item=gridSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=5)
  1203. border.Add(item=sizer, proportion=0, flag=wx.ALL | wx.EXPAND, border=3)
  1204. panel.SetSizer(border)
  1205. return panel
  1206. def __CreateAdvancedPage(self, notebook):
  1207. """Create notebook page for advanced settings"""
  1208. panel = wx.Panel(parent=notebook, id=wx.ID_ANY)
  1209. notebook.AddPage(page=panel, text=_("Advanced"))
  1210. border = wx.BoxSizer(wx.VERTICAL)
  1211. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Advanced settings"))
  1212. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1213. gridSizer = wx.GridBagSizer (hgap=3, vgap=3)
  1214. gridSizer.AddGrowableCol(0)
  1215. row = 0
  1216. #
  1217. # place where to store settings
  1218. #
  1219. gridSizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY,
  1220. label=_("Place where to store settings:")),
  1221. flag=wx.ALIGN_LEFT |
  1222. wx.ALIGN_CENTER_VERTICAL,
  1223. pos=(row, 0))
  1224. settingsFile = wx.Choice(parent=panel, id=wx.ID_ANY, size=(125, -1),
  1225. choices=self.settings.Get(group='advanced', key='settingsFile',
  1226. subkey='choices', internal=True),
  1227. name='GetStringSelection')
  1228. settingsFile.SetStringSelection(self.settings.Get(group='advanced', key='settingsFile', subkey='type'))
  1229. self.winId['advanced:settingsFile:type'] = settingsFile.GetId()
  1230. gridSizer.Add(item=settingsFile,
  1231. flag=wx.ALIGN_RIGHT |
  1232. wx.ALIGN_CENTER_VERTICAL,
  1233. pos=(row, 1))
  1234. row += 1
  1235. #
  1236. # icon theme
  1237. #
  1238. gridSizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY,
  1239. label=_("Icon theme:")),
  1240. flag=wx.ALIGN_LEFT |
  1241. wx.ALIGN_CENTER_VERTICAL,
  1242. pos=(row, 0))
  1243. iconTheme = wx.Choice(parent=panel, id=wx.ID_ANY, size=(125, -1),
  1244. choices=self.settings.Get(group='advanced', key='iconTheme',
  1245. subkey='choices', internal=True),
  1246. name="GetStringSelection")
  1247. iconTheme.SetStringSelection(self.settings.Get(group='advanced', key='iconTheme', subkey='type'))
  1248. self.winId['advanced:iconTheme:type'] = iconTheme.GetId()
  1249. gridSizer.Add(item=iconTheme,
  1250. flag=wx.ALIGN_RIGHT |
  1251. wx.ALIGN_CENTER_VERTICAL,
  1252. pos=(row, 1))
  1253. row += 1
  1254. iconNote = wordwrap(_("Note: Requires GUI restart."),
  1255. self.GetSize()[0]-50, wx.ClientDC(self))
  1256. gridSizer.Add(item=wx.StaticText(parent=panel, id=wx.ID_ANY,
  1257. label=iconNote),
  1258. flag=wx.ALIGN_LEFT |
  1259. wx.ALIGN_CENTER_VERTICAL,
  1260. pos=(row, 0), span=(1, 2))
  1261. row += 1
  1262. sizer.Add(item=gridSizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=5)
  1263. border.Add(item=sizer, proportion=1, flag=wx.ALL | wx.EXPAND, border=3)
  1264. panel.SetSizer(border)
  1265. return panel
  1266. def OnSetFont(self, event):
  1267. """'Set font' button pressed"""
  1268. dlg = DefaultFontDialog(parent=self, id=wx.ID_ANY,
  1269. title=_('Select default display font'),
  1270. style=wx.DEFAULT_DIALOG_STYLE)
  1271. if dlg.ShowModal() == wx.ID_OK:
  1272. # set default font and encoding environmental variables
  1273. if dlg.font:
  1274. os.environ["GRASS_FONT"] = dlg.font
  1275. self.settings.Set(group='display', value=dlg.font,
  1276. key='font', subkey='type')
  1277. if dlg.encoding and \
  1278. dlg.encoding != "ISO-8859-1":
  1279. os.environ["GRASS_ENCODING"] = dlg.encoding
  1280. self.settings.Set(group='display', value=dlg.encoding,
  1281. key='font', subkey='encoding')
  1282. dlg.Destroy()
  1283. event.Skip()
  1284. def OnSave(self, event):
  1285. """Button 'Save' pressed"""
  1286. if self.__UpdateSettings():
  1287. file = self.settings.SaveToFile()
  1288. self.parent.goutput.WriteLog(_('Settings saved to file \'%s\'.') % file)
  1289. self.Close()
  1290. def OnApply(self, event):
  1291. """Button 'Apply' pressed"""
  1292. if self.__UpdateSettings():
  1293. self.Close()
  1294. def OnCancel(self, event):
  1295. """Button 'Cancel' pressed"""
  1296. self.Close()
  1297. def OnDefault(self, event):
  1298. """Button 'Set to default' pressed"""
  1299. self.settings.userSettings = copy.deepcopy(self.settings.defaultSettings)
  1300. # update widgets
  1301. for gks in self.winId.keys():
  1302. try:
  1303. group, key, subkey = gks.split(':')
  1304. value = self.settings.Get(group, key, subkey)
  1305. except ValueError:
  1306. group, key, subkey, subkey1 = gks.split(':')
  1307. value = self.settings.Get(group, key, [subkey, subkey1])
  1308. win = self.FindWindowById(self.winId[gks])
  1309. if win.GetName() in ('GetValue', 'IsChecked'):
  1310. value = win.SetValue(value)
  1311. elif win.GetName() == 'GetSelection':
  1312. value = win.SetSelection(value)
  1313. elif win.GetName() == 'GetStringSelection':
  1314. value = win.SetStringSelection(value)
  1315. else:
  1316. value = win.SetValue(value)
  1317. def __UpdateSettings(self):
  1318. """Update user settings"""
  1319. for item in self.winId.keys():
  1320. try:
  1321. group, key, subkey = item.split(':')
  1322. subkey1 = None
  1323. except ValueError:
  1324. group, key, subkey, subkey1 = item.split(':')
  1325. id = self.winId[item]
  1326. win = self.FindWindowById(id)
  1327. if win.GetName() == 'GetValue':
  1328. value = win.GetValue()
  1329. elif win.GetName() == 'GetSelection':
  1330. value = win.GetSelection()
  1331. elif win.GetName() == 'IsChecked':
  1332. value = win.IsChecked()
  1333. elif win.GetName() == 'GetStringSelection':
  1334. value = win.GetStringSelection()
  1335. elif win.GetName() == 'GetColour':
  1336. value = tuple(win.GetValue())
  1337. else:
  1338. value = win.GetValue()
  1339. if key == 'keycolumn' and value == '':
  1340. wx.MessageBox(parent=self,
  1341. message=_("Key column cannot be empty string."),
  1342. caption=_("Error"), style=wx.OK | wx.ICON_ERROR)
  1343. win.SetValue(self.settings.Get(group='atm', key='keycolumn', subkey='value'))
  1344. return False
  1345. if subkey1:
  1346. self.settings.Set(group, value, key, [subkey, subkey1])
  1347. else:
  1348. self.settings.Set(group, value, key, subkey)
  1349. #
  1350. # update default window dimension
  1351. #
  1352. if self.settings.Get(group='general', key='defWindowPos', subkey='enabled') is True:
  1353. dim = ''
  1354. # layer manager
  1355. pos = self.parent.GetPosition()
  1356. size = self.parent.GetSize()
  1357. dim = '%d,%d,%d,%d' % (pos[0], pos[1], size[0], size[1])
  1358. # opened displays
  1359. for page in range(0, self.parent.gm_cb.GetPageCount()):
  1360. pos = self.parent.gm_cb.GetPage(page).maptree.mapdisplay.GetPosition()
  1361. size = self.parent.gm_cb.GetPage(page).maptree.mapdisplay.GetSize()
  1362. dim += ',%d,%d,%d,%d' % (pos[0], pos[1], size[0], size[1])
  1363. self.settings.Set(group='general', key='defWindowPos', subkey='dim', value=dim)
  1364. else:
  1365. self.settings.Set(group='general', key='defWindowPos', subkey='dim', value='')
  1366. return True
  1367. class DefaultFontDialog(wx.Dialog):
  1368. """
  1369. Opens a file selection dialog to select default font
  1370. to use in all GRASS displays
  1371. """
  1372. def __init__(self, parent, id, title,
  1373. pos=wx.DefaultPosition, size=wx.DefaultSize,
  1374. style=wx.DEFAULT_DIALOG_STYLE,
  1375. settings=globalSettings):
  1376. self.settings = settings
  1377. wx.Dialog.__init__(self, parent, id, title, pos, size, style)
  1378. panel = wx.Panel(parent=self, id=wx.ID_ANY)
  1379. if "GRASS_FONT" in os.environ:
  1380. self.font = os.environ["GRASS_FONT"]
  1381. else:
  1382. self.font = self.settings.Get(group='display',
  1383. key='font', subkey='type')
  1384. self.fontlist = self.GetFonts()
  1385. self.encoding = self.settings.Get(group='display',
  1386. key='font', subkey='encoding')
  1387. border = wx.BoxSizer(wx.VERTICAL)
  1388. box = wx.StaticBox (parent=panel, id=wx.ID_ANY, label=" %s " % _("Font settings"))
  1389. sizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  1390. gridSizer = wx.GridBagSizer (hgap=5, vgap=5)
  1391. gridSizer.AddGrowableCol(0)
  1392. label = wx.StaticText(parent=panel, id=wx.ID_ANY,
  1393. label=_("Select font:"))
  1394. gridSizer.Add(item=label,
  1395. flag=wx.ALIGN_TOP,
  1396. pos=(0,0))
  1397. self.fontlb = wx.ListBox(parent=panel, id=wx.ID_ANY, pos=wx.DefaultPosition,
  1398. choices=self.fontlist,
  1399. style=wx.LB_SINGLE|wx.LB_SORT)
  1400. self.Bind(wx.EVT_LISTBOX, self.EvtListBox, self.fontlb)
  1401. self.Bind(wx.EVT_LISTBOX_DCLICK, self.EvtListBoxDClick, self.fontlb)
  1402. if self.font:
  1403. self.fontlb.SetStringSelection(self.font, True)
  1404. gridSizer.Add(item=self.fontlb,
  1405. flag=wx.EXPAND, pos=(0, 1))
  1406. label = wx.StaticText(parent=panel, id=wx.ID_ANY,
  1407. label=("Character encoding:"))
  1408. gridSizer.Add(item=label,
  1409. flag=wx.ALIGN_CENTER_VERTICAL,
  1410. pos=(1, 0))
  1411. self.textentry = wx.TextCtrl(parent=panel, id=wx.ID_ANY,
  1412. value=self.encoding)
  1413. gridSizer.Add(item=self.textentry,
  1414. flag=wx.EXPAND, pos=(1, 1))
  1415. self.textentry.Bind(wx.EVT_TEXT, self.OnEncoding)
  1416. sizer.Add(item=gridSizer, proportion=1,
  1417. flag=wx.EXPAND | wx.ALL,
  1418. border=5)
  1419. border.Add(item=sizer, proportion=1,
  1420. flag=wx.ALL | wx.EXPAND, border=3)
  1421. btnsizer = wx.StdDialogButtonSizer()
  1422. btn = wx.Button(parent=panel, id=wx.ID_OK)
  1423. btn.SetDefault()
  1424. btnsizer.AddButton(btn)
  1425. btn = wx.Button(parent=panel, id=wx.ID_CANCEL)
  1426. btnsizer.AddButton(btn)
  1427. btnsizer.Realize()
  1428. border.Add(item=btnsizer, proportion=0,
  1429. flag=wx.EXPAND | wx.ALIGN_RIGHT | wx.ALL, border=5)
  1430. panel.SetAutoLayout(True)
  1431. panel.SetSizer(border)
  1432. border.Fit(self)
  1433. self.Layout()
  1434. def EvtRadioBox(self, event):
  1435. if event.GetInt() == 0:
  1436. self.fonttype = 'grassfont'
  1437. elif event.GetInt() == 1:
  1438. self.fonttype = 'truetype'
  1439. self.fontlist = self.GetFonts(self.fonttype)
  1440. self.fontlb.SetItems(self.fontlist)
  1441. def OnEncoding(self, event):
  1442. self.encoding = event.GetString()
  1443. def EvtListBox(self, event):
  1444. self.font = event.GetString()
  1445. event.Skip()
  1446. def EvtListBoxDClick(self, event):
  1447. self.font = event.GetString()
  1448. event.Skip()
  1449. def GetFonts(self):
  1450. """
  1451. parses fonts directory or fretypecap file to get a list of fonts for the listbox
  1452. """
  1453. fontlist = []
  1454. cmd = ["d.font", "-l"]
  1455. ret = gcmd.RunCommand('d.font',
  1456. read = True,
  1457. flags = 'l')
  1458. if not ret:
  1459. return fontlist
  1460. dfonts = ret.split('\n')
  1461. dfonts.sort(lambda x,y: cmp(x.lower(), y.lower()))
  1462. for item in range(len(dfonts)):
  1463. # ignore duplicate fonts and those starting with #
  1464. if not dfonts[item].startswith('#') and \
  1465. dfonts[item] != dfonts[item-1]:
  1466. fontlist.append(dfonts[item])
  1467. return fontlist
  1468. class MapsetAccess(wx.Dialog):
  1469. """
  1470. Controls setting options and displaying/hiding map overlay decorations
  1471. """
  1472. def __init__(self, parent, id, title=_('Set/unset access to mapsets in current location'),
  1473. pos=wx.DefaultPosition, size=(350, 400),
  1474. style=wx.DEFAULT_DIALOG_STYLE|wx.RESIZE_BORDER):
  1475. wx.Dialog.__init__(self, parent, id, title, pos, size, style)
  1476. self.all_mapsets = utils.ListOfMapsets(all=True)
  1477. self.accessible_mapsets = utils.ListOfMapsets(all=False)
  1478. self.curr_mapset = grass.gisenv()['MAPSET']
  1479. # make a checklistbox from available mapsets and check those that are active
  1480. sizer = wx.BoxSizer(wx.VERTICAL)
  1481. label = wx.StaticText(parent=self, id=wx.ID_ANY,
  1482. label=_("Check mapset to make it accessible, uncheck it to hide it.%s"
  1483. "Note: PERMANENT and current mapset are always accessible.") % os.linesep)
  1484. sizer.Add(item=label, proportion=0,
  1485. flag=wx.ALL, border=5)
  1486. self.mapsetlb = CheckListMapset(parent=self)
  1487. self.mapsetlb.LoadData(self.all_mapsets)
  1488. sizer.Add(item=self.mapsetlb, proportion=1,
  1489. flag=wx.ALL | wx.EXPAND, border=5)
  1490. # check all accessible mapsets
  1491. for mset in self.accessible_mapsets:
  1492. self.mapsetlb.CheckItem(self.all_mapsets.index(mset), True)
  1493. # dialog buttons
  1494. line = wx.StaticLine(parent=self, id=wx.ID_ANY,
  1495. style=wx.LI_HORIZONTAL)
  1496. sizer.Add(item=line, proportion=0,
  1497. flag=wx.EXPAND | wx.ALIGN_CENTRE | wx.ALL, border=5)
  1498. btnsizer = wx.StdDialogButtonSizer()
  1499. okbtn = wx.Button(self, wx.ID_OK)
  1500. okbtn.SetDefault()
  1501. btnsizer.AddButton(okbtn)
  1502. cancelbtn = wx.Button(self, wx.ID_CANCEL)
  1503. btnsizer.AddButton(cancelbtn)
  1504. btnsizer.Realize()
  1505. sizer.Add(item=btnsizer, proportion=0,
  1506. flag=wx.EXPAND | wx.ALIGN_RIGHT | wx.ALL, border=5)
  1507. # do layout
  1508. self.Layout()
  1509. self.SetSizer(sizer)
  1510. sizer.Fit(self)
  1511. self.SetMinSize(size)
  1512. def GetMapsets(self):
  1513. """Get list of checked mapsets"""
  1514. ms = []
  1515. i = 0
  1516. for mset in self.all_mapsets:
  1517. if self.mapsetlb.IsChecked(i):
  1518. ms.append(mset)
  1519. i += 1
  1520. return ms
  1521. class CheckListMapset(wx.ListCtrl, listmix.ListCtrlAutoWidthMixin, listmix.CheckListCtrlMixin):
  1522. """List of mapset/owner/group"""
  1523. def __init__(self, parent, pos=wx.DefaultPosition,
  1524. log=None):
  1525. self.parent = parent
  1526. wx.ListCtrl.__init__(self, parent, wx.ID_ANY,
  1527. style=wx.LC_REPORT)
  1528. listmix.CheckListCtrlMixin.__init__(self)
  1529. self.log = log
  1530. # setup mixins
  1531. listmix.ListCtrlAutoWidthMixin.__init__(self)
  1532. def LoadData(self, mapsets):
  1533. """Load data into list"""
  1534. self.InsertColumn(0, _('Mapset'))
  1535. self.InsertColumn(1, _('Owner'))
  1536. self.InsertColumn(2, _('Group'))
  1537. gisenv = grass.gisenv()
  1538. locationPath = os.path.join(gisenv['GISDBASE'], gisenv['LOCATION_NAME'])
  1539. for mapset in mapsets:
  1540. index = self.InsertStringItem(sys.maxint, mapset)
  1541. mapsetPath = os.path.join(locationPath,
  1542. mapset)
  1543. stat_info = os.stat(mapsetPath)
  1544. if os.name in ('posix', 'mac'):
  1545. self.SetStringItem(index, 1, "%s" % pwd.getpwuid(stat_info.st_uid)[0])
  1546. # FIXME: get group name
  1547. self.SetStringItem(index, 2, "%-8s" % stat_info.st_gid)
  1548. else:
  1549. # FIXME: no pwd under MS Windows (owner: 0, group: 0)
  1550. self.SetStringItem(index, 1, "%-8s" % stat_info.st_uid)
  1551. self.SetStringItem(index, 2, "%-8s" % stat_info.st_gid)
  1552. self.SetColumnWidth(col=0, width=wx.LIST_AUTOSIZE)
  1553. self.SetColumnWidth(col=1, width=wx.LIST_AUTOSIZE)
  1554. def OnCheckItem(self, index, flag):
  1555. """Mapset checked/unchecked"""
  1556. mapset = self.parent.all_mapsets[index]
  1557. if mapset == 'PERMANENT' or mapset == self.parent.curr_mapset:
  1558. self.CheckItem(index, True)