menudata.py 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  1. """!
  2. @package menudata.py
  3. @brief Complex list for main menu entries for GRASS wxPython GUI.
  4. Classes:
  5. - Data
  6. - MenuTree
  7. Usage:
  8. @code
  9. python menudata.py action [file]
  10. @endcode
  11. where <i>action</i>:
  12. - strings (default)
  13. - tree
  14. - commands
  15. (C) 2007-2009 by the GRASS Development Team
  16. This program is free software under the GNU General Public
  17. License (>=v2). Read the file COPYING that comes with GRASS
  18. for details.
  19. @author Michael Barton (Arizona State University)
  20. @author Yann Chemin <yann.chemin gmail.com>
  21. @author Martin Landa <landa.martin gmail.com>
  22. @author Glynn Clements
  23. """
  24. import os
  25. import sys
  26. import pprint
  27. try:
  28. import xml.etree.ElementTree as etree
  29. except ImportError:
  30. import elementtree.ElementTree as etree # Python <= 2.4
  31. class Data:
  32. '''!Data object that returns menu descriptions to be used in wxgui.py.'''
  33. def __init__(self, filename=None):
  34. if not filename:
  35. gisbase = os.getenv('GISBASE')
  36. filename = os.path.join(gisbase, 'etc', 'wxpython', 'xml', 'menudata.xml')
  37. self.tree = etree.parse(filename)
  38. def getMenuItem(self, mi):
  39. if mi.tag == 'separator':
  40. return ('', '', '', '', '')
  41. elif mi.tag == 'menuitem':
  42. label = _(mi.find('label').text)
  43. help = _(mi.find('help').text)
  44. handler = mi.find('handler').text
  45. gcmd = mi.find('command') # optional
  46. keywords = mi.find('keywords') # optional
  47. shortcut = mi.find('shortcut') # optional
  48. if gcmd != None:
  49. gcmd = gcmd.text
  50. else:
  51. gcmd = ""
  52. if keywords != None:
  53. keywords = keywords.text
  54. else:
  55. keywords = ""
  56. if shortcut != None:
  57. shortcut = shortcut.text
  58. else:
  59. shortcut = ""
  60. return (label, help, handler, gcmd, keywords, shortcut)
  61. elif mi.tag == 'menu':
  62. return self.getMenu(mi)
  63. else:
  64. raise Exception()
  65. def getMenu(self, m):
  66. label = _(m.find('label').text)
  67. items = m.find('items')
  68. return (label, tuple(map(self.getMenuItem, items)))
  69. def getMenuBar(self, mb):
  70. return tuple(map(self.getMenu, mb.findall('menu')))
  71. def getMenuData(self, md):
  72. return list(map(self.getMenuBar, md.findall('menubar')))
  73. def GetMenu(self):
  74. return self.getMenuData(self.tree.getroot())
  75. def PrintStrings(self, fh):
  76. """!Print menu strings to file (used for localization)
  77. @param fh file descriptor"""
  78. fh.write('menustrings = [\n')
  79. for node in self.tree.getiterator():
  80. if node.tag in ['label', 'help']:
  81. fh.write(' _(%r),\n' % node.text)
  82. fh.write(' \'\']\n')
  83. def PrintTree(self, fh):
  84. """!Print menu tree to file
  85. @param fh file descriptor"""
  86. level = 0
  87. for eachMenuData in self.GetMenu():
  88. for label, items in eachMenuData:
  89. fh.write('- %s\n' % label)
  90. self.__PrintTreeItems(fh, level + 1, items)
  91. def __PrintTreeItems(self, fh, level, menuData):
  92. """!Print menu tree items to file (used by PrintTree)
  93. @param fh file descriptor
  94. @param level menu level
  95. @param menuData menu data to print out"""
  96. for eachItem in menuData:
  97. if len(eachItem) == 2:
  98. if eachItem[0]:
  99. fh.write('%s - %s\n' % (' ' * level, eachItem[0]))
  100. self.__PrintTreeItems(fh, level + 1, eachItem[1])
  101. else:
  102. if eachItem[0]:
  103. fh.write('%s - %s\n' % (' ' * level, eachItem[0]))
  104. def PrintCommands(self, fh, itemSep = ' | ', menuSep = ' > '):
  105. """!Print commands list (command | menu item > menu item)
  106. @param fh file descriptor
  107. """
  108. level = 0
  109. for eachMenuData in self.GetMenu():
  110. for label, items in eachMenuData:
  111. menuItems = [label, ]
  112. self.__PrintCommandsItems(fh, level + 1, items,
  113. menuItems, itemSep, menuSep)
  114. def __PrintCommandsItems(self, fh, level, menuData,
  115. menuItems, itemSep, menuSep):
  116. """!Print commands item (used by PrintCommands)
  117. @param fh file descriptor
  118. @param menuItems list of menu items
  119. """
  120. for eachItem in menuData:
  121. if len(eachItem) == 2:
  122. if eachItem[0]:
  123. try:
  124. menuItems[level] = eachItem[0]
  125. except IndexError:
  126. menuItems.append(eachItem[0])
  127. self.__PrintCommandsItems(fh, level + 1, eachItem[1],
  128. menuItems, itemSep, menuSep)
  129. else:
  130. try:
  131. del menuItems[level]
  132. except IndexError:
  133. pass
  134. if eachItem[3]:
  135. fh.write('%s%s' % (eachItem[3], itemSep))
  136. fh.write(menuSep.join(menuItems))
  137. fh.write('%s%s' % (menuSep, eachItem[0]))
  138. fh.write('\n')
  139. def GetModules(self):
  140. """!Create dictionary of modules used to search module by
  141. keywords, description, etc."""
  142. modules = dict()
  143. for node in self.tree.getiterator():
  144. if node.tag == 'menuitem':
  145. module = description = ''
  146. keywords = []
  147. for child in node.getchildren():
  148. if child.tag == 'help':
  149. description = child.text
  150. if child.tag == 'command':
  151. module = child.text
  152. if child.tag == 'keywords':
  153. if child.text:
  154. keywords = child.text.split(',')
  155. if module:
  156. modules[module] = { 'desc': description,
  157. 'keywords' : keywords }
  158. if len(keywords) < 1:
  159. print >> sys.stderr, "WARNING: Module <%s> has no keywords" % module
  160. return modules
  161. if __name__ == "__main__":
  162. import sys
  163. # i18N
  164. import gettext
  165. gettext.install('grasswxpy', os.path.join(os.getenv("GISBASE"), 'locale'), unicode=True)
  166. file = None
  167. if len(sys.argv) == 1:
  168. action = 'strings'
  169. elif len(sys.argv) > 1:
  170. action = sys.argv[1]
  171. if len(sys.argv) > 2:
  172. file = sys.argv[2]
  173. if file:
  174. data = Data(file)
  175. else:
  176. data = Data()
  177. if action == 'strings':
  178. data.PrintStrings(sys.stdout)
  179. elif action == 'tree':
  180. data.PrintTree(sys.stdout)
  181. elif action == 'commands':
  182. data.PrintCommands(sys.stdout)
  183. elif action == 'dump':
  184. pprint.pprint(data.GetMenu(), stream = sys.stdout, indent = 2)
  185. sys.exit(0)