dialogs.py 30 KB

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