widgets.py 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691
  1. """
  2. @package vnet.widgets
  3. @brief Base class for list of points.
  4. Classes:
  5. - widgets::PointsList
  6. - widgets::EditItem
  7. (C) 2012 by the GRASS Development Team
  8. This program is free software under the GNU General Public License
  9. (>=v2). Read the file COPYING that comes with GRASS for details.
  10. @author Original author Michael Barton
  11. @author Original version improved by Martin Landa <landa.martin gmail.com>
  12. @author Rewritten by Markus Metz redesign georectfier -> GCP Manage
  13. @author Stepan Turek <stepan.turek seznam.cz> (Created PointsList from GCPList) (GSoC 2012, mentor: Martin Landa)
  14. """
  15. import os
  16. import sys
  17. import six
  18. from copy import copy, deepcopy
  19. import wx
  20. from wx.lib.mixins.listctrl import ColumnSorterMixin, \
  21. ListCtrlAutoWidthMixin
  22. from core import globalvar
  23. from core.gcmd import GError
  24. from gui_core.widgets import FloatValidator, IntegerValidator
  25. from gui_core.wrap import (
  26. BitmapFromImage, Button, ComboBox, ListCtrl, Panel, StaticBox,
  27. StaticText, TextCtrl, CheckListCtrlMixin
  28. )
  29. if sys.version_info.major >= 3:
  30. basestring = str
  31. class PointsList(ListCtrl,
  32. CheckListCtrlMixin,
  33. ListCtrlAutoWidthMixin,
  34. ColumnSorterMixin):
  35. def __init__(self, parent, cols, id=wx.ID_ANY,
  36. pos=wx.DefaultPosition, size=wx.DefaultSize,
  37. style=wx.LC_REPORT | wx.SUNKEN_BORDER | wx.LC_HRULES |
  38. wx.LC_SINGLE_SEL):
  39. """Creates list for points.
  40. PointsList class was created from GCPList class in GCP manager. It is possible
  41. to be shared by GCP and VNET front end.
  42. Important parameters:
  43. :param cols: is list containing list items. which represents columns.
  44. This columns will be added in order as they are in list.
  45. Class will add as first column "use" with number of point and checkbox.
  46. Structure of list item must be this:
  47. -1. item: column name
  48. -2. item: column label
  49. -3. item: If column is editable by user, it must contain convert function to convert
  50. inserted string to it's type for sorting. Use None for not editable
  51. columns. Values for insertion can be in list. This allows insert
  52. just values in the list.
  53. -4. item: Default value for column cell. Value should be given in it's type
  54. in order to sorting would work properly. If 3. item is list, it must be index
  55. of some item in the list.
  56. Example of cols parameter:
  57. column name, column label, convert function, default val
  58. @code
  59. cols = [
  60. ['E', _('source E'), float, 0.0],
  61. ['N', _('source N'), float, 0.0],
  62. ['E', _('target E'), float, 0.0],
  63. ['N', _('target N'), float, 0.0],
  64. ['F_Err', _('Forward error'), None, 0],
  65. ['B_Err', _(Backward error'), None, 0]
  66. ['type', _('type'), [_(""), _("Start point"), _("End point")], 0] # Select from 3 choices ("Start point", "End point"),
  67. # Choice with index 0 ("") is default.
  68. ]
  69. @endcode
  70. """
  71. ListCtrl.__init__(self, parent, id, pos, size, style)
  72. # Mixin settings
  73. CheckListCtrlMixin.__init__(self)
  74. ListCtrlAutoWidthMixin.__init__(self)
  75. # TextEditMixin.__init__(self)
  76. # inserts first column with points numbers and checkboxes
  77. cols.insert(0, ['use', _('use'), False, 0])
  78. self.colsData = cols
  79. self.dataTypes = {
  80. "colName": 0,
  81. "colLabel": 1,
  82. "colEditable": 2,
  83. "itemDefaultValue": 3} # just for better understanding
  84. # information whether list items are checked or not
  85. self.CheckList = []
  86. self._createCols()
  87. self.hiddenCols = {}
  88. self.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnItemSelected)
  89. self.Bind(wx.EVT_LIST_ITEM_ACTIVATED, self.OnItemActivated)
  90. self.Bind(wx.EVT_LIST_COL_CLICK, self.OnColClick)
  91. self.selected = wx.NOT_FOUND
  92. self.selectedkey = -1
  93. # CheckListCtrlMixin must set an ImageList first
  94. self.il = self.GetImageList(wx.IMAGE_LIST_SMALL)
  95. # images for column sorting
  96. SmallUpArrow = BitmapFromImage(self.getSmallUpArrowImage())
  97. SmallDnArrow = BitmapFromImage(self.getSmallDnArrowImage())
  98. self.sm_dn = self.il.Add(SmallDnArrow)
  99. self.sm_up = self.il.Add(SmallUpArrow)
  100. # initialize column sorter
  101. self.itemDataMap = []
  102. ncols = self.GetColumnCount()
  103. ColumnSorterMixin.__init__(self, ncols)
  104. # init to ascending sort on first click
  105. self._colSortFlag = [1] * ncols
  106. # same structure as itemDataMap, information about choice index selected
  107. # if cell is in column without values to choose then is -1
  108. self.selIdxs = []
  109. self.ResizeColumns()
  110. self.SetColumnWidth(0, 50)
  111. def _createCols(self):
  112. """Creates columns in list"""
  113. if 0:
  114. # normal, simple columns
  115. for col in enumerate(self.colsData):
  116. iLabel = self.dataTypes["colLabel"]
  117. self.InsertColumn(col[0], col[1][iLabel])
  118. else:
  119. # the hard way: we want images on the column header
  120. info = wx.ListItem()
  121. info.SetMask(
  122. wx.LIST_MASK_TEXT | wx.LIST_MASK_IMAGE | wx.LIST_MASK_FORMAT)
  123. info.SetImage(-1)
  124. if globalvar.wxPythonPhoenix:
  125. info.Format = wx.LIST_FORMAT_LEFT
  126. else:
  127. info.m_format = wx.LIST_FORMAT_LEFT
  128. for col in enumerate(self.colsData):
  129. iLabel = self.dataTypes["colLabel"]
  130. info.SetText(col[1][iLabel])
  131. if globalvar.wxPythonPhoenix:
  132. self.InsertColumn(col[0], info)
  133. else:
  134. self.InsertColumnInfo(col[0], info)
  135. def AddItem(self):
  136. """Appends an item to list with default values"""
  137. iDefVal = self.dataTypes["itemDefaultValue"]
  138. iColEd = self.dataTypes["colEditable"]
  139. itemData = []
  140. itemIndexes = []
  141. for col in self.colsData:
  142. if type(col[iColEd]).__name__ == "list":
  143. itemData.append(col[iColEd][col[iDefVal]])
  144. itemIndexes.append(col[iDefVal])
  145. else:
  146. itemData.append(col[iDefVal])
  147. itemIndexes.append(-1) # not a choise column
  148. self.selIdxs.append(itemIndexes)
  149. for hCol in six.itervalues(self.hiddenCols):
  150. defVal = hCol['colsData'][iDefVal]
  151. if type(hCol['colsData'][iColEd]).__name__ == "list":
  152. hCol['itemDataMap'].append(hCol['colsData'][iColEd][defVal])
  153. hCol['selIdxs'].append(defVal)
  154. else:
  155. hCol['itemDataMap'].append(defVal)
  156. hCol['selIdxs'].append(-1)
  157. self.selectedkey = self.GetItemCount()
  158. itemData[0] = self.selectedkey + 1
  159. self.itemDataMap.append(copy(itemData))
  160. self.Append(list(map(str, itemData)))
  161. self.selected = self.GetItemCount() - 1
  162. self.SetItemData(self.selected, self.selectedkey)
  163. self.SetItemState(self.selected,
  164. wx.LIST_STATE_SELECTED,
  165. wx.LIST_STATE_SELECTED)
  166. self.ResizeColumns()
  167. return self.selected
  168. def GetCellValue(self, key, colName):
  169. """Get value in cell of list using key (same regardless of sorting)"""
  170. colNum = self._getColumnNum(colName)
  171. if colNum < 0:
  172. return None
  173. iColEd = self.dataTypes["colEditable"]
  174. if self.selIdxs[key][colNum] != -1:
  175. return self.selIdxs[key][colNum]
  176. return self.itemDataMap[key][colNum]
  177. def GetCellSelIdx(self, key, colName):
  178. """Get selected index in cell of list using key (same regardless of sorting)
  179. :return: number of chosen value, if column has values to choose
  180. :return: -1 if column does not has values to choose
  181. """
  182. colNum = self._getColumnNum(colName)
  183. iColEd = self.dataTypes["colEditable"]
  184. return self.selIdxs[key][colNum]
  185. def EditCellIndex(self, index, colName, cellData):
  186. """Changes value in list using key (same regardless of sorting)"""
  187. colNum = self._getColumnNum(colName)
  188. key = self.GetItemData(index)
  189. iColEd = self.dataTypes["colEditable"]
  190. if type(self.colsData[colNum][iColEd]).__name__ == "list":
  191. cellVal = self.colsData[colNum][iColEd][cellData]
  192. self.selIdxs[key][colNum] = cellData
  193. else:
  194. cellVal = cellData
  195. self.selIdxs[key][colNum] = -1
  196. self.itemDataMap[key][colNum] = cellVal
  197. if not isinstance(cellVal, basestring):
  198. cellVal = str(cellVal)
  199. self.SetItem(index, colNum, cellVal)
  200. def EditCellKey(self, key, colName, cellData):
  201. """Changes value in list using index (changes during sorting)"""
  202. colNum = self._getColumnNum(colName)
  203. iColEd = self.dataTypes["colEditable"]
  204. if type(self.colsData[colNum][iColEd]).__name__ == "list":
  205. cellVal = self.colsData[colNum][iColEd][cellData]
  206. self.selIdxs[key][colNum] = cellData
  207. else:
  208. cellVal = cellData
  209. self.selIdxs[key][colNum] = -1
  210. self.itemDataMap[key][colNum] = cellVal
  211. index = self._findIndex(key)
  212. if index != -1:
  213. if not isinstance(cellVal, basestring):
  214. cellVal = str(cellVal)
  215. self.SetItem(index, colNum, cellVal)
  216. def _findIndex(self, key):
  217. """Find index for key"""
  218. index = -1
  219. while True:
  220. index = self.GetNextItem(index,
  221. wx.LIST_NEXT_BELOW)
  222. if key == self.GetItemData(index):
  223. return index
  224. if index == -1:
  225. break
  226. return -1
  227. def ChangeColEditable(self, colName, colType):
  228. """Change 3. item in constructor parameter cols (see the class constructor hint)"""
  229. colNum = self._getColumnNum(colName)
  230. iColEd = self.dataTypes["colEditable"]
  231. self.colsData[colNum][iColEd] = colType
  232. def DeleteItem(self):
  233. """Delete selected item in list"""
  234. if self.selected == wx.NOT_FOUND:
  235. return
  236. key = self.GetItemData(self.selected)
  237. ListCtrl.DeleteItem(self, self.selected)
  238. del self.itemDataMap[key]
  239. self.selIdxs.pop(key)
  240. # update hidden columns
  241. for hCol in six.itervalues(self.hiddenCols):
  242. hCol['itemDataMap'].pop(key)
  243. hCol['selIdxs'].pop(key)
  244. # update key and point number
  245. for newkey in range(key, len(self.itemDataMap)):
  246. index = self.FindItemData(-1, newkey + 1)
  247. self.itemDataMap[newkey][0] = newkey
  248. self.SetItem(index, 0, str(newkey + 1))
  249. self.SetItemData(index, newkey)
  250. # update selected
  251. if self.GetItemCount() > 0:
  252. if self.selected < self.GetItemCount():
  253. self.selectedkey = self.GetItemData(self.selected)
  254. else:
  255. self.selected = self.GetItemCount() - 1
  256. self.selectedkey = self.GetItemData(self.selected)
  257. self.SetItemState(self.selected,
  258. wx.LIST_STATE_SELECTED,
  259. wx.LIST_STATE_SELECTED)
  260. else:
  261. self.selected = wx.NOT_FOUND
  262. self.selectedkey = -1
  263. def ClearItem(self, event):
  264. """"Set all values to default in selected item of points list and uncheck it."""
  265. if self.selected == wx.NOT_FOUND:
  266. return
  267. index = self.selected
  268. iDefVal = self.dataTypes["itemDefaultValue"]
  269. iColEd = self.dataTypes["colEditable"]
  270. i = 0
  271. for col in self.colsData:
  272. if i == 0:
  273. i += 1
  274. continue
  275. if type(col[iColEd]).__name__ == "list":
  276. self.EditCell(index, i, col[iColEd][col[iDefVal]])
  277. else:
  278. self.EditCell(index, i, col[iDefVal])
  279. i += 1
  280. self.CheckItem(index, False)
  281. def ResizeColumns(self, minWidth=[90, 120]):
  282. """Resize columns"""
  283. for i in range(self.GetColumnCount()):
  284. self.SetColumnWidth(i, wx.LIST_AUTOSIZE)
  285. # first column is checkbox, don't set to minWidth
  286. if i > 0 and self.GetColumnWidth(i) < minWidth[i > 4]:
  287. self.SetColumnWidth(i, minWidth[i > 4])
  288. self.SendSizeEvent()
  289. def GetSelected(self):
  290. """Get index of selected item."""
  291. return self.selected
  292. # Used by the ColumnSorterMixin, see wx/lib/mixins/listctrl.py
  293. def GetSortImages(self):
  294. return (self.sm_dn, self.sm_up)
  295. # Used by the ColumnSorterMixin, see wx/lib/mixins/listctrl.py
  296. def GetListCtrl(self):
  297. return self
  298. def OnItemActivated(self, event):
  299. """When item is double clicked, open editor to edit editable columns."""
  300. data = []
  301. index = event.GetIndex()
  302. key = self.GetItemData(index)
  303. changed = False
  304. iColEd = self.dataTypes["colEditable"]
  305. for col in enumerate(self.colsData):
  306. if col[1][iColEd]:
  307. data.append([col[0], # culumn number
  308. self.GetItem(
  309. index, col[0]).GetText(), # cell value
  310. col[1][iColEd]]) # convert function for type check
  311. if not data:
  312. return
  313. dlg = self.CreateEditDialog(data=data, pointNo=key)
  314. if dlg.ShowModal() == wx.ID_OK:
  315. editedData = dlg.GetValues() # string
  316. if len(editedData) == 0:
  317. GError(parent=self, message=_(
  318. "Invalid value inserted. Operation canceled."))
  319. else:
  320. i = 0
  321. for editedCell in editedData:
  322. if editedCell[1] != data[i][1]:
  323. value = editedCell[1]
  324. if not isinstance(editedCell[1], basestring):
  325. value = str(editedCell[1])
  326. self.SetItem(index, editedCell[0], value)
  327. self.itemDataMap[key][editedCell[0]] = editedCell[1]
  328. changed = True
  329. i += 1
  330. self.selIdxs[key] = dlg.GetSelectionIndexes()
  331. dlg.Destroy()
  332. return changed, key
  333. def CreateEditDialog(self, data, pointNo):
  334. """Helper function
  335. It is possible to define in child derived class
  336. and adapt created dialog (e. g. it's title...)
  337. """
  338. return EditItem(parent=self, id=wx.ID_ANY, data=data, pointNo=pointNo)
  339. def OnColClick(self, event):
  340. """ListCtrl forgets selected item..."""
  341. self.selected = self.FindItemData(-1, self.selectedkey)
  342. self.SetItemState(self.selected,
  343. wx.LIST_STATE_SELECTED,
  344. wx.LIST_STATE_SELECTED)
  345. event.Skip()
  346. def OnItemSelected(self, event):
  347. """Updates class attributes holding information about selected item"""
  348. if self.selected != event.GetIndex():
  349. self.selected = event.GetIndex()
  350. self.selectedkey = self.GetItemData(self.selected)
  351. event.Skip()
  352. def getSmallUpArrowImage(self):
  353. """Get arrow up symbol for indication of sorting"""
  354. stream = open(
  355. os.path.join(
  356. globalvar.IMGDIR,
  357. 'small_up_arrow.png'),
  358. 'rb')
  359. try:
  360. img = wx.Image(stream)
  361. finally:
  362. stream.close()
  363. return img
  364. def getSmallDnArrowImage(self):
  365. """Get arrow down symbol for indication of sorting"""
  366. stream = open(
  367. os.path.join(
  368. globalvar.IMGDIR,
  369. 'small_down_arrow.png'),
  370. 'rb')
  371. try:
  372. img = wx.Image(stream)
  373. finally:
  374. stream.close()
  375. return img
  376. def _getColumnNum(self, colName):
  377. """Get position of column among showed columns
  378. :param colName: name of column
  379. :type colName: str
  380. :return: index of columns or -1 if col was not found
  381. """
  382. for iCol, col in enumerate(self.colsData):
  383. if colName == col[0]:
  384. return iCol
  385. return -1
  386. def HideColumn(self, colName):
  387. """Hide column (hidden columns are not editable)
  388. :param colName: name of column
  389. :type colName: str
  390. :return: True - if column was hidden
  391. :return: False - if position is not valid or column is not showed
  392. """
  393. colNum = self._getColumnNum(colName)
  394. if colNum == -1:
  395. return False
  396. hiddenCol = self.GetColumn(colNum)
  397. self.DeleteColumn(colNum)
  398. self.hiddenCols[colName] = {}
  399. self.hiddenCols[colName]['wxCol'] = hiddenCol
  400. hiddenMaps = []
  401. hiddenSelIdxs = []
  402. for item in self.itemDataMap:
  403. hiddenMaps.append(item.pop(colNum))
  404. for item in self.selIdxs:
  405. hiddenSelIdxs.append(item.pop(colNum))
  406. self.hiddenCols[colName]['itemDataMap'] = hiddenMaps
  407. self.hiddenCols[colName]['selIdxs'] = hiddenSelIdxs
  408. self.hiddenCols[colName]['colsData'] = self.colsData.pop(colNum)
  409. self.ResizeColumns()
  410. return True
  411. def ShowColumn(self, colName, pos):
  412. """Show column
  413. :param colName: name of column
  414. :type colName: str
  415. :param pos: zero based index of position among showed columns (including added 'use' column)
  416. :return: True if column was shown
  417. :return: False if position is not valid or column is not hidden
  418. """
  419. if pos < 0 and pos >= self.self.GetColumnCount():
  420. return False
  421. if colName in self.hiddenCols:
  422. col = self.hiddenCols[colName]
  423. for item in enumerate(self.itemDataMap):
  424. item[1].insert(pos, col['itemDataMap'][item[0]])
  425. for item in enumerate(self.selIdxs):
  426. item[1].insert(pos, col['selIdxs'][item[0]])
  427. self.colsData.insert(pos, col['colsData'])
  428. self.InsertColumnItem(pos, col['wxCol'])
  429. self.ResizeColumns()
  430. del self.hiddenCols[colName]
  431. return True
  432. return False
  433. def IsShown(self, colName):
  434. """Is column shown
  435. :param colName: name of column
  436. :type colName: str
  437. :return: True - if is shown
  438. :return: False - if is not shown
  439. """
  440. if self._getColumnNum(colName) == -1:
  441. return False
  442. else:
  443. return True
  444. class EditItem(wx.Dialog):
  445. def __init__(
  446. self, parent, data, pointNo, itemCap="Point No.", id=wx.ID_ANY,
  447. title=_("Edit point"),
  448. style=wx.DEFAULT_DIALOG_STYLE):
  449. """Dialog for editing item cells in list"""
  450. wx.Dialog.__init__(self, parent, id, title=_(title), style=style)
  451. self.parent = parent
  452. panel = Panel(parent=self)
  453. sizer = wx.BoxSizer(wx.VERTICAL)
  454. box = StaticBox(parent=panel, id=wx.ID_ANY,
  455. label=" %s %s " % (_(itemCap), str(pointNo + 1)))
  456. boxSizer = wx.StaticBoxSizer(box, wx.VERTICAL)
  457. # source coordinates
  458. gridSizer = wx.GridBagSizer(vgap=5, hgap=5)
  459. self.fields = []
  460. self.data = deepcopy(data)
  461. col = 0
  462. row = 0
  463. iField = 0
  464. for cell in self.data:
  465. # Select
  466. if type(cell[2]).__name__ == "list":
  467. self.fields.append(ComboBox(parent=panel, id=wx.ID_ANY,
  468. choices=cell[2],
  469. style=wx.CB_READONLY,
  470. size=(110, -1)))
  471. # Text field
  472. else:
  473. if cell[2] == float:
  474. validator = FloatValidator()
  475. elif cell[2] == int:
  476. validator = IntegerValidator()
  477. else:
  478. validator = None
  479. if validator:
  480. self.fields.append(
  481. TextCtrl(
  482. parent=panel, id=wx.ID_ANY, validator=validator,
  483. size=(150, -1)))
  484. else:
  485. self.fields.append(TextCtrl(parent=panel, id=wx.ID_ANY,
  486. size=(150, -1)))
  487. value = cell[1]
  488. if not isinstance(cell[1], basestring):
  489. value = str(cell[1])
  490. self.fields[iField].SetValue(value)
  491. label = StaticText(
  492. parent=panel,
  493. id=wx.ID_ANY,
  494. label=_(
  495. parent.GetColumn(
  496. cell[0]).GetText()) +
  497. ":") # name of column)
  498. gridSizer.Add(label,
  499. flag=wx.ALIGN_CENTER_VERTICAL,
  500. pos=(row, col))
  501. col += 1
  502. gridSizer.Add(self.fields[iField],
  503. pos=(row, col))
  504. if col % 3 == 0:
  505. col = 0
  506. row += 1
  507. else:
  508. col += 1
  509. iField += 1
  510. boxSizer.Add(gridSizer, proportion=1,
  511. flag=wx.EXPAND | wx.ALL, border=5)
  512. sizer.Add(boxSizer, proportion=1,
  513. flag=wx.EXPAND | wx.ALL, border=5)
  514. #
  515. # buttons
  516. #
  517. self.btnCancel = Button(panel, wx.ID_CANCEL)
  518. self.btnOk = Button(panel, wx.ID_OK)
  519. self.btnOk.SetDefault()
  520. btnSizer = wx.StdDialogButtonSizer()
  521. btnSizer.AddButton(self.btnCancel)
  522. btnSizer.AddButton(self.btnOk)
  523. btnSizer.Realize()
  524. sizer.Add(btnSizer, proportion=0,
  525. flag=wx.ALIGN_RIGHT | wx.ALL, border=5)
  526. panel.SetSizer(sizer)
  527. sizer.Fit(self)
  528. def GetValues(self):
  529. """Return list of values (as strings).
  530. """
  531. iField = 0
  532. for cell in self.data:
  533. value = self.fields[iField].GetValue()
  534. if type(cell[2]).__name__ == "list":
  535. cell[1] = value
  536. else:
  537. try:
  538. cell[1] = cell[2](value)
  539. except ValueError:
  540. return []
  541. iField += 1
  542. return self.data
  543. def GetSelectionIndexes(self):
  544. """Return indexes of selected values (works just for choice
  545. columns).
  546. """
  547. iField = 0
  548. itemIndexes = []
  549. for cell in self.parent.colsData:
  550. if type(cell[2]).__name__ == "list":
  551. itemIndexes.append(self.fields[iField].GetSelection())
  552. else:
  553. itemIndexes.append(-1) # not a choise column
  554. if cell[2]:
  555. iField += 1
  556. return itemIndexes