preferences.py 69 KB

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