dialogs.py 30 KB

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