makemd.py 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. #!/usr/bin/python
  2. # -*- coding: utf-8 -*-
  3. import sys
  4. import xml.etree.ElementTree as ET
  5. input_list = []
  6. for arg in sys.argv[1:]:
  7. input_list.append(arg)
  8. if len(input_list) < 1:
  9. print 'usage: makemd.py <classes.xml>'
  10. sys.exit(0)
  11. def validate_tag(elem, tag):
  12. if elem.tag != tag:
  13. print "Tag mismatch, expected '" + tag + "', got " + elem.tag
  14. sys.exit(255)
  15. class_names = []
  16. classes = {}
  17. def make_class_list(class_list, columns):
  18. f = open('class_list.md', 'wb')
  19. prev = 0
  20. col_max = len(class_list) / columns + 1
  21. print ('col max is ', col_max)
  22. col_count = 0
  23. row_count = 0
  24. last_initial = ''
  25. fit_columns = []
  26. for n in range(0, columns):
  27. fit_columns += [[]]
  28. indexers = []
  29. last_initial = ''
  30. idx = 0
  31. for n in class_list:
  32. col = idx / col_max
  33. if col >= columns:
  34. col = columns - 1
  35. fit_columns[col] += [n]
  36. idx += 1
  37. if n[:1] != last_initial:
  38. indexers += [n]
  39. last_initial = n[:1]
  40. row_max = 0
  41. f.write("\n")
  42. for n in range(0, columns):
  43. if len(fit_columns[n]) > row_max:
  44. row_max = len(fit_columns[n])
  45. f.write("| ")
  46. for n in range(0, columns):
  47. f.write(" | |")
  48. f.write("\n")
  49. f.write("| ")
  50. for n in range(0, columns):
  51. f.write(" --- | ------- |")
  52. f.write("\n")
  53. for r in range(0, row_max):
  54. s = '| '
  55. for c in range(0, columns):
  56. if r >= len(fit_columns[c]):
  57. continue
  58. classname = fit_columns[c][r]
  59. initial = classname[0]
  60. if classname in indexers:
  61. s += '**' + initial + '** | '
  62. else:
  63. s += ' | '
  64. s += '[' + classname + '](class_'+ classname.lower()+') | '
  65. s += '\n'
  66. f.write(s)
  67. def dokuize_text(txt):
  68. return txt
  69. def dokuize_text(text):
  70. pos = 0
  71. while True:
  72. pos = text.find('[', pos)
  73. if pos == -1:
  74. break
  75. endq_pos = text.find(']', pos + 1)
  76. if endq_pos == -1:
  77. break
  78. pre_text = text[:pos]
  79. post_text = text[endq_pos + 1:]
  80. tag_text = text[pos + 1:endq_pos]
  81. if tag_text in class_names:
  82. tag_text = make_type(tag_text)
  83. else:
  84. # command
  85. cmd = tag_text
  86. space_pos = tag_text.find(' ')
  87. if cmd.find('html') == 0:
  88. cmd = tag_text[:space_pos]
  89. param = tag_text[space_pos + 1:]
  90. tag_text = '<' + param + '>'
  91. elif cmd.find('method') == 0:
  92. cmd = tag_text[:space_pos]
  93. param = tag_text[space_pos + 1:]
  94. if param.find('.') != -1:
  95. (class_param, method_param) = param.split('.')
  96. tag_text = '['+class_param+'.'+method_param.replace("_","&#95;")+'](' + class_param.lower() + '#' \
  97. + method_param + ')'
  98. else:
  99. tag_text = '[' + param.replace("_","&#95;") + '](#' + param + ')'
  100. elif cmd.find('image=') == 0:
  101. tag_text = '![](' + cmd[6:] + ')'
  102. elif cmd.find('url=') == 0:
  103. tag_text = '[' + cmd[4:] + ']('+cmd[4:]
  104. elif cmd == '/url':
  105. tag_text = ')'
  106. elif cmd == 'center':
  107. tag_text = ''
  108. elif cmd == '/center':
  109. tag_text = ''
  110. elif cmd == 'br':
  111. tag_text = '\n'
  112. elif cmd == 'i' or cmd == '/i':
  113. tag_text = '_'
  114. elif cmd == 'b' or cmd == '/b':
  115. tag_text = '**'
  116. elif cmd == 'u' or cmd == '/u':
  117. tag_text = '__'
  118. else:
  119. tag_text = '[' + tag_text + ']'
  120. text = pre_text + tag_text + post_text
  121. pos = len(pre_text) + len(tag_text)
  122. # tnode = ET.SubElement(parent,"div")
  123. # tnode.text=text
  124. return text
  125. def make_type(t):
  126. global class_names
  127. if t in class_names:
  128. return '[' + t + '](class_' + t.lower() + ')'
  129. return t
  130. def make_method(
  131. f,
  132. name,
  133. m,
  134. declare,
  135. event=False,
  136. ):
  137. s = ' * '
  138. ret_type = 'void'
  139. args = list(m)
  140. mdata = {}
  141. mdata['argidx'] = []
  142. for a in args:
  143. if a.tag == 'return':
  144. idx = -1
  145. elif a.tag == 'argument':
  146. idx = int(a.attrib['index'])
  147. else:
  148. continue
  149. mdata['argidx'].append(idx)
  150. mdata[idx] = a
  151. if not event:
  152. if -1 in mdata['argidx']:
  153. s += make_type(mdata[-1].attrib['type'])
  154. else:
  155. s += 'void'
  156. s += ' '
  157. if declare:
  158. # span.attrib["class"]="funcdecl"
  159. # a=ET.SubElement(span,"a")
  160. # a.attrib["name"]=name+"_"+m.attrib["name"]
  161. # a.text=name+"::"+m.attrib["name"]
  162. s += ' **'+m.attrib['name'].replace("_","&#95;")+'** '
  163. else:
  164. s += ' **['+ m.attrib['name'].replace("_","&#95;")+'](#' + m.attrib['name'] + ')** '
  165. s += ' **(**'
  166. argfound = False
  167. for a in mdata['argidx']:
  168. arg = mdata[a]
  169. if a < 0:
  170. continue
  171. if a > 0:
  172. s += ', '
  173. else:
  174. s += ' '
  175. s += make_type(arg.attrib['type'])
  176. if 'name' in arg.attrib:
  177. s += ' ' + arg.attrib['name']
  178. else:
  179. s += ' arg' + str(a)
  180. if 'default' in arg.attrib:
  181. s += '=' + arg.attrib['default']
  182. argfound = True
  183. if argfound:
  184. s += ' '
  185. s += ' **)**'
  186. if 'qualifiers' in m.attrib:
  187. s += ' ' + m.attrib['qualifiers']
  188. f.write(s + '\n')
  189. def make_doku_class(node):
  190. name = node.attrib['name']
  191. f = open("class_"+name.lower() + '.md', 'wb')
  192. f.write('# ' + name + ' \n')
  193. if 'inherits' in node.attrib:
  194. inh = node.attrib['inherits'].strip()
  195. f.write('####**Inherits:** '+make_type(inh)+'\n')
  196. if 'category' in node.attrib:
  197. f.write('####**Category:** ' + node.attrib['category'].strip()
  198. + '\n')
  199. briefd = node.find('brief_description')
  200. if briefd != None:
  201. f.write('\n### Brief Description \n')
  202. f.write(dokuize_text(briefd.text.strip()) + '\n')
  203. methods = node.find('methods')
  204. if methods != None and len(list(methods)) > 0:
  205. f.write('\n### Member Functions \n')
  206. for m in list(methods):
  207. make_method(f, node.attrib['name'], m, False)
  208. events = node.find('signals')
  209. if events != None and len(list(events)) > 0:
  210. f.write('\n### Signals \n')
  211. for m in list(events):
  212. make_method(f, node.attrib['name'], m, True, True)
  213. members = node.find('members')
  214. if members != None and len(list(members)) > 0:
  215. f.write('\n### Member Variables \n')
  216. for c in list(members):
  217. s = ' * '
  218. s += make_type(c.attrib['type']) + ' '
  219. s += '**' + c.attrib['name'] + '**'
  220. if c.text.strip() != '':
  221. s += ' - ' + c.text.strip()
  222. f.write(s + '\n')
  223. constants = node.find('constants')
  224. if constants != None and len(list(constants)) > 0:
  225. f.write('\n### Numeric Constants \n')
  226. for c in list(constants):
  227. s = ' * '
  228. s += '**' + c.attrib['name'] + '**'
  229. if 'value' in c.attrib:
  230. s += ' = **' + c.attrib['value'] + '**'
  231. if c.text.strip() != '':
  232. s += ' - ' + c.text.strip()
  233. f.write(s + '\n')
  234. descr = node.find('description')
  235. if descr != None and descr.text.strip() != '':
  236. f.write('\n### Description \n')
  237. f.write(dokuize_text(descr.text.strip()) + '\n')
  238. methods = node.find('methods')
  239. if methods != None and len(list(methods)) > 0:
  240. f.write('\n### Member Function Description \n')
  241. for m in list(methods):
  242. d = m.find('description')
  243. if d == None or d.text.strip() == '':
  244. continue
  245. f.write('\n#### <a name="'+m.attrib['name']+'">' + m.attrib['name'] + '</a>\n')
  246. make_method(f, node.attrib['name'], m, True)
  247. f.write('\n')
  248. f.write(dokuize_text(d.text.strip()))
  249. f.write('\n')
  250. for file in input_list:
  251. tree = ET.parse(file)
  252. doc = tree.getroot()
  253. if 'version' not in doc.attrib:
  254. print "Version missing from 'doc'"
  255. sys.exit(255)
  256. version = doc.attrib['version']
  257. for c in list(doc):
  258. if c.attrib['name'] in class_names:
  259. continue
  260. class_names.append(c.attrib['name'])
  261. classes[c.attrib['name']] = c
  262. class_names.sort()
  263. make_class_list(class_names, 2)
  264. for cn in class_names:
  265. c = classes[cn]
  266. make_doku_class(c)