query.py 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  1. """
  2. @package gui_core.query
  3. @brief wxGUI query dialog
  4. Classes:
  5. - query::QueryDialog
  6. (C) 2013 by the GRASS Development Team
  7. This program is free software under the GNU General Public License
  8. (>=v2). Read the file COPYING that comes with GRASS for details.
  9. @author Anna Kratochvilova <kratochanna gmail.com>
  10. """
  11. import os
  12. import wx
  13. from core.utils import _
  14. from core.gcmd import DecodeString
  15. from gui_core.treeview import TreeListView
  16. from core.treemodel import TreeModel, DictNode
  17. from grass.pydispatch.signal import Signal
  18. class QueryDialog(wx.Dialog):
  19. def __init__(self, parent, data=None):
  20. wx.Dialog.__init__(self, parent, id=wx.ID_ANY,
  21. title=_("Query results"),
  22. size=(420, 400),
  23. style=wx.DEFAULT_DIALOG_STYLE | wx.RESIZE_BORDER)
  24. # send query output to console
  25. self.redirectOutput = Signal('QueryDialog.redirectOutput')
  26. self.data = data
  27. self.panel = wx.Panel(self, id=wx.ID_ANY)
  28. self.mainSizer = wx.BoxSizer(wx.VERTICAL)
  29. helpText = wx.StaticText(self.panel, wx.ID_ANY, label=_(
  30. "Right click to copy selected values to clipboard."))
  31. helpText.SetForegroundColour(
  32. wx.SystemSettings.GetColour(
  33. wx.SYS_COLOUR_GRAYTEXT))
  34. self.mainSizer.Add(helpText, proportion=0, flag=wx.ALL, border=5)
  35. self._colNames = [_("Feature"), _("Value")]
  36. self._model = QueryTreeBuilder(self.data, column=self._colNames[1])
  37. self.tree = TreeListView(model=self._model, parent=self.panel,
  38. columns=self._colNames,
  39. style=wx.TR_DEFAULT_STYLE | wx.TR_HIDE_ROOT |
  40. wx.TR_FULL_ROW_HIGHLIGHT | wx.TR_MULTIPLE)
  41. self.tree.SetColumnWidth(0, 220)
  42. self.tree.SetColumnWidth(1, 1000)
  43. self.tree.ExpandAll(self._model.root)
  44. self.tree.contextMenu.connect(self.ShowContextMenu)
  45. self.mainSizer.Add(
  46. self.tree,
  47. proportion=1,
  48. flag=wx.EXPAND | wx.ALL,
  49. border=5)
  50. close = wx.Button(self.panel, id=wx.ID_CLOSE)
  51. close.Bind(wx.EVT_BUTTON, lambda event: self.Close())
  52. copy = wx.Button(
  53. self.panel,
  54. id=wx.ID_ANY,
  55. label=_("Copy all to clipboard"))
  56. copy.Bind(wx.EVT_BUTTON, self.Copy)
  57. self.Bind(wx.EVT_CLOSE, self.OnClose)
  58. self.redirect = wx.CheckBox(self.panel, label=_("Redirect to console"))
  59. self.redirect.SetValue(False)
  60. self.redirect.Bind(
  61. wx.EVT_CHECKBOX,
  62. lambda evt: self._onRedirect(
  63. evt.IsChecked()))
  64. hbox = wx.BoxSizer(wx.HORIZONTAL)
  65. hbox.Add(
  66. self.redirect,
  67. proportion=0,
  68. flag=wx.EXPAND | wx.RIGHT,
  69. border=5)
  70. hbox.AddStretchSpacer(1)
  71. hbox.Add(copy, proportion=0, flag=wx.EXPAND | wx.RIGHT, border=5)
  72. hbox.Add(close, proportion=0, flag=wx.EXPAND | wx.ALL, border=0)
  73. self.mainSizer.Add(
  74. hbox,
  75. proportion=0,
  76. flag=wx.EXPAND | wx.ALL,
  77. border=5)
  78. self.panel.SetSizer(self.mainSizer)
  79. self.mainSizer.Fit(self.panel)
  80. # for Windows
  81. self.SendSizeEvent()
  82. def SetData(self, data):
  83. state = self.tree.GetExpansionState()
  84. self.data = data
  85. self._model = QueryTreeBuilder(self.data, column=self._colNames[1])
  86. self.tree.SetModel(self._model)
  87. self.tree.SetExpansionState(state)
  88. if self.redirect.IsChecked():
  89. self.redirectOutput.emit(output=self._textToRedirect())
  90. def Copy(self, event):
  91. text = printResults(self._model, self._colNames[1])
  92. self._copyText(text)
  93. def ShowContextMenu(self, node):
  94. """Show context menu.
  95. Menu for copying distinguishes single and multiple selection.
  96. """
  97. nodes = self.tree.GetSelected()
  98. if not nodes:
  99. return
  100. menu = wx.Menu()
  101. texts = []
  102. if len(nodes) > 1:
  103. values = []
  104. for node in nodes:
  105. values.append(
  106. (node.label, node.data[
  107. self._colNames[1]] if node.data else ''))
  108. col1 = '\n'.join([val[1] for val in values if val[1]])
  109. col2 = '\n'.join([val[0] for val in values if val[0]])
  110. table = '\n'.join([val[0] + ': ' + val[1] for val in values])
  111. texts.append(
  112. (_("Copy from '%s' column") %
  113. self._colNames[1], col1))
  114. texts.append(
  115. (_("Copy from '%s' column") %
  116. self._colNames[0], col2))
  117. texts.append((_("Copy selected lines"), table))
  118. else:
  119. label1 = nodes[0].label
  120. texts.append((_("Copy '%s'" % self._cutLabel(label1)), label1))
  121. if nodes[0].data and nodes[0].data[self._colNames[1]]:
  122. label2 = nodes[0].data[self._colNames[1]]
  123. texts.insert(
  124. 0, (_(
  125. "Copy '%s'" %
  126. self._cutLabel(label2)), label2))
  127. texts.append((_("Copy line"), label1 + ': ' + label2))
  128. ids = []
  129. for text in texts:
  130. id = wx.NewId()
  131. ids.append(id)
  132. self.Bind(
  133. wx.EVT_MENU,
  134. lambda evt,
  135. t=text[1],
  136. id=id: self._copyText(t),
  137. id=id)
  138. menu.Append(id, text[0])
  139. # show the popup menu
  140. self.PopupMenu(menu)
  141. menu.Destroy()
  142. for id in ids:
  143. self.Unbind(wx.EVT_MENU, id=id)
  144. def _onRedirect(self, redirect):
  145. """Emits instructions to redirect query results.
  146. :param redirect: True to start redirecting, False to stop
  147. """
  148. if redirect:
  149. self.redirectOutput.emit(output=_("Query results:"), style='cmd')
  150. self.redirectOutput.emit(output=self._textToRedirect())
  151. else:
  152. self.redirectOutput.emit(output=_(" "), style='cmd')
  153. def _textToRedirect(self):
  154. text = printResults(self._model, self._colNames[1])
  155. text += '\n' + "-" * 50 + '\n'
  156. return text
  157. def _cutLabel(self, label):
  158. limit = 15
  159. if len(label) > limit:
  160. return label[:limit] + '...'
  161. return label
  162. def _copyText(self, text):
  163. """Helper function for copying"""
  164. if wx.TheClipboard.Open():
  165. do = wx.TextDataObject()
  166. do.SetText(text)
  167. wx.TheClipboard.SetData(do)
  168. wx.TheClipboard.Close()
  169. def OnClose(self, event):
  170. if self.redirect.IsChecked():
  171. self._onRedirect(False)
  172. self.Destroy()
  173. event.Skip()
  174. def QueryTreeBuilder(data, column):
  175. """Builds tree model from query results.
  176. Convert to unicode.
  177. :param data: query results as a dictionary
  178. :param column: column name
  179. :return: tree model
  180. """
  181. def addNode(parent, data, model):
  182. for k, v in data.iteritems():
  183. if isinstance(v, str):
  184. k = DecodeString(k)
  185. if isinstance(v, dict):
  186. node = model.AppendNode(parent=parent, label=k)
  187. addNode(parent=node, data=v, model=model)
  188. else:
  189. if not isinstance(v, basestring):
  190. v = str(v)
  191. elif isinstance(v, str):
  192. v = DecodeString(v)
  193. node = model.AppendNode(parent=parent, label=k,
  194. data={column: v})
  195. model = TreeModel(DictNode)
  196. for part in data:
  197. addNode(parent=model.root, data=part, model=model)
  198. return model
  199. def printResults(model, valueCol):
  200. """Print all results to string.
  201. :param model: results tree model
  202. :param valueCol: column name with value to be printed
  203. """
  204. def printTree(node, textList, valueCol, indent=0):
  205. if node.data.get(valueCol, '') or node.children:
  206. textList.append(
  207. indent * ' ' + node.label + ': ' + node.data.get(valueCol, ''))
  208. for child in node.children:
  209. printTree(
  210. node=child,
  211. textList=textList,
  212. valueCol=valueCol,
  213. indent=indent + 2)
  214. textList = []
  215. for child in model.root.children:
  216. printTree(node=child, textList=textList, valueCol=valueCol)
  217. return '\n'.join(textList)
  218. def PrepareQueryResults(coordinates, result):
  219. """Prepare query results as a Query dialog input.
  220. Adds coordinates, improves vector results tree structure.
  221. """
  222. data = []
  223. data.append({_("east, north"): ", ".join(map(str, coordinates))})
  224. for part in result:
  225. if 'Map' in part:
  226. itemText = part['Map']
  227. if 'Mapset' in part:
  228. itemText += '@' + part['Mapset']
  229. del part['Mapset']
  230. del part['Map']
  231. if part:
  232. data.append({itemText: part})
  233. else:
  234. data.append({itemText: _("Nothing found")})
  235. else:
  236. data.append(part)
  237. return data
  238. def test():
  239. app = wx.App()
  240. from grass.script import vector as gvect
  241. from grass.script import raster as grast
  242. testdata1 = grast.raster_what(
  243. map=('elevation_shade@PERMANENT', 'landclass96'),
  244. coord=[(638509.051416, 224742.348346)],
  245. localized=True)
  246. testdata2 = gvect.vector_what(
  247. map=(
  248. 'firestations', 'bridges'), coord=(
  249. 633177.897487, 221352.921257), distance=10)
  250. testdata = testdata1 + testdata2
  251. data = PrepareQueryResults(
  252. coordinates=(
  253. 638509.051416,
  254. 224742.348346),
  255. result=testdata)
  256. frame = QueryDialog(parent=None, data=data)
  257. frame.ShowModal()
  258. frame.Destroy()
  259. app.MainLoop()
  260. if __name__ == "__main__":
  261. test()