dialogs.py 30 KB

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