dbm_dialogs.py 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671
  1. """!
  2. @package dbm_dialogs.py
  3. @brief DBM-related dialogs
  4. List of classes:
  5. - DisplayAttributesDialog
  6. - ModifyTableRecord
  7. (C) 2007-2011 by the GRASS Development Team
  8. This program is free software under the GNU General Public
  9. License (>=v2). Read the file COPYING that comes with GRASS
  10. for details.
  11. @author Martin Landa <landa.martin gmail.com>
  12. """
  13. import os
  14. import globalvar
  15. import wx
  16. import wx.lib.scrolledpanel as scrolled
  17. import gcmd
  18. from debug import Debug
  19. from preferences import globalSettings as UserSettings
  20. from dbm_base import VectorDBInfo
  21. class DisplayAttributesDialog(wx.Dialog):
  22. def __init__(self, parent, map,
  23. query = None, cats = None, line = None,
  24. style = wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER,
  25. pos = wx.DefaultPosition,
  26. action = "add", ignoreError = False):
  27. """!Standard dialog used to add/update/display attributes linked
  28. to the vector map.
  29. Attribute data can be selected based on layer and category number
  30. or coordinates.
  31. @param parent
  32. @param map vector map
  33. @param query query coordinates and distance (used for v.edit)
  34. @param cats {layer: cats}
  35. @param line feature id (requested for cats)
  36. @param style
  37. @param pos
  38. @param action (add, update, display)
  39. @param ignoreError True to ignore errors
  40. """
  41. self.parent = parent # mapdisplay.BufferedWindow
  42. self.map = map
  43. self.action = action
  44. # ids/cats of selected features
  45. # fid : {layer : cats}
  46. self.cats = {}
  47. self.fid = -1 # feature id
  48. # get layer/table/column information
  49. self.mapDBInfo = VectorDBInfo(self.map)
  50. layers = self.mapDBInfo.layers.keys() # get available layers
  51. # check if db connection / layer exists
  52. if len(layers) <= 0:
  53. if not ignoreError:
  54. dlg = wx.MessageDialog(parent = self.parent,
  55. message = _("No attribute table found.\n\n"
  56. "Do you want to create a new attribute table "
  57. "and defined a link to vector map <%s>?") % self.map,
  58. caption = _("Create table?"),
  59. style = wx.YES_NO | wx.NO_DEFAULT | wx.ICON_QUESTION)
  60. if dlg.ShowModal() == wx.ID_YES:
  61. lmgr = self.parent.lmgr
  62. lmgr.OnShowAttributeTable(event = None, selection = 'layers')
  63. dlg.Destroy()
  64. self.mapDBInfo = None
  65. wx.Dialog.__init__(self, parent = self.parent, id = wx.ID_ANY,
  66. title = "", style = style, pos = pos)
  67. # dialog body
  68. mainSizer = wx.BoxSizer(wx.VERTICAL)
  69. # notebook
  70. self.notebook = wx.Notebook(parent = self, id = wx.ID_ANY, style = wx.BK_DEFAULT)
  71. self.closeDialog = wx.CheckBox(parent = self, id = wx.ID_ANY,
  72. label = _("Close dialog on submit"))
  73. self.closeDialog.SetValue(True)
  74. if self.action == 'display':
  75. self.closeDialog.Enable(False)
  76. # feature id (text/choice for duplicates)
  77. self.fidMulti = wx.Choice(parent = self, id = wx.ID_ANY,
  78. size = (150, -1))
  79. self.fidMulti.Bind(wx.EVT_CHOICE, self.OnFeature)
  80. self.fidText = wx.StaticText(parent = self, id = wx.ID_ANY)
  81. self.noFoundMsg = wx.StaticText(parent = self, id = wx.ID_ANY,
  82. label = _("No attributes found"))
  83. self.UpdateDialog(query = query, cats = cats)
  84. # set title
  85. if self.action == "update":
  86. self.SetTitle(_("Update attributes"))
  87. elif self.action == "add":
  88. self.SetTitle(_("Define attributes"))
  89. else:
  90. self.SetTitle(_("Display attributes"))
  91. # buttons
  92. btnCancel = wx.Button(self, wx.ID_CANCEL)
  93. btnReset = wx.Button(self, wx.ID_UNDO, _("&Reload"))
  94. btnSubmit = wx.Button(self, wx.ID_OK, _("&Submit"))
  95. if self.action == 'display':
  96. btnSubmit.Enable(False)
  97. btnSizer = wx.StdDialogButtonSizer()
  98. btnSizer.AddButton(btnCancel)
  99. btnSizer.AddButton(btnReset)
  100. btnSizer.SetNegativeButton(btnReset)
  101. btnSubmit.SetDefault()
  102. btnSizer.AddButton(btnSubmit)
  103. btnSizer.Realize()
  104. mainSizer.Add(item = self.noFoundMsg, proportion = 0,
  105. flag = wx.EXPAND | wx.ALL, border = 5)
  106. mainSizer.Add(item = self.notebook, proportion = 1,
  107. flag = wx.EXPAND | wx.ALL, border = 5)
  108. fidSizer = wx.BoxSizer(wx.HORIZONTAL)
  109. fidSizer.Add(item = wx.StaticText(parent = self, id = wx.ID_ANY,
  110. label = _("Feature id:")),
  111. proportion = 0, border = 5,
  112. flag = wx.ALIGN_CENTER_VERTICAL)
  113. fidSizer.Add(item = self.fidMulti, proportion = 0,
  114. flag = wx.EXPAND | wx.ALL, border = 5)
  115. fidSizer.Add(item = self.fidText, proportion = 0,
  116. flag = wx.EXPAND | wx.ALL, border = 5)
  117. mainSizer.Add(item = fidSizer, proportion = 0,
  118. flag = wx.EXPAND | wx.LEFT | wx.RIGHT, border = 5)
  119. mainSizer.Add(item = self.closeDialog, proportion = 0, flag = wx.EXPAND | wx.LEFT | wx.RIGHT,
  120. border = 5)
  121. mainSizer.Add(item = btnSizer, proportion = 0,
  122. flag = wx.EXPAND | wx.ALL | wx.ALIGN_CENTER, border = 5)
  123. # bindigs
  124. btnReset.Bind(wx.EVT_BUTTON, self.OnReset)
  125. btnSubmit.Bind(wx.EVT_BUTTON, self.OnSubmit)
  126. btnCancel.Bind(wx.EVT_BUTTON, self.OnCancel)
  127. self.SetSizer(mainSizer)
  128. mainSizer.Fit(self)
  129. # set min size for dialog
  130. w, h = self.GetBestSize()
  131. if h < 200:
  132. self.SetMinSize((w, 200))
  133. else:
  134. self.SetMinSize(self.GetBestSize())
  135. if self.notebook.GetPageCount() == 0:
  136. Debug.msg(2, "DisplayAttributesDialog(): Nothing found!")
  137. ### self.mapDBInfo = None
  138. def OnSQLStatement(self, event):
  139. """!Update SQL statement"""
  140. pass
  141. def IsFound(self):
  142. """!Check for status
  143. @return True on attributes found
  144. @return False attributes not found
  145. """
  146. return bool(self.mapDBInfo and self.notebook.GetPageCount() > 0)
  147. def GetSQLString(self, updateValues = False):
  148. """!Create SQL statement string based on self.sqlStatement
  149. If updateValues is True, update dataFrame according to values
  150. in textfields.
  151. """
  152. sqlCommands = []
  153. # find updated values for each layer/category
  154. for layer in self.mapDBInfo.layers.keys(): # for each layer
  155. table = self.mapDBInfo.GetTable(layer)
  156. key = self.mapDBInfo.GetKeyColumn(layer)
  157. columns = self.mapDBInfo.GetTableDesc(table)
  158. for idx in range(len(columns[key]['values'])): # for each category
  159. updatedColumns = []
  160. updatedValues = []
  161. for name in columns.keys():
  162. if name == key:
  163. cat = columns[name]['values'][idx]
  164. continue
  165. type = columns[name]['type']
  166. value = columns[name]['values'][idx]
  167. id = columns[name]['ids'][idx]
  168. try:
  169. newvalue = self.FindWindowById(id).GetValue()
  170. except:
  171. newvalue = self.FindWindowById(id).GetLabel()
  172. if newvalue == '':
  173. newvalue = None
  174. if newvalue != value:
  175. updatedColumns.append(name)
  176. if newvalue is None:
  177. updatedValues.append('NULL')
  178. else:
  179. if type != 'character':
  180. updatedValues.append(newvalue)
  181. else:
  182. updatedValues.append("'" + newvalue + "'")
  183. columns[name]['values'][idx] = newvalue
  184. if self.action != "add" and len(updatedValues) == 0:
  185. continue
  186. if self.action == "add":
  187. sqlString = "INSERT INTO %s (%s," % (table, key)
  188. else:
  189. sqlString = "UPDATE %s SET " % table
  190. for idx in range(len(updatedColumns)):
  191. name = updatedColumns[idx]
  192. if self.action == "add":
  193. sqlString += name + ","
  194. else:
  195. sqlString += name + "=" + updatedValues[idx] + ","
  196. sqlString = sqlString[:-1] # remove last comma
  197. if self.action == "add":
  198. sqlString += ") VALUES (%s," % cat
  199. for value in updatedValues:
  200. sqlString += str(value) + ","
  201. sqlString = sqlString[:-1] # remove last comma
  202. sqlString += ")"
  203. else:
  204. sqlString += " WHERE %s=%s" % (key, cat)
  205. sqlCommands.append(sqlString)
  206. # for each category
  207. # for each layer END
  208. Debug.msg(3, "DisplayAttributesDialog.GetSQLString(): %s" % sqlCommands)
  209. return sqlCommands
  210. def OnReset(self, event = None):
  211. """!Reset form"""
  212. for layer in self.mapDBInfo.layers.keys():
  213. table = self.mapDBInfo.layers[layer]["table"]
  214. key = self.mapDBInfo.layers[layer]["key"]
  215. columns = self.mapDBInfo.tables[table]
  216. for idx in range(len(columns[key]['values'])):
  217. for name in columns.keys():
  218. type = columns[name]['type']
  219. value = columns[name]['values'][idx]
  220. if value is None:
  221. value = ''
  222. try:
  223. id = columns[name]['ids'][idx]
  224. except IndexError:
  225. id = wx.NOT_FOUND
  226. if name != key and id != wx.NOT_FOUND:
  227. self.FindWindowById(id).SetValue(str(value))
  228. def OnCancel(self, event):
  229. """!Cancel button pressed
  230. """
  231. self.parent.parent.dialogs['attributes'] = None
  232. if hasattr(self, "digit"):
  233. self.parent.digit.GetDisplay().SetSelected([])
  234. self.parent.UpdateMap(render = False)
  235. else:
  236. self.parent.parent.OnRender(None)
  237. self.Close()
  238. def OnSubmit(self, event):
  239. """!Submit records"""
  240. layer = 1
  241. for sql in self.GetSQLString(updateValues = True):
  242. enc = UserSettings.Get(group = 'atm', key = 'encoding', subkey = 'value')
  243. if not enc and 'GRASS_DB_ENCODING' in os.environ:
  244. enc = os.environ['GRASS_DB_ENCODING']
  245. if enc:
  246. sql = sql.encode(enc)
  247. driver, database = self.mapDBInfo.GetDbSettings(layer)
  248. Debug.msg(1, "SQL: %s" % sql)
  249. gcmd.RunCommand('db.execute',
  250. parent = self,
  251. quiet = True,
  252. input = '-',
  253. stdin = sql,
  254. driver = driver,
  255. database = database)
  256. layer += 1
  257. if self.closeDialog.IsChecked():
  258. self.OnCancel(event)
  259. def OnFeature(self, event):
  260. self.fid = int(event.GetString())
  261. self.UpdateDialog(cats = self.cats, fid = self.fid)
  262. def GetCats(self):
  263. """!Get id of selected vector object or 'None' if nothing selected
  264. @param id if true return ids otherwise cats
  265. """
  266. if self.fid < 0:
  267. return None
  268. return self.cats[self.fid]
  269. def GetFid(self):
  270. """!Get selected feature id"""
  271. return self.fid
  272. def UpdateDialog(self, map = None, query = None, cats = None, fid = -1,
  273. action = None):
  274. """!Update dialog
  275. @param map name of vector map
  276. @param query
  277. @param cats
  278. @param fid feature id
  279. @param action add, update, display or None
  280. @return True if updated
  281. @return False
  282. """
  283. if action:
  284. self.action = action
  285. if action == 'display':
  286. enabled = False
  287. else:
  288. enabled = True
  289. self.closeDialog.Enable(enabled)
  290. self.FindWindowById(wx.ID_OK).Enable(enabled)
  291. if map:
  292. self.map = map
  293. # get layer/table/column information
  294. self.mapDBInfo = VectorDBInfo(self.map)
  295. if not self.mapDBInfo:
  296. return False
  297. self.mapDBInfo.Reset()
  298. layers = self.mapDBInfo.layers.keys() # get available layers
  299. # id of selected line
  300. if query: # select by position
  301. data = self.mapDBInfo.SelectByPoint(query[0],
  302. query[1])
  303. self.cats = {}
  304. if data and 'Layer' in data:
  305. idx = 0
  306. for layer in data['Layer']:
  307. layer = int(layer)
  308. if 'Id' in data:
  309. tfid = int(data['Id'][idx])
  310. else:
  311. tfid = 0 # Area / Volume
  312. if not tfid in self.cats:
  313. self.cats[tfid] = {}
  314. if not layer in self.cats[tfid]:
  315. self.cats[tfid][layer] = []
  316. cat = int(data['Category'][idx])
  317. self.cats[tfid][layer].append(cat)
  318. idx += 1
  319. else:
  320. self.cats = cats
  321. if fid > 0:
  322. self.fid = fid
  323. elif len(self.cats.keys()) > 0:
  324. self.fid = self.cats.keys()[0]
  325. else:
  326. self.fid = -1
  327. if len(self.cats.keys()) == 1:
  328. self.fidMulti.Show(False)
  329. self.fidText.Show(True)
  330. if self.fid > 0:
  331. self.fidText.SetLabel("%d" % self.fid)
  332. else:
  333. self.fidText.SetLabel(_("Unknown"))
  334. else:
  335. self.fidMulti.Show(True)
  336. self.fidText.Show(False)
  337. choices = []
  338. for tfid in self.cats.keys():
  339. choices.append(str(tfid))
  340. self.fidMulti.SetItems(choices)
  341. self.fidMulti.SetStringSelection(str(self.fid))
  342. # reset notebook
  343. self.notebook.DeleteAllPages()
  344. for layer in layers: # for each layer
  345. if not query: # select by layer/cat
  346. if self.fid > 0 and layer in self.cats[self.fid]:
  347. for cat in self.cats[self.fid][layer]:
  348. nselected = self.mapDBInfo.SelectFromTable(layer,
  349. where = "%s=%d" % \
  350. (self.mapDBInfo.layers[layer]['key'],
  351. cat))
  352. else:
  353. nselected = 0
  354. # if nselected <= 0 and self.action != "add":
  355. # continue # nothing selected ...
  356. if self.action == "add":
  357. if nselected <= 0:
  358. if layer in self.cats[self.fid]:
  359. table = self.mapDBInfo.layers[layer]["table"]
  360. key = self.mapDBInfo.layers[layer]["key"]
  361. columns = self.mapDBInfo.tables[table]
  362. for name in columns.keys():
  363. if name == key:
  364. for cat in self.cats[self.fid][layer]:
  365. self.mapDBInfo.tables[table][name]['values'].append(cat)
  366. else:
  367. self.mapDBInfo.tables[table][name]['values'].append(None)
  368. else: # change status 'add' -> 'update'
  369. self.action = "update"
  370. table = self.mapDBInfo.layers[layer]["table"]
  371. key = self.mapDBInfo.layers[layer]["key"]
  372. columns = self.mapDBInfo.tables[table]
  373. for idx in range(len(columns[key]['values'])):
  374. for name in columns.keys():
  375. if name == key:
  376. cat = int(columns[name]['values'][idx])
  377. break
  378. # use scrolled panel instead (and fix initial max height of the window to 480px)
  379. panel = scrolled.ScrolledPanel(parent = self.notebook, id = wx.ID_ANY,
  380. size = (-1, 150))
  381. panel.SetupScrolling(scroll_x = False)
  382. self.notebook.AddPage(page = panel, text = " %s %d / %s %d" % (_("Layer"), layer,
  383. _("Category"), cat))
  384. # notebook body
  385. border = wx.BoxSizer(wx.VERTICAL)
  386. flexSizer = wx.FlexGridSizer (cols = 4, hgap = 3, vgap = 3)
  387. flexSizer.AddGrowableCol(3)
  388. # columns (sorted by index)
  389. names = [''] * len(columns.keys())
  390. for name in columns.keys():
  391. names[columns[name]['index']] = name
  392. for name in names:
  393. if name == key: # skip key column (category)
  394. continue
  395. vtype = columns[name]['type']
  396. if columns[name]['values'][idx] is not None:
  397. if columns[name]['ctype'] != type(''):
  398. value = str(columns[name]['values'][idx])
  399. else:
  400. value = columns[name]['values'][idx]
  401. else:
  402. value = ''
  403. colName = wx.StaticText(parent = panel, id = wx.ID_ANY,
  404. label = name)
  405. colType = wx.StaticText(parent = panel, id = wx.ID_ANY,
  406. label = "[" + vtype.lower() + "]")
  407. delimiter = wx.StaticText(parent = panel, id = wx.ID_ANY, label = ":")
  408. colValue = wx.TextCtrl(parent = panel, id = wx.ID_ANY, value = value)
  409. colValue.SetName(name)
  410. self.Bind(wx.EVT_TEXT, self.OnSQLStatement, colValue)
  411. if self.action == 'display':
  412. colValue.SetWindowStyle(wx.TE_READONLY)
  413. flexSizer.Add(colName, proportion = 0,
  414. flag = wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  415. flexSizer.Add(colType, proportion = 0,
  416. flag = wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  417. flexSizer.Add(delimiter, proportion = 0,
  418. flag = wx.FIXED_MINSIZE | wx.ALIGN_CENTER_VERTICAL)
  419. flexSizer.Add(colValue, proportion = 1,
  420. flag = wx.EXPAND | wx.ALIGN_CENTER_VERTICAL)
  421. # add widget reference to self.columns
  422. columns[name]['ids'].append(colValue.GetId()) # name, type, values, id
  423. # for each attribute (including category) END
  424. border.Add(item = flexSizer, proportion = 1, flag = wx.ALL | wx.EXPAND, border = 5)
  425. panel.SetSizer(border)
  426. # for each category END
  427. # for each layer END
  428. if self.notebook.GetPageCount() == 0:
  429. self.noFoundMsg.Show(True)
  430. else:
  431. self.noFoundMsg.Show(False)
  432. self.Layout()
  433. return True
  434. def SetColumnValue(self, layer, column, value):
  435. """!Set attrbute value
  436. @param column column name
  437. @param value value
  438. """
  439. table = self.mapDBInfo.GetTable(layer)
  440. columns = self.mapDBInfo.GetTableDesc(table)
  441. for key, col in columns.iteritems():
  442. if key == column:
  443. col['values'] = [col['ctype'](value),]
  444. break
  445. class ModifyTableRecord(wx.Dialog):
  446. def __init__(self, parent, title, data, keyEditable = (-1, True),
  447. id = wx.ID_ANY, style = wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER):
  448. """!Dialog for inserting/updating table record
  449. @param data a list: [(column, value)]
  450. @param KeyEditable (id, editable?) indicates if textarea for key column
  451. is editable(True) or not
  452. """
  453. # parent -> VDigitWindow
  454. wx.Dialog.__init__(self, parent, id, title, style = style)
  455. self.CenterOnParent()
  456. self.keyId = keyEditable[0]
  457. self.panel = wx.Panel(parent = self, id = wx.ID_ANY)
  458. box = wx.StaticBox(parent = self.panel, id = wx.ID_ANY)
  459. box.Hide()
  460. self.dataPanel = scrolled.ScrolledPanel(parent = self.panel, id = wx.ID_ANY,
  461. style = wx.TAB_TRAVERSAL)
  462. self.dataPanel.SetupScrolling(scroll_x = False)
  463. # buttons
  464. self.btnCancel = wx.Button(self.panel, wx.ID_CANCEL)
  465. self.btnSubmit = wx.Button(self.panel, wx.ID_OK, _("&Submit"))
  466. self.btnSubmit.SetDefault()
  467. # data area
  468. self.widgets = []
  469. cId = 0
  470. self.usebox = False
  471. self.cat = None
  472. winFocus = False
  473. for column, value in data:
  474. if self.keyId == cId:
  475. self.cat = int(value)
  476. if not keyEditable[1]:
  477. self.usebox = True
  478. box.SetLabel(" %s %d " % (_("Category"), self.cat))
  479. box.Show()
  480. self.boxSizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  481. cId += 1
  482. continue
  483. else:
  484. valueWin = wx.SpinCtrl(parent = self.dataPanel, id = wx.ID_ANY,
  485. value = value, min = -1e9, max = 1e9, size = (250, -1))
  486. else:
  487. valueWin = wx.TextCtrl(parent = self.dataPanel, id = wx.ID_ANY,
  488. value = value, size = (250, -1))
  489. if not winFocus:
  490. wx.CallAfter(valueWin.SetFocus)
  491. winFocus = True
  492. label = wx.StaticText(parent = self.dataPanel, id = wx.ID_ANY,
  493. label = column + ":")
  494. self.widgets.append((label.GetId(), valueWin.GetId()))
  495. cId += 1
  496. self._layout()
  497. def _layout(self):
  498. """!Do layout"""
  499. sizer = wx.BoxSizer(wx.VERTICAL)
  500. # data area
  501. dataSizer = wx.FlexGridSizer (cols = 2, hgap = 3, vgap = 3)
  502. dataSizer.AddGrowableCol(1)
  503. for labelId, valueId in self.widgets:
  504. label = self.FindWindowById(labelId)
  505. value = self.FindWindowById(valueId)
  506. dataSizer.Add(label, proportion = 0,
  507. flag = wx.ALIGN_CENTER_VERTICAL)
  508. dataSizer.Add(value, proportion = 0,
  509. flag = wx.EXPAND | wx.ALIGN_CENTER_VERTICAL)
  510. self.dataPanel.SetAutoLayout(True)
  511. self.dataPanel.SetSizer(dataSizer)
  512. dataSizer.Fit(self.dataPanel)
  513. if self.usebox:
  514. self.boxSizer.Add(item = self.dataPanel, proportion = 1,
  515. flag = wx.EXPAND | wx.ALL, border = 5)
  516. # buttons
  517. btnSizer = wx.StdDialogButtonSizer()
  518. btnSizer.AddButton(self.btnCancel)
  519. btnSizer.AddButton(self.btnSubmit)
  520. btnSizer.Realize()
  521. if not self.usebox:
  522. sizer.Add(item = self.dataPanel, proportion = 1,
  523. flag = wx.EXPAND | wx.ALL, border = 5)
  524. else:
  525. sizer.Add(item = self.boxSizer, proportion = 1,
  526. flag = wx.EXPAND | wx.ALL, border = 5)
  527. sizer.Add(item = btnSizer, proportion = 0,
  528. flag = wx.EXPAND | wx.ALL, border = 5)
  529. framewidth = self.GetSize()[0]
  530. self.SetMinSize((framewidth,150))
  531. self.SetMaxSize((framewidth,300))
  532. # sizer.SetSizeHints(self.panel)
  533. self.panel.SetAutoLayout(True)
  534. self.panel.SetSizer(sizer)
  535. sizer.Fit(self.panel)
  536. self.Layout()
  537. def GetValues(self, columns = None):
  538. """!Return list of values (casted to string).
  539. If columns is given (list), return only values of given columns.
  540. """
  541. valueList = []
  542. for labelId, valueId in self.widgets:
  543. column = self.FindWindowById(labelId).GetLabel().replace(':', '')
  544. if columns is None or column in columns:
  545. value = str(self.FindWindowById(valueId).GetValue())
  546. valueList.append(value)
  547. # add key value
  548. if self.usebox:
  549. valueList.insert(self.keyId, str(self.cat))
  550. return valueList