dialogs.py 30 KB

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