widgets.py 51 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678
  1. """
  2. @package gui_core.widgets
  3. @brief Core GUI widgets
  4. Classes:
  5. - widgets::GNotebook
  6. - widgets::ScrolledPanel
  7. - widgets::NumTextCtrl
  8. - widgets::FloatSlider
  9. - widgets::SymbolButton
  10. - widgets::StaticWrapText
  11. - widgets::BaseValidator
  12. - widgets::CoordinatesValidator
  13. - widgets::IntegerValidator
  14. - widgets::FloatValidator
  15. - widgets::EmailValidator
  16. - widgets::TimeISOValidator
  17. - widgets::MapValidator
  18. - widgets::NTCValidator
  19. - widgets::SimpleValidator
  20. - widgets::GenericValidator
  21. - widgets::GListCtrl
  22. - widgets::SearchModuleWidget
  23. - widgets::ManageSettingsWidget
  24. - widgets::PictureComboBox
  25. - widgets::ColorTablesComboBox
  26. - widgets::BarscalesComboBox
  27. - widgets::NArrowsComboBox
  28. - widgets::LayersList
  29. @todo:
  30. - move validators to a separate file gui_core/validators.py
  31. (C) 2008-2014 by the GRASS Development Team
  32. This program is free software under the GNU General Public License
  33. (>=v2). Read the file COPYING that comes with GRASS for details.
  34. @author Martin Landa <landa.martin gmail.com> (Google SoC 2008/2010)
  35. @author Enhancements by Michael Barton <michael.barton asu.edu>
  36. @author Anna Kratochvilova <kratochanna gmail.com> (Google SoC 2011)
  37. @author Stepan Turek <stepan.turek seznam.cz> (ManageSettingsWidget - created from GdalSelect)
  38. @author Matej Krejci <matejkrejci gmail.com> (Google GSoC 2014; EmailValidator, TimeISOValidator)
  39. """
  40. import os
  41. import sys
  42. import string
  43. import re
  44. import six
  45. from bisect import bisect
  46. from datetime import datetime
  47. from core.globalvar import wxPythonPhoenix
  48. import wx
  49. import wx.lib.mixins.listctrl as listmix
  50. import wx.lib.scrolledpanel as SP
  51. from wx.lib.stattext import GenStaticText
  52. from wx.lib.wordwrap import wordwrap
  53. if wxPythonPhoenix:
  54. import wx.adv
  55. from wx.adv import OwnerDrawnComboBox
  56. else:
  57. import wx.combo
  58. from wx.combo import OwnerDrawnComboBox
  59. try:
  60. import wx.lib.agw.flatnotebook as FN
  61. except ImportError:
  62. import wx.lib.flatnotebook as FN
  63. try:
  64. from wx.lib.buttons import ThemedGenBitmapTextButton as BitmapTextButton
  65. except ImportError: # not sure about TGBTButton version
  66. from wx.lib.buttons import GenBitmapTextButton as BitmapTextButton
  67. try:
  68. import wx.lib.agw.customtreectrl as CT
  69. except ImportError:
  70. import wx.lib.customtreectrl as CT
  71. if wxPythonPhoenix:
  72. from wx import Validator as Validator
  73. else:
  74. from wx import PyValidator as Validator
  75. from grass.script import core as grass
  76. from grass.pydispatch.signal import Signal
  77. from core import globalvar
  78. from core.utils import _
  79. from core.gcmd import GMessage, GError
  80. from core.debug import Debug
  81. from gui_core.wrap import Button, SearchCtrl, StaticText, StaticBox, \
  82. TextCtrl, Menu, Rect, EmptyBitmap, ListCtrl
  83. class NotebookController:
  84. """Provides handling of notebook page names.
  85. Translates page names to page indices.
  86. Class is aggregated in notebook subclasses.
  87. Notebook subclasses must delegate methods to controller.
  88. Methods inherited from notebook class must be delegated explicitly
  89. and other methods can be delegated by @c __getattr__.
  90. """
  91. def __init__(self, classObject, widget):
  92. """
  93. :param classObject: notebook class name (object, i.e. FlatNotebook)
  94. :param widget: notebook instance
  95. """
  96. self.notebookPages = {}
  97. self.classObject = classObject
  98. self.widget = widget
  99. self.highlightedTextEnd = _(" (...)")
  100. self.BindPageChanged()
  101. def BindPageChanged(self):
  102. """Binds page changed event."""
  103. self.widget.Bind(wx.EVT_NOTEBOOK_PAGE_CHANGED, self.OnRemoveHighlight)
  104. def AddPage(self, **kwargs):
  105. """Add a new page
  106. """
  107. if 'name' in kwargs:
  108. self.notebookPages[kwargs['name']] = kwargs['page']
  109. del kwargs['name']
  110. self.classObject.AddPage(self.widget, **kwargs)
  111. def InsertPage(self, **kwargs):
  112. """Insert a new page
  113. """
  114. if 'name' in kwargs:
  115. self.notebookPages[kwargs['name']] = kwargs['page']
  116. del kwargs['name']
  117. try:
  118. self.classObject.InsertPage(self.widget, **kwargs)
  119. except TypeError as e: # documentation says 'index', but certain versions of wx require 'n'
  120. kwargs['n'] = kwargs['index']
  121. del kwargs['index']
  122. self.classObject.InsertPage(self.widget, **kwargs)
  123. def DeletePage(self, page):
  124. """Delete page
  125. :param page: name
  126. :return: True if page was deleted, False if not exists
  127. """
  128. delPageIndex = self.GetPageIndexByName(page)
  129. if delPageIndex != -1:
  130. ret = self.classObject.DeletePage(self.widget, delPageIndex)
  131. if ret:
  132. del self.notebookPages[page]
  133. return ret
  134. else:
  135. return False
  136. def RemovePage(self, page):
  137. """Delete page without deleting the associated window.
  138. :param page: name
  139. :return: True if page was deleted, False if not exists
  140. """
  141. delPageIndex = self.GetPageIndexByName(page)
  142. if delPageIndex != -1:
  143. ret = self.classObject.RemovePage(self.widget, delPageIndex)
  144. if ret:
  145. del self.notebookPages[page]
  146. return ret
  147. else:
  148. return False
  149. def SetSelectionByName(self, page):
  150. """Set active notebook page.
  151. :param page: name, eg. 'layers', 'output', 'search', 'pyshell', 'nviz'
  152. (depends on concrete notebook instance)
  153. """
  154. idx = self.GetPageIndexByName(page)
  155. if self.classObject.GetSelection(self.widget) != idx:
  156. self.classObject.SetSelection(self.widget, idx)
  157. self.RemoveHighlight(idx)
  158. def OnRemoveHighlight(self, event):
  159. """Highlighted tab name should be removed."""
  160. page = event.GetSelection()
  161. self.RemoveHighlight(page)
  162. event.Skip()
  163. def RemoveHighlight(self, page):
  164. """Removes highlight string from notebook tab name if necessary.
  165. :param page: index
  166. """
  167. text = self.classObject.GetPageText(self.widget, page)
  168. if text.endswith(self.highlightedTextEnd):
  169. text = text.replace(self.highlightedTextEnd, '')
  170. self.classObject.SetPageText(self.widget, page, text)
  171. def GetPageIndexByName(self, page):
  172. """Get notebook page index
  173. :param page: name
  174. """
  175. if page not in self.notebookPages:
  176. return -1
  177. for pageIndex in range(self.classObject.GetPageCount(self.widget)):
  178. if self.notebookPages[page] == self.classObject.GetPage(
  179. self.widget, pageIndex):
  180. break
  181. return pageIndex
  182. def HighlightPageByName(self, page):
  183. pageIndex = self.GetPageIndexByName(page)
  184. self.HighlightPage(pageIndex)
  185. def HighlightPage(self, index):
  186. if self.classObject.GetSelection(self.widget) != index:
  187. text = self.classObject.GetPageText(self.widget, index)
  188. if not text.endswith(self.highlightedTextEnd):
  189. text += self.highlightedTextEnd
  190. self.classObject.SetPageText(self.widget, index, text)
  191. def SetPageImage(self, page, index):
  192. """Sets image index for page
  193. :param page: page name
  194. :param index: image index (in wx.ImageList)
  195. """
  196. pageIndex = self.GetPageIndexByName(page)
  197. self.classObject.SetPageImage(self.widget, pageIndex, index)
  198. class FlatNotebookController(NotebookController):
  199. """Controller specialized for FN.FlatNotebook subclasses"""
  200. def __init__(self, classObject, widget):
  201. NotebookController.__init__(self, classObject, widget)
  202. def BindPageChanged(self):
  203. self.widget.Bind(
  204. FN.EVT_FLATNOTEBOOK_PAGE_CHANGED,
  205. self.OnRemoveHighlight)
  206. def GetPageIndexByName(self, page):
  207. """Get notebook page index
  208. :param page: name
  209. """
  210. if page not in self.notebookPages:
  211. return -1
  212. return self.classObject.GetPageIndex(
  213. self.widget, self.notebookPages[page])
  214. def InsertPage(self, **kwargs):
  215. """Insert a new page
  216. """
  217. if 'name' in kwargs:
  218. self.notebookPages[kwargs['name']] = kwargs['page']
  219. del kwargs['name']
  220. kwargs['indx'] = kwargs['index']
  221. del kwargs['index']
  222. self.classObject.InsertPage(self.widget, **kwargs)
  223. class GNotebook(FN.FlatNotebook):
  224. """Generic notebook widget.
  225. Enables advanced style settings.
  226. Problems with hidden tabs and does not respect system colors (native look).
  227. """
  228. def __init__(self, parent, style, **kwargs):
  229. if globalvar.hasAgw:
  230. FN.FlatNotebook.__init__(self, parent, id=wx.ID_ANY,
  231. agwStyle=style, **kwargs)
  232. else:
  233. FN.FlatNotebook.__init__(self, parent, id=wx.ID_ANY,
  234. style=style, **kwargs)
  235. self.controller = FlatNotebookController(classObject=FN.FlatNotebook,
  236. widget=self)
  237. def AddPage(self, **kwargs):
  238. """@copydoc NotebookController::AddPage()"""
  239. self.controller.AddPage(**kwargs)
  240. def InsertNBPage(self, **kwargs):
  241. """@copydoc NotebookController::InsertPage()"""
  242. self.controller.InsertPage(**kwargs)
  243. def DeleteNBPage(self, page):
  244. """@copydoc NotebookController::DeletePage()"""
  245. return self.controller.DeletePage(page)
  246. def RemoveNBPage(self, page):
  247. """@copydoc NotebookController::RemovePage()"""
  248. return self.controller.RemovePage(page)
  249. def SetPageImage(self, page, index):
  250. """Does nothing because we don't want images for this style"""
  251. pass
  252. def __getattr__(self, name):
  253. return getattr(self.controller, name)
  254. class FormNotebook(wx.Notebook):
  255. """Notebook widget.
  256. Respects native look.
  257. """
  258. def __init__(self, parent, style):
  259. wx.Notebook.__init__(self, parent, id=wx.ID_ANY, style=style)
  260. self.controller = NotebookController(classObject=wx.Notebook,
  261. widget=self)
  262. def AddPage(self, **kwargs):
  263. """@copydoc NotebookController::AddPage()"""
  264. self.controller.AddPage(**kwargs)
  265. def InsertNBPage(self, **kwargs):
  266. """@copydoc NotebookController::InsertPage()"""
  267. self.controller.InsertPage(**kwargs)
  268. def DeleteNBPage(self, page):
  269. """@copydoc NotebookController::DeletePage()"""
  270. return self.controller.DeletePage(page)
  271. def RemoveNBPage(self, page):
  272. """@copydoc NotebookController::RemovePage()"""
  273. return self.controller.RemovePage(page)
  274. def SetPageImage(self, page, index):
  275. """@copydoc NotebookController::SetPageImage()"""
  276. return self.controller.SetPageImage(page, index)
  277. def __getattr__(self, name):
  278. return getattr(self.controller, name)
  279. class FormListbook(wx.Listbook):
  280. """Notebook widget.
  281. Respects native look.
  282. """
  283. def __init__(self, parent, style):
  284. wx.Listbook.__init__(self, parent, id=wx.ID_ANY, style=style)
  285. self.controller = NotebookController(classObject=wx.Listbook,
  286. widget=self)
  287. def AddPage(self, **kwargs):
  288. """@copydoc NotebookController::AddPage()"""
  289. self.controller.AddPage(**kwargs)
  290. def InsertPage_(self, **kwargs):
  291. """@copydoc NotebookController::InsertPage()"""
  292. self.controller.InsertPage(**kwargs)
  293. def DeletePage(self, page):
  294. """@copydoc NotebookController::DeletePage()"""
  295. return self.controller.DeletePage(page)
  296. def RemovePage(self, page):
  297. """@copydoc NotebookController::RemovePage()"""
  298. return self.controller.RemovePage(page)
  299. def SetPageImage(self, page, index):
  300. """@copydoc NotebookController::SetPageImage()"""
  301. return self.controller.SetPageImage(page, index)
  302. def __getattr__(self, name):
  303. return getattr(self.controller, name)
  304. class ScrolledPanel(SP.ScrolledPanel):
  305. """Custom ScrolledPanel to avoid strange behaviour concerning focus"""
  306. def __init__(self, parent, style=wx.TAB_TRAVERSAL):
  307. SP.ScrolledPanel.__init__(self, parent=parent, id=wx.ID_ANY,
  308. style=style)
  309. def OnChildFocus(self, event):
  310. pass
  311. class NumTextCtrl(wx.TextCtrl):
  312. """Class derived from wx.TextCtrl for numerical values only"""
  313. def __init__(self, parent, **kwargs):
  314. ## self.precision = kwargs.pop('prec')
  315. wx.TextCtrl.__init__(self, parent=parent,
  316. validator=NTCValidator(flag='DIGIT_ONLY'),
  317. **kwargs)
  318. def SetValue(self, value):
  319. super(NumTextCtrl, self).SetValue(str(value))
  320. def GetValue(self):
  321. val = super(NumTextCtrl, self).GetValue()
  322. if val == '':
  323. val = '0'
  324. try:
  325. return float(val)
  326. except ValueError:
  327. val = ''.join(''.join(val.split('-')).split('.'))
  328. return float(val)
  329. def SetRange(self, min, max):
  330. pass
  331. class FloatSlider(wx.Slider):
  332. """Class derived from wx.Slider for floats"""
  333. def __init__(self, **kwargs):
  334. Debug.msg(1, "FloatSlider.__init__()")
  335. wx.Slider.__init__(self, **kwargs)
  336. self.coef = 1.
  337. # init range
  338. self.minValueOrig = 0
  339. self.maxValueOrig = 1
  340. def SetValue(self, value):
  341. value *= self.coef
  342. if abs(value) < 1 and value != 0:
  343. while abs(value) < 1:
  344. value *= 100
  345. self.coef *= 100
  346. super(FloatSlider, self).SetRange(self.minValueOrig * self.coef,
  347. self.maxValueOrig * self.coef)
  348. super(FloatSlider, self).SetValue(value)
  349. Debug.msg(4, "FloatSlider.SetValue(): value = %f" % value)
  350. def SetRange(self, minValue, maxValue):
  351. self.coef = 1.
  352. self.minValueOrig = minValue
  353. self.maxValueOrig = maxValue
  354. if abs(minValue) < 1 or abs(maxValue) < 1:
  355. while (abs(minValue) < 1 and minValue != 0) or (
  356. abs(maxValue) < 1 and maxValue != 0):
  357. minValue *= 100
  358. maxValue *= 100
  359. self.coef *= 100
  360. super(
  361. FloatSlider,
  362. self).SetValue(
  363. super(
  364. FloatSlider,
  365. self).GetValue() *
  366. self.coef)
  367. super(FloatSlider, self).SetRange(minValue, maxValue)
  368. Debug.msg(
  369. 4, "FloatSlider.SetRange(): minValue = %f, maxValue = %f" %
  370. (minValue, maxValue))
  371. def GetValue(self):
  372. val = super(FloatSlider, self).GetValue()
  373. Debug.msg(4, "FloatSlider.GetValue(): value = %f" % (val / self.coef))
  374. return val / self.coef
  375. class SymbolButton(BitmapTextButton):
  376. """Button with symbol and label."""
  377. def __init__(self, parent, usage, label, **kwargs):
  378. """Constructor
  379. :param parent: parent (usually wx.Panel)
  380. :param usage: determines usage and picture
  381. :param label: displayed label
  382. """
  383. size = (15, 15)
  384. buffer = EmptyBitmap(*size)
  385. BitmapTextButton.__init__(self, parent=parent, label=" " + label,
  386. bitmap=buffer, **kwargs)
  387. dc = wx.MemoryDC()
  388. dc.SelectObject(buffer)
  389. maskColor = wx.Colour(255, 255, 255)
  390. dc.SetBrush(wx.Brush(maskColor))
  391. dc.Clear()
  392. if usage == 'record':
  393. self.DrawRecord(dc, size)
  394. elif usage == 'stop':
  395. self.DrawStop(dc, size)
  396. elif usage == 'play':
  397. self.DrawPlay(dc, size)
  398. elif usage == 'pause':
  399. self.DrawPause(dc, size)
  400. if sys.platform != "win32":
  401. buffer.SetMaskColour(maskColor)
  402. self.SetBitmapLabel(buffer)
  403. dc.SelectObject(wx.NullBitmap)
  404. def DrawRecord(self, dc, size):
  405. """Draw record symbol"""
  406. dc.SetBrush(wx.Brush(wx.Colour(255, 0, 0)))
  407. dc.DrawCircle(size[0] / 2, size[1] / 2, size[0] / 2)
  408. def DrawStop(self, dc, size):
  409. """Draw stop symbol"""
  410. dc.SetBrush(wx.Brush(wx.Colour(50, 50, 50)))
  411. dc.DrawRectangle(0, 0, size[0], size[1])
  412. def DrawPlay(self, dc, size):
  413. """Draw play symbol"""
  414. dc.SetBrush(wx.Brush(wx.Colour(0, 255, 0)))
  415. points = (wx.Point(0, 0), wx.Point(0, size[1]), wx.Point(size[0],
  416. size[1] / 2))
  417. dc.DrawPolygon(points)
  418. def DrawPause(self, dc, size):
  419. """Draw pause symbol"""
  420. dc.SetBrush(wx.Brush(wx.Colour(50, 50, 50)))
  421. dc.DrawRectangle(0, 0, 2 * size[0] / 5, size[1])
  422. dc.DrawRectangle(3 * size[0] / 5, 0, 2 * size[0] / 5, size[1])
  423. class StaticWrapText(GenStaticText):
  424. """A Static Text widget that wraps its text to fit parents width,
  425. enlarging its height if necessary."""
  426. def __init__(self, parent, id=wx.ID_ANY,
  427. label='', margin=0, *args, **kwds):
  428. self._margin = margin
  429. self._initialLabel = label
  430. self.init = False
  431. GenStaticText.__init__(self, parent, id, label, *args, **kwds)
  432. self.Bind(wx.EVT_SIZE, self.OnSize)
  433. def DoGetBestSize(self):
  434. """Overriden method which reports widget's best size."""
  435. if not self.init:
  436. self.init = True
  437. self._updateLabel()
  438. parent = self.GetParent()
  439. newExtent = wx.ClientDC(parent).GetMultiLineTextExtent(self.GetLabel())
  440. # when starting, width is very small and height is big which creates
  441. # very high windows
  442. if newExtent[0] < newExtent[1]:
  443. return (0, 0)
  444. return newExtent[:2]
  445. def OnSize(self, event):
  446. self._updateLabel()
  447. event.Skip()
  448. def _updateLabel(self):
  449. """Calculates size of wrapped label"""
  450. parent = self.GetParent()
  451. newLabel = wordwrap(text=self._initialLabel, width=parent.GetSize()[0],
  452. dc=wx.ClientDC(parent), breakLongWords=True,
  453. margin=self._margin)
  454. GenStaticText.SetLabel(self, newLabel)
  455. def SetLabel(self, label):
  456. self._initialLabel = label
  457. self._updateLabel()
  458. class BaseValidator(Validator):
  459. def __init__(self):
  460. Validator.__init__(self)
  461. self.Bind(wx.EVT_TEXT, self.OnText)
  462. def OnText(self, event):
  463. """Do validation"""
  464. self.Validate()
  465. event.Skip()
  466. def Validate(self):
  467. """Validate input"""
  468. textCtrl = self.GetWindow()
  469. text = textCtrl.GetValue()
  470. if text:
  471. try:
  472. self.type(text)
  473. except ValueError:
  474. self._notvalid()
  475. return False
  476. self._valid()
  477. return True
  478. def _notvalid(self):
  479. textCtrl = self.GetWindow()
  480. textCtrl.SetBackgroundColour("grey")
  481. textCtrl.SetFocus()
  482. textCtrl.Refresh()
  483. def _valid(self):
  484. textCtrl = self.GetWindow()
  485. sysColor = wx.SystemSettings.GetColour(wx.SYS_COLOUR_WINDOW)
  486. textCtrl.SetBackgroundColour(sysColor)
  487. textCtrl.Refresh()
  488. return True
  489. def TransferToWindow(self):
  490. return True # Prevent wxDialog from complaining.
  491. def TransferFromWindow(self):
  492. return True # Prevent wxDialog from complaining.
  493. class CoordinatesValidator(BaseValidator):
  494. """Validator for coordinates input (list of floats separated by comma)"""
  495. def __init__(self):
  496. BaseValidator.__init__(self)
  497. def Validate(self):
  498. """Validate input"""
  499. textCtrl = self.GetWindow()
  500. text = textCtrl.GetValue()
  501. if text:
  502. try:
  503. text = text.split(',')
  504. for t in text:
  505. float(t)
  506. if len(text) % 2 != 0:
  507. return False
  508. except ValueError:
  509. self._notvalid()
  510. return False
  511. self._valid()
  512. return True
  513. def Clone(self):
  514. """Clone validator"""
  515. return CoordinatesValidator()
  516. class IntegerValidator(BaseValidator):
  517. """Validator for floating-point input"""
  518. def __init__(self):
  519. BaseValidator.__init__(self)
  520. self.type = int
  521. def Clone(self):
  522. """Clone validator"""
  523. return IntegerValidator()
  524. class FloatValidator(BaseValidator):
  525. """Validator for floating-point input"""
  526. def __init__(self):
  527. BaseValidator.__init__(self)
  528. self.type = float
  529. def Clone(self):
  530. """Clone validator"""
  531. return FloatValidator()
  532. class EmailValidator(BaseValidator):
  533. """Validator for email input"""
  534. def __init__(self):
  535. BaseValidator.__init__(self)
  536. def Validate(self):
  537. """Validate input"""
  538. textCtrl = self.GetWindow()
  539. text = textCtrl.GetValue()
  540. if text:
  541. if re.match(r'\b[\w.-]+@[\w.-]+.\w{2,4}\b', text) is None:
  542. self._notvalid()
  543. return False
  544. self._valid()
  545. return True
  546. def Clone(self):
  547. """Clone validator"""
  548. return EmailValidator()
  549. class TimeISOValidator(BaseValidator):
  550. """Validator for time ISO format (YYYY-MM-DD) input"""
  551. def __init__(self):
  552. BaseValidator.__init__(self)
  553. def Validate(self):
  554. """Validate input"""
  555. textCtrl = self.GetWindow()
  556. text = textCtrl.GetValue()
  557. if text:
  558. try:
  559. datetime.strptime(text, '%Y-%m-%d')
  560. except:
  561. self._notvalid()
  562. return False
  563. self._valid()
  564. return True
  565. def Clone(self):
  566. """Clone validator"""
  567. return TimeISOValidator()
  568. class NTCValidator(Validator):
  569. """validates input in textctrls, taken from wxpython demo"""
  570. def __init__(self, flag=None):
  571. Validator.__init__(self)
  572. self.flag = flag
  573. self.Bind(wx.EVT_CHAR, self.OnChar)
  574. def Clone(self):
  575. return NTCValidator(self.flag)
  576. def OnChar(self, event):
  577. key = event.GetKeyCode()
  578. if key < wx.WXK_SPACE or key == wx.WXK_DELETE or key > 255:
  579. event.Skip()
  580. return
  581. if self.flag == 'DIGIT_ONLY' and chr(key) in string.digits + '.-':
  582. event.Skip()
  583. return
  584. if not wx.Validator_IsSilent():
  585. wx.Bell()
  586. # Returning without calling even.Skip eats the event before it
  587. # gets to the text control
  588. return
  589. class SimpleValidator(Validator):
  590. """This validator is used to ensure that the user has entered something
  591. into the text object editor dialog's text field.
  592. """
  593. def __init__(self, callback):
  594. """Standard constructor.
  595. """
  596. Validator.__init__(self)
  597. self.callback = callback
  598. def Clone(self):
  599. """Standard cloner.
  600. Note that every validator must implement the Clone() method.
  601. """
  602. return SimpleValidator(self.callback)
  603. def Validate(self, win):
  604. """Validate the contents of the given text control.
  605. """
  606. ctrl = self.GetWindow()
  607. text = ctrl.GetValue()
  608. if len(text) == 0:
  609. self.callback(ctrl)
  610. return False
  611. else:
  612. return True
  613. def TransferToWindow(self):
  614. """Transfer data from validator to window.
  615. The default implementation returns False, indicating that an
  616. error occurred. We simply return True, as we don't do any data
  617. transfer.
  618. """
  619. return True # Prevent wxDialog from complaining.
  620. def TransferFromWindow(self):
  621. """Transfer data from window to validator.
  622. The default implementation returns False, indicating that an
  623. error occurred. We simply return True, as we don't do any data
  624. transfer.
  625. """
  626. return True # Prevent wxDialog from complaining.
  627. class GenericValidator(Validator):
  628. """This validator checks condition and calls callback
  629. in case the condition is not fulfilled.
  630. """
  631. def __init__(self, condition, callback):
  632. """Standard constructor.
  633. :param condition: function which accepts string value and returns T/F
  634. :param callback: function which is called when condition is not fulfilled
  635. """
  636. Validator.__init__(self)
  637. self._condition = condition
  638. self._callback = callback
  639. def Clone(self):
  640. """Standard cloner.
  641. Note that every validator must implement the Clone() method.
  642. """
  643. return GenericValidator(self._condition, self._callback)
  644. def Validate(self, win):
  645. """Validate the contents of the given text control.
  646. """
  647. ctrl = self.GetWindow()
  648. text = ctrl.GetValue()
  649. if not self._condition(text):
  650. self._callback(ctrl)
  651. return False
  652. else:
  653. return True
  654. def TransferToWindow(self):
  655. """Transfer data from validator to window.
  656. """
  657. return True # Prevent wxDialog from complaining.
  658. def TransferFromWindow(self):
  659. """Transfer data from window to validator.
  660. """
  661. return True # Prevent wxDialog from complaining.
  662. class MapValidator(GenericValidator):
  663. """Validator for map name input
  664. See G_legal_filename()
  665. """
  666. def __init__(self):
  667. def _mapNameValidationFailed(ctrl):
  668. message = _(
  669. "Name <%(name)s> is not a valid name for GRASS map. "
  670. "Please use only ASCII characters excluding %(chars)s "
  671. "and space.") % {
  672. 'name': ctrl.GetValue(),
  673. 'chars': '/"\'@,=*~'}
  674. GError(message, caption=_("Invalid name"))
  675. GenericValidator.__init__(self,
  676. grass.legal_name,
  677. _mapNameValidationFailed)
  678. class SingleSymbolPanel(wx.Panel):
  679. """Panel for displaying one symbol.
  680. Changes background when selected. Assumes that parent will catch
  681. events emitted on mouse click. Used in gui_core::dialog::SymbolDialog.
  682. """
  683. def __init__(self, parent, symbolPath):
  684. """Panel constructor
  685. Signal symbolSelectionChanged - symbol selected
  686. - attribute 'name' (symbol name)
  687. - attribute 'doubleClick' (underlying cause)
  688. :param parent: parent (gui_core::dialog::SymbolDialog)
  689. :param symbolPath: absolute path to symbol
  690. """
  691. self.symbolSelectionChanged = Signal(
  692. 'SingleSymbolPanel.symbolSelectionChanged')
  693. wx.Panel.__init__(self, parent, id=wx.ID_ANY, style=wx.BORDER_RAISED)
  694. self.SetName(os.path.splitext(os.path.basename(symbolPath))[0])
  695. self.sBmp = wx.StaticBitmap(self, wx.ID_ANY, wx.Bitmap(symbolPath))
  696. self.selected = False
  697. self.selectColor = wx.SystemSettings.GetColour(wx.SYS_COLOUR_HIGHLIGHT)
  698. self.deselectColor = wx.SystemSettings.GetColour(wx.SYS_COLOUR_WINDOW)
  699. sizer = wx.BoxSizer()
  700. sizer.Add(
  701. self.sBmp,
  702. proportion=0,
  703. flag=wx.ALL | wx.ALIGN_CENTER,
  704. border=5)
  705. self.SetBackgroundColour(self.deselectColor)
  706. self.SetMinSize(self.GetBestSize())
  707. self.SetSizerAndFit(sizer)
  708. # binding to both (staticBitmap, Panel) necessary
  709. self.sBmp.Bind(wx.EVT_LEFT_DOWN, self.OnLeftDown)
  710. self.Bind(wx.EVT_LEFT_DOWN, self.OnLeftDown)
  711. self.Bind(wx.EVT_LEFT_DCLICK, self.OnDoubleClick)
  712. self.sBmp.Bind(wx.EVT_LEFT_DCLICK, self.OnDoubleClick)
  713. def OnLeftDown(self, event):
  714. """Panel selected, background changes"""
  715. self.selected = True
  716. self.SetBackgroundColour(self.selectColor)
  717. self.Refresh()
  718. event.Skip()
  719. self.symbolSelectionChanged.emit(
  720. name=self.GetName(), doubleClick=False)
  721. def OnDoubleClick(self, event):
  722. self.symbolSelectionChanged.emit(name=self.GetName(), doubleClick=True)
  723. def Deselect(self):
  724. """Panel deselected, background changes back to default"""
  725. self.selected = False
  726. self.SetBackgroundColour(self.deselectColor)
  727. self.Refresh()
  728. def Select(self):
  729. """Select panel, no event emitted"""
  730. self.selected = True
  731. self.SetBackgroundColour(self.selectColor)
  732. self.Refresh()
  733. class GListCtrl(ListCtrl, listmix.ListCtrlAutoWidthMixin,
  734. listmix.CheckListCtrlMixin):
  735. """Generic ListCtrl with popup menu to select/deselect all
  736. items"""
  737. def __init__(self, parent):
  738. self.parent = parent
  739. ListCtrl.__init__(self, parent, id=wx.ID_ANY,
  740. style=wx.LC_REPORT)
  741. listmix.CheckListCtrlMixin.__init__(self)
  742. # setup mixins
  743. listmix.ListCtrlAutoWidthMixin.__init__(self)
  744. self.Bind(wx.EVT_COMMAND_RIGHT_CLICK, self.OnPopupMenu) # wxMSW
  745. self.Bind(wx.EVT_RIGHT_UP, self.OnPopupMenu) # wxGTK
  746. def OnPopupMenu(self, event):
  747. """Show popup menu"""
  748. if self.GetItemCount() < 1:
  749. return
  750. if not hasattr(self, "popupDataID1"):
  751. self.popupDataID1 = wx.NewId()
  752. self.popupDataID2 = wx.NewId()
  753. self.Bind(wx.EVT_MENU, self.OnSelectAll, id=self.popupDataID1)
  754. self.Bind(wx.EVT_MENU, self.OnSelectNone, id=self.popupDataID2)
  755. # generate popup-menu
  756. menu = Menu()
  757. menu.Append(self.popupDataID1, _("Select all"))
  758. menu.Append(self.popupDataID2, _("Deselect all"))
  759. self.PopupMenu(menu)
  760. menu.Destroy()
  761. def OnSelectAll(self, event):
  762. """Select all items"""
  763. item = -1
  764. while True:
  765. item = self.GetNextItem(item)
  766. if item == -1:
  767. break
  768. self.CheckItem(item, True)
  769. event.Skip()
  770. def OnSelectNone(self, event):
  771. """Deselect items"""
  772. item = -1
  773. while True:
  774. item = self.GetNextItem(item, wx.LIST_STATE_SELECTED)
  775. if item == -1:
  776. break
  777. self.CheckItem(item, False)
  778. event.Skip()
  779. def GetData(self, checked=None):
  780. """Get list data"""
  781. data = []
  782. checkedList = []
  783. item = -1
  784. while True:
  785. row = []
  786. item = self.GetNextItem(item)
  787. if item == -1:
  788. break
  789. isChecked = self.IsChecked(item)
  790. if checked is not None and checked != isChecked:
  791. continue
  792. checkedList.append(isChecked)
  793. for i in range(self.GetColumnCount()):
  794. row.append(self.GetItem(item, i).GetText())
  795. row.append(item)
  796. data.append(tuple(row))
  797. if checked is not None:
  798. return tuple(data)
  799. else:
  800. return (tuple(data), tuple(checkedList))
  801. def LoadData(self, data=None, selectOne=True):
  802. """Load data into list"""
  803. self.DeleteAllItems()
  804. if data is None:
  805. return
  806. idx = 0
  807. for item in data:
  808. if wxPythonPhoenix:
  809. index = self.InsertItem(idx, str(item[0]))
  810. else:
  811. index = self.InsertStringItem(idx, str(item[0]))
  812. for i in range(1, self.GetColumnCount()):
  813. self.SetStringItem(index, i, item[i])
  814. idx += 1
  815. # check by default only on one item
  816. if len(data) == 1 and selectOne:
  817. self.CheckItem(index, True)
  818. class SearchModuleWidget(wx.Panel):
  819. """Search module widget (used e.g. in SearchModuleWindow)
  820. Signals:
  821. moduleSelected - attribute 'name' is module name
  822. showSearchResult - attribute 'result' is a node (representing module)
  823. showNotification - attribute 'message'
  824. """
  825. def __init__(self, parent, model,
  826. showChoice=True, showTip=False, **kwargs):
  827. self._showTip = showTip
  828. self._showChoice = showChoice
  829. self._model = model
  830. self._results = [] # list of found nodes
  831. self._resultIndex = -1
  832. self._searchKeys = ['description', 'keywords', 'command']
  833. self._oldValue = ''
  834. self.moduleSelected = Signal('SearchModuleWidget.moduleSelected')
  835. self.showSearchResult = Signal('SearchModuleWidget.showSearchResult')
  836. self.showNotification = Signal('SearchModuleWidget.showNotification')
  837. wx.Panel.__init__(self, parent=parent, id=wx.ID_ANY, **kwargs)
  838. if sys.platform == 'win32':
  839. self._search = TextCtrl(
  840. parent=self, id=wx.ID_ANY, size=(-1, 25),
  841. style=wx.TE_PROCESS_ENTER)
  842. else:
  843. self._search = SearchCtrl(
  844. parent=self, id=wx.ID_ANY, size=(-1, 25),
  845. style=wx.TE_PROCESS_ENTER)
  846. self._search.SetDescriptiveText(_('Fulltext search'))
  847. self._search.SetToolTip(
  848. _("Type to search in all modules. Press Enter for next match."))
  849. self._search.Bind(wx.EVT_TEXT, self.OnSearchModule)
  850. self._search.Bind(wx.EVT_TEXT_ENTER, self.OnEnter)
  851. if self._showTip:
  852. self._searchTip = StaticWrapText(parent=self, id=wx.ID_ANY,
  853. label="Choose a module", size=(-1, 35))
  854. if self._showChoice:
  855. self._searchChoice = wx.Choice(parent=self, id=wx.ID_ANY)
  856. self._searchChoice.SetItems(
  857. self._searchModule(
  858. keys=['command'], value=''))
  859. self._searchChoice.Bind(wx.EVT_CHOICE, self.OnSelectModule)
  860. self._layout()
  861. def _layout(self):
  862. """Do layout"""
  863. sizer = wx.BoxSizer(wx.HORIZONTAL)
  864. boxSizer = wx.BoxSizer(wx.VERTICAL)
  865. boxSizer.Add(self._search,
  866. flag=wx.ALIGN_CENTER_VERTICAL | wx.EXPAND | wx.BOTTOM,
  867. border=5)
  868. if self._showChoice:
  869. hSizer = wx.BoxSizer(wx.HORIZONTAL)
  870. hSizer.Add(self._searchChoice,
  871. flag=wx.ALIGN_CENTER_VERTICAL | wx.EXPAND | wx.BOTTOM,
  872. border=5)
  873. hSizer.AddStretchSpacer()
  874. boxSizer.Add(hSizer, flag=wx.EXPAND)
  875. if self._showTip:
  876. boxSizer.Add(self._searchTip,
  877. flag=wx.ALIGN_CENTER_VERTICAL | wx.EXPAND)
  878. sizer.Add(boxSizer, proportion=1)
  879. self.SetSizer(sizer)
  880. sizer.Fit(self)
  881. def OnEnter(self, event):
  882. """Process EVT_TEXT_ENTER to show search results"""
  883. self._showSearchResult()
  884. event.Skip()
  885. def _showSearchResult(self):
  886. if self._results:
  887. self._resultIndex += 1
  888. if self._resultIndex == len(self._results):
  889. self._resultIndex = 0
  890. self.showSearchResult.emit(result=self._results[self._resultIndex])
  891. def OnSearchModule(self, event):
  892. """Search module by keywords or description"""
  893. value = self._search.GetValue()
  894. if value == self._oldValue:
  895. event.Skip()
  896. return
  897. self._oldValue = value
  898. if len(value) <= 2:
  899. if len(value) == 0: # reset
  900. commands = self._searchModule(keys=['command'], value='')
  901. else:
  902. self.showNotification.emit(
  903. message=_("Searching, please type more characters."))
  904. return
  905. else:
  906. commands = self._searchModule(keys=self._searchKeys, value=value)
  907. if self._showChoice:
  908. self._searchChoice.SetItems(commands)
  909. if commands:
  910. self._searchChoice.SetSelection(0)
  911. label = _("%d modules match") % len(commands)
  912. if self._showTip:
  913. self._searchTip.SetLabel(label)
  914. self.showNotification.emit(message=label)
  915. event.Skip()
  916. def _searchModule(self, keys, value):
  917. """Search modules by keys
  918. :param keys: list of keys
  919. :param value: patter to match
  920. """
  921. nodes = set()
  922. for key in keys:
  923. nodes.update(self._model.SearchNodes(key=key, value=value))
  924. nodes = list(nodes)
  925. nodes.sort(key=lambda node: self._model.GetIndexOfNode(node))
  926. self._results = nodes
  927. self._resultIndex = -1
  928. commands = sorted([node.data['command']
  929. for node in nodes if node.data['command']])
  930. return commands
  931. def OnSelectModule(self, event):
  932. """Module selected from choice, update command prompt"""
  933. cmd = self._searchChoice.GetStringSelection()
  934. self.moduleSelected.emit(name=cmd)
  935. if self._showTip:
  936. for module in self._results:
  937. if cmd == module.data['command']:
  938. self._searchTip.SetLabel(module.data['description'])
  939. break
  940. def Reset(self):
  941. """Reset widget"""
  942. self._search.SetValue('')
  943. if self._showTip:
  944. self._searchTip.SetLabel('Choose a module')
  945. class ManageSettingsWidget(wx.Panel):
  946. """Widget which allows loading and saving settings into file."""
  947. def __init__(self, parent, settingsFile):
  948. """
  949. Signals:
  950. settingsChanged - called when users changes setting
  951. - attribute 'data' with chosen setting data
  952. settingsSaving - called when settings are saving
  953. - attribute 'name' with chosen settings name
  954. settingsLoaded - called when settings are loaded
  955. - attribute 'settings' is dict with loaded settings
  956. {nameofsetting : settingdata, ....}
  957. :param settingsFile: path to file, where settings will be saved and loaded from
  958. """
  959. self.settingsFile = settingsFile
  960. self.settingsChanged = Signal('ManageSettingsWidget.settingsChanged')
  961. self.settingsSaving = Signal('ManageSettingsWidget.settingsSaving')
  962. self.settingsLoaded = Signal('ManageSettingsWidget.settingsLoaded')
  963. wx.Panel.__init__(self, parent=parent, id=wx.ID_ANY)
  964. self.settingsBox = StaticBox(parent=self, id=wx.ID_ANY,
  965. label=" %s " % _("Profiles"))
  966. self.settingsChoice = wx.Choice(parent=self, id=wx.ID_ANY)
  967. self.settingsChoice.Bind(wx.EVT_CHOICE, self.OnSettingsChanged)
  968. self.btnSettingsSave = Button(parent=self, id=wx.ID_SAVE)
  969. self.btnSettingsSave.Bind(wx.EVT_BUTTON, self.OnSettingsSave)
  970. self.btnSettingsSave.SetToolTip(_("Save current settings"))
  971. self.btnSettingsDel = Button(parent=self, id=wx.ID_REMOVE)
  972. self.btnSettingsDel.Bind(wx.EVT_BUTTON, self.OnSettingsDelete)
  973. self.btnSettingsSave.SetToolTip(
  974. _("Delete currently selected settings"))
  975. # escaping with '$' character - index in self.esc_chars
  976. self.e_char_i = 0
  977. self.esc_chars = ['$', ';']
  978. self._settings = self._loadSettings() # -> self.settingsChoice.SetItems()
  979. self.settingsLoaded.emit(settings=self._settings)
  980. self.data_to_save = []
  981. self._layout()
  982. self.SetSizer(self.settingsSizer)
  983. self.settingsSizer.Fit(self)
  984. def _layout(self):
  985. self.settingsSizer = wx.StaticBoxSizer(self.settingsBox, wx.HORIZONTAL)
  986. self.settingsSizer.Add(
  987. StaticText(
  988. parent=self,
  989. id=wx.ID_ANY,
  990. label=_("Load:")),
  991. flag=wx.ALIGN_CENTER_VERTICAL | wx.RIGHT | wx.LEFT,
  992. border=5)
  993. self.settingsSizer.Add(
  994. self.settingsChoice,
  995. proportion=1,
  996. flag=wx.EXPAND | wx.BOTTOM,
  997. border=3)
  998. self.settingsSizer.Add(self.btnSettingsSave,
  999. flag=wx.LEFT | wx.RIGHT | wx.BOTTOM, border=3)
  1000. self.settingsSizer.Add(self.btnSettingsDel,
  1001. flag=wx.RIGHT | wx.BOTTOM, border=3)
  1002. def OnSettingsChanged(self, event):
  1003. """Load named settings"""
  1004. name = event.GetString()
  1005. if name not in self._settings:
  1006. GError(parent=self,
  1007. message=_("Settings <%s> not found") % name)
  1008. return
  1009. data = self._settings[name]
  1010. self.settingsChanged.emit(data=data)
  1011. def GetSettings(self):
  1012. """Load named settings"""
  1013. return self._settings.copy()
  1014. def OnSettingsSave(self, event):
  1015. """Save settings"""
  1016. dlg = wx.TextEntryDialog(parent=self,
  1017. message=_("Name:"),
  1018. caption=_("Save settings"))
  1019. if dlg.ShowModal() == wx.ID_OK:
  1020. name = dlg.GetValue()
  1021. if not name:
  1022. GMessage(parent=self,
  1023. message=_("Name not given, settings is not saved."))
  1024. else:
  1025. self.settingsSaving.emit(name=name)
  1026. dlg.Destroy()
  1027. def SaveSettings(self, name):
  1028. # check if settings item already exists
  1029. if name in self._settings:
  1030. dlgOwt = wx.MessageDialog(
  1031. self,
  1032. message=_(
  1033. "Settings <%s> already exists. "
  1034. "Do you want to overwrite the settings?") %
  1035. name,
  1036. caption=_("Save settings"),
  1037. style=wx.YES_NO | wx.YES_DEFAULT | wx.ICON_QUESTION)
  1038. if dlgOwt.ShowModal() != wx.ID_YES:
  1039. dlgOwt.Destroy()
  1040. return
  1041. if self.data_to_save:
  1042. self._settings[name] = self.data_to_save
  1043. self._saveSettings()
  1044. self.settingsChoice.SetStringSelection(name)
  1045. self.data_to_save = []
  1046. def _saveSettings(self):
  1047. """Save settings and reload if successful"""
  1048. if self._writeSettings() == 0:
  1049. self._settings = self._loadSettings()
  1050. def SetDataToSave(self, data):
  1051. """Set data for setting, which will be saved.
  1052. :param data: - list of strings, which will be saved
  1053. """
  1054. self.data_to_save = data
  1055. def SetSettings(self, settings):
  1056. """Set settings
  1057. :param settings: - dict with all settigs {nameofsetting : settingdata, ....}
  1058. """
  1059. self._settings = settings
  1060. self._saveSettings()
  1061. def AddSettings(self, settings):
  1062. """Add settings
  1063. :param settings: - dict with all settigs {nameofsetting : settingdata, ....}
  1064. """
  1065. self._settings = dict(self._settings.items() + settings.items())
  1066. self._saveSettings()
  1067. def OnSettingsDelete(self, event):
  1068. """Save settings
  1069. """
  1070. name = self.settingsChoice.GetStringSelection()
  1071. if not name:
  1072. GMessage(parent=self,
  1073. message=_("No settings is defined. Operation canceled."))
  1074. return
  1075. self._settings.pop(name)
  1076. if self._writeSettings() == 0:
  1077. self._settings = self._loadSettings()
  1078. def _writeSettings(self):
  1079. """Save settings into the file
  1080. :return: 0 on success
  1081. :return: -1 on failure
  1082. """
  1083. try:
  1084. fd = open(self.settingsFile, 'w')
  1085. fd.write('format_version=2.0\n')
  1086. for key, values in six.iteritems(self._settings):
  1087. first = True
  1088. for v in values:
  1089. # escaping characters
  1090. for e_ch in self.esc_chars:
  1091. v = v.replace(
  1092. e_ch, self.esc_chars[
  1093. self.e_char_i] + e_ch)
  1094. if first:
  1095. # escaping characters
  1096. for e_ch in self.esc_chars:
  1097. key = key.replace(
  1098. e_ch, self.esc_chars[
  1099. self.e_char_i] + e_ch)
  1100. fd.write('%s;%s;' % (key, v))
  1101. first = False
  1102. else:
  1103. fd.write('%s;' % (v))
  1104. fd.write('\n')
  1105. except IOError:
  1106. GError(parent=self,
  1107. message=_("Unable to save settings"))
  1108. return -1
  1109. fd.close()
  1110. return 0
  1111. def _loadSettings(self):
  1112. """Load settings from the file
  1113. The file is defined by self.SettingsFile.
  1114. :return: parsed dict
  1115. :return: empty dict on error
  1116. """
  1117. data = dict()
  1118. if not os.path.exists(self.settingsFile):
  1119. return data
  1120. try:
  1121. fd = open(self.settingsFile, 'r')
  1122. except IOError:
  1123. return data
  1124. fd_lines = fd.readlines()
  1125. if not fd_lines:
  1126. fd.close()
  1127. return data
  1128. if fd_lines[0].strip() == 'format_version=2.0':
  1129. data = self._loadSettings_v2(fd_lines)
  1130. else:
  1131. data = self._loadSettings_v1(fd_lines)
  1132. self.settingsChoice.SetItems(sorted(data.keys()))
  1133. fd.close()
  1134. self.settingsLoaded.emit(settings=data)
  1135. return data
  1136. def _loadSettings_v2(self, fd_lines):
  1137. """Load settings from the file in format version 2.0
  1138. The file is defined by self.SettingsFile.
  1139. :return: parsed dict
  1140. :return: empty dict on error
  1141. """
  1142. data = dict()
  1143. for line in fd_lines[1:]:
  1144. try:
  1145. lineData = []
  1146. line = line.rstrip('\n')
  1147. i_last_found = i_last = 0
  1148. key = ''
  1149. while True:
  1150. idx = line.find(';', i_last)
  1151. if idx < 0:
  1152. break
  1153. elif idx != 0:
  1154. # find out whether it is separator
  1155. # $$$$; - it is separator
  1156. # $$$$$; - it is not separator
  1157. i_esc_chars = 0
  1158. while True:
  1159. if line[idx - (i_esc_chars + 1)
  1160. ] == self.esc_chars[self.e_char_i]:
  1161. i_esc_chars += 1
  1162. else:
  1163. break
  1164. if i_esc_chars % 2 != 0:
  1165. i_last = idx + 1
  1166. continue
  1167. lineItem = line[i_last_found: idx]
  1168. # unescape characters
  1169. for e_ch in self.esc_chars:
  1170. lineItem = lineItem.replace(
  1171. self.esc_chars[self.e_char_i] + e_ch, e_ch)
  1172. if i_last_found == 0:
  1173. key = lineItem
  1174. else:
  1175. lineData.append(lineItem)
  1176. i_last_found = i_last = idx + 1
  1177. if key and lineData:
  1178. data[key] = lineData
  1179. except ValueError:
  1180. pass
  1181. return data
  1182. def _loadSettings_v1(self, fd_lines):
  1183. """Load settings from the file in format version 1.0 (backward compatibility)
  1184. The file is defined by self.SettingsFile.
  1185. :return: parsed dict
  1186. :return: empty dict on error
  1187. """
  1188. data = dict()
  1189. for line in fd_lines:
  1190. try:
  1191. lineData = line.rstrip('\n').split(';')
  1192. if len(lineData) > 4:
  1193. # type, dsn, format, options
  1194. data[
  1195. lineData[0]] = (
  1196. lineData[1],
  1197. lineData[2],
  1198. lineData[3],
  1199. lineData[4])
  1200. else:
  1201. data[
  1202. lineData[0]] = (
  1203. lineData[1],
  1204. lineData[2],
  1205. lineData[3],
  1206. '')
  1207. except ValueError:
  1208. pass
  1209. return data
  1210. class PictureComboBox(OwnerDrawnComboBox):
  1211. """Abstract class of ComboBox with pictures.
  1212. Derived class has to specify has to specify _getPath method.
  1213. """
  1214. def OnDrawItem(self, dc, rect, item, flags):
  1215. """Overridden from OwnerDrawnComboBox.
  1216. Called to draw each item in the list.
  1217. """
  1218. if item == wx.NOT_FOUND:
  1219. # painting the control, but there is no valid item selected yet
  1220. return
  1221. r = Rect(*rect) # make a copy
  1222. r.Deflate(3, 5)
  1223. # for painting the items in the popup
  1224. bitmap = self.GetPictureBitmap(self.GetString(item))
  1225. if bitmap:
  1226. dc.DrawBitmap(
  1227. bitmap, r.x, r.y + (r.height - bitmap.GetHeight()) / 2)
  1228. width = bitmap.GetWidth() + 10
  1229. else:
  1230. width = 0
  1231. dc.DrawText(self.GetString(item),
  1232. r.x + width,
  1233. (r.y + 0) + (r.height - dc.GetCharHeight()) / 2)
  1234. def OnMeasureItem(self, item):
  1235. """Overridden from OwnerDrawnComboBox, should return the height.
  1236. Needed to display an item in the popup, or -1 for default.
  1237. """
  1238. return 24
  1239. def GetPictureBitmap(self, name):
  1240. """Returns bitmap for given picture name.
  1241. :param str colorTable: name of color table
  1242. """
  1243. if not hasattr(self, 'bitmaps'):
  1244. self.bitmaps = {}
  1245. if name in self.bitmaps:
  1246. return self.bitmaps[name]
  1247. path = self._getPath(name)
  1248. if os.path.exists(path):
  1249. bitmap = wx.Bitmap(path)
  1250. self.bitmaps[name] = bitmap
  1251. return bitmap
  1252. return None
  1253. class ColorTablesComboBox(PictureComboBox):
  1254. """ComboBox with drawn color tables (created by thumbnails.py).
  1255. Used in r(3).colors dialog."""
  1256. def _getPath(self, name):
  1257. return os.path.join(
  1258. os.getenv("GISBASE"),
  1259. "docs", "html", "colortables", "%s.png" % name)
  1260. class BarscalesComboBox(PictureComboBox):
  1261. """ComboBox with barscales for d.barscale."""
  1262. def _getPath(self, name):
  1263. return os.path.join(
  1264. os.getenv("GISBASE"),
  1265. "docs", "html", "barscales", name + '.png')
  1266. class NArrowsComboBox(PictureComboBox):
  1267. """ComboBox with north arrows for d.barscale."""
  1268. def _getPath(self, name):
  1269. return os.path.join(
  1270. os.getenv("GISBASE"),
  1271. "docs", "html", "northarrows", "%s.png" % name)
  1272. class LayersList(GListCtrl, listmix.TextEditMixin):
  1273. """List of layers to be imported (dxf, shp...)"""
  1274. def __init__(self, parent, columns, log=None):
  1275. GListCtrl.__init__(self, parent)
  1276. self.log = log
  1277. # setup mixins
  1278. listmix.TextEditMixin.__init__(self)
  1279. for i in range(len(columns)):
  1280. self.InsertColumn(i, columns[i])
  1281. width = []
  1282. if len(columns) == 3:
  1283. width = (65, 200)
  1284. elif len(columns) == 4:
  1285. width = (65, 200, 90)
  1286. elif len(columns) == 5:
  1287. width = (65, 180, 90, 70)
  1288. for i in range(len(width)):
  1289. self.SetColumnWidth(col=i, width=width[i])
  1290. def OnLeftDown(self, event):
  1291. """Allow editing only output name
  1292. Code taken from TextEditMixin class.
  1293. """
  1294. x, y = event.GetPosition()
  1295. colLocs = [0]
  1296. loc = 0
  1297. for n in range(self.GetColumnCount()):
  1298. loc = loc + self.GetColumnWidth(n)
  1299. colLocs.append(loc)
  1300. col = bisect(colLocs, x + self.GetScrollPos(wx.HORIZONTAL)) - 1
  1301. if col == self.GetColumnCount() - 1:
  1302. listmix.TextEditMixin.OnLeftDown(self, event)
  1303. else:
  1304. event.Skip()
  1305. def GetLayers(self):
  1306. """Get list of layers (layer name, output name, list id)"""
  1307. layers = []
  1308. data = self.GetData(checked=True)
  1309. for itm in data:
  1310. layer = itm[1]
  1311. ftype = itm[2]
  1312. if '/' in ftype:
  1313. layer += '|%s' % ftype.split('/', 1)[0]
  1314. output = itm[self.GetColumnCount() - 1]
  1315. layers.append((layer, output, itm[-1]))
  1316. return layers