dialogs.py 30 KB

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