widgets.py 45 KB

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