419 lines
13 KiB
Python
419 lines
13 KiB
Python
|
#!/usr/bin/python
|
||
|
import monkDebug as debug
|
||
|
import sys
|
||
|
import monkTools
|
||
|
#import CppHeaderParser
|
||
|
import re
|
||
|
import codeBB
|
||
|
import collections
|
||
|
|
||
|
global_class_link = {
|
||
|
"std::string" : "http://www.cplusplus.com/reference/string/string/",
|
||
|
"std::u16string" : "http://www.cplusplus.com/reference/string/u16string/",
|
||
|
"std::u32string" : "http://www.cplusplus.com/reference/string/u32string/",
|
||
|
"std::wstring" : "http://www.cplusplus.com/reference/string/wstring/",
|
||
|
"std::vector" : "http://www.cplusplus.com/reference/vector/vector/"
|
||
|
}
|
||
|
|
||
|
|
||
|
def replace_type(match):
|
||
|
value = "<span class=\"code-type\">" + match.group() + "</span>"
|
||
|
return value
|
||
|
|
||
|
def replace_storage_keyword(match):
|
||
|
value = "<span class=\"code-storage-keyword\">" + match.group() + "</span>"
|
||
|
return value
|
||
|
|
||
|
def display_color(valBase):
|
||
|
# storage keyword :
|
||
|
p = re.compile("(inline|const|class|virtual|private|public|protected|friend|const|extern|auto|register|static|volatile|typedef|struct|union|enum)")
|
||
|
val = p.sub(replace_storage_keyword, valBase)
|
||
|
# type :
|
||
|
p = re.compile("(bool|BOOL|char(16_t|32_t)?|double|float|u?int(8|16|32|64|128)?(_t)?|long|short|signed|size_t|unsigned|void|(I|U)(8|16|32|64|128))")
|
||
|
val = p.sub(replace_type, val)
|
||
|
return val, len(valBase)
|
||
|
|
||
|
def display_type(type, myDoc):
|
||
|
type = type.replace("inline ", "")
|
||
|
lenght = 0;
|
||
|
isFirst = True
|
||
|
out = ''
|
||
|
# we split all the element in list sepa=rated with space to keep class... and standard c+ class
|
||
|
for element in type.split(' '):
|
||
|
if isFirst == False:
|
||
|
out += " "
|
||
|
lenght += 1
|
||
|
isFirst = False
|
||
|
# check if the element in internal at the current lib
|
||
|
name, link = myDoc.get_class_link(element)
|
||
|
if len(link) != 0:
|
||
|
out += "<a href=\"" + link + "\" class=\"code-type\">" + name + "</a>"
|
||
|
lenght += len(element)
|
||
|
# Ckeck if the variable in a standard class:
|
||
|
elif element in global_class_link.keys():
|
||
|
out += "<a href=\"" + global_class_link[element] + "\" class=\"code-type\">" + element + "</a>"
|
||
|
lenght += len(element)
|
||
|
else:
|
||
|
data, lenghtTmp = display_color(element)
|
||
|
out += data
|
||
|
lenght += lenghtTmp
|
||
|
# get every subelement class :
|
||
|
return [out,lenght]
|
||
|
|
||
|
def display_doxygen_param(comment, input, output):
|
||
|
data = "<b>Parameter"
|
||
|
if input == True:
|
||
|
data += " [input]"
|
||
|
if output == True:
|
||
|
data += " [output]"
|
||
|
data += ":</b> "
|
||
|
#extract first element:
|
||
|
val = comment.find(" ")
|
||
|
var = comment[:val]
|
||
|
endComment = comment[val:]
|
||
|
# TODO : Check if it exist in the parameter list ...
|
||
|
data += "<span class=\"code-argument\">" + var + "</span> " + endComment
|
||
|
|
||
|
data += "<br/>"
|
||
|
return data
|
||
|
|
||
|
|
||
|
def parse_doxygen(data) :
|
||
|
streams = data.split("@")
|
||
|
data2 = ''
|
||
|
for element in streams:
|
||
|
if element[:1] == "\n" \
|
||
|
or element[:2] == "\n\n":
|
||
|
# nothing to do : Nomale case of the first \n
|
||
|
None
|
||
|
elif element[:6] == "brief ":
|
||
|
data2 += element[6:]
|
||
|
data2 += "<br/>"
|
||
|
|
||
|
for element in streams:
|
||
|
if element[:1] == "\n" \
|
||
|
or element[:2] == "\n\n":
|
||
|
# nothing to do : Nomale case of the first \n
|
||
|
None
|
||
|
elif element[:5] == "note ":
|
||
|
data2 += "<b>Notes:</b> "
|
||
|
data2 += element[5:]
|
||
|
data2 += "<br/> "
|
||
|
|
||
|
data3 = ''
|
||
|
for element in streams:
|
||
|
if element[:1] == "\n" \
|
||
|
or element[:2] == "\n\n":
|
||
|
# nothing to do : Nomale case of the first \n
|
||
|
None
|
||
|
elif element[:14] == "param[in,out] " \
|
||
|
or element[:14] == "param[out,in] ":
|
||
|
data3 += display_doxygen_param(element[14:], True, True)
|
||
|
elif element[:10] == "param[in] ":
|
||
|
data3 += display_doxygen_param(element[10:], True, False)
|
||
|
elif element[:11] == "param[out] ":
|
||
|
data3 += display_doxygen_param(element[11:], False, True)
|
||
|
elif element[:6] == "param ":
|
||
|
data3 += display_doxygen_param(element[6:], False, False)
|
||
|
elif element[:7] == "return ":
|
||
|
data3 += "<b>Return:</b> "
|
||
|
data3 += element[7:]
|
||
|
data3 += "<br/>"
|
||
|
if data3 != '':
|
||
|
data2 += "<ul>\n"
|
||
|
data2 += data3
|
||
|
data2 += "</ul>\n"
|
||
|
return data2
|
||
|
|
||
|
def white_space(size) :
|
||
|
ret = ''
|
||
|
for iii in range(len(ret), size):
|
||
|
ret += " "
|
||
|
return ret
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
def calsulateSizeFunction(function, size) :
|
||
|
if len(function["name"]) > size:
|
||
|
return len(function["name"])+1
|
||
|
return size
|
||
|
|
||
|
def calsulateSizeReturn(function, size) :
|
||
|
if len(function["rtnType"]) > size:
|
||
|
return len(function["rtnType"])+1
|
||
|
return size
|
||
|
|
||
|
|
||
|
def addSub(tree, filterSubNamespace=False):
|
||
|
return ""
|
||
|
|
||
|
|
||
|
# ##############################################################
|
||
|
# NEW function ...
|
||
|
# ##############################################################
|
||
|
|
||
|
|
||
|
|
||
|
def generate_menu(element, namespaceStack=[], level=1):
|
||
|
listBase = element.get_all_sub_type(['namespace'])
|
||
|
if len(listBase) == 0:
|
||
|
return ""
|
||
|
ret = ""
|
||
|
ret += '<ul class="niveau' + str(level) + '">\n'
|
||
|
for element in listBase:
|
||
|
namespaceStack.append(element['node'].get_name())
|
||
|
retTmp = generate_menu(element['node'], namespaceStack, level+1)
|
||
|
namespaceStack.pop()
|
||
|
if retTmp != "":
|
||
|
subMenu = ' class="sousmenu"'
|
||
|
else:
|
||
|
subMenu = ''
|
||
|
ret += ' <li' + subMenu + '>' + generate_link(element['node'], namespaceStack) + '\n'
|
||
|
ret += retTmp
|
||
|
ret += ' </li>\n'
|
||
|
ret += '</ul>\n'
|
||
|
return ret
|
||
|
|
||
|
def generate_html_page_name(element, namespaceStack):
|
||
|
link = ""
|
||
|
for name in namespaceStack:
|
||
|
link += name + "__"
|
||
|
return element.get_node_type() + "_" + link + element.get_name() + '.html'
|
||
|
|
||
|
def generate_name(element, namespaceStack):
|
||
|
link = ""
|
||
|
for name in namespaceStack:
|
||
|
link += name + "::"
|
||
|
return element.get_node_type() + ": " + link + element.get_name()
|
||
|
|
||
|
|
||
|
def generate_link(element, namespaceStack):
|
||
|
return '<a href="' + generate_html_page_name(element, namespaceStack) + '">' + element.get_name() + '</a>'
|
||
|
|
||
|
def calculate_methode_size(list):
|
||
|
returnSize = 0;
|
||
|
methodeSize = 0;
|
||
|
for element in list:
|
||
|
retType = ""
|
||
|
if element['node'].get_virtual() == True:
|
||
|
retType += 'virtual '
|
||
|
retType += element['node'].get_return_type().to_str()
|
||
|
tmpLen = len(retType)
|
||
|
if returnSize < tmpLen:
|
||
|
returnSize = tmpLen
|
||
|
tmpLen = len(element['node'].get_name())
|
||
|
if methodeSize < tmpLen:
|
||
|
methodeSize = tmpLen
|
||
|
return [returnSize, methodeSize]
|
||
|
|
||
|
|
||
|
def write_methode(element, namespaceStack, displaySize = None, link = True):
|
||
|
if displaySize == None:
|
||
|
displaySize = calculate_methode_size([element])
|
||
|
ret = ""
|
||
|
if 'access' in element.keys():
|
||
|
if element['access'] == 'private':
|
||
|
ret += '- '
|
||
|
elif element['access'] == 'protected':
|
||
|
ret += '# '
|
||
|
elif element['access'] == 'public':
|
||
|
ret += '+ '
|
||
|
else:
|
||
|
ret += ' '
|
||
|
retType = ""
|
||
|
if element['node'].get_virtual() == True:
|
||
|
retType += 'virtual '
|
||
|
retType += element['node'].get_return_type().to_str()
|
||
|
if retType != "":
|
||
|
retType2 = re.sub("<","<", retType)
|
||
|
retType2 = re.sub(">",">", retType2)
|
||
|
ret += retType2
|
||
|
ret += " "
|
||
|
ret += white_space(displaySize[0] - len(retType))
|
||
|
name = element['node'].get_name()
|
||
|
if link == True:
|
||
|
ret += '<a class="code-function" href="#' + str(element['node'].get_uid()) + '">' + name + '</a>'
|
||
|
else:
|
||
|
ret += '<span class="code-function">' + name + '</span>'
|
||
|
ret += white_space(displaySize[1] - len(name)) + ' ('
|
||
|
listParam = element['node'].get_param()
|
||
|
first = True
|
||
|
for param in listParam:
|
||
|
if first == False:
|
||
|
ret += ',<br/>'
|
||
|
ret += white_space(displaySize[0] + displaySize[1] +5)
|
||
|
first = False
|
||
|
retParam = param.get_type().to_str()
|
||
|
if retParam != "":
|
||
|
ret += retParam
|
||
|
ret += " "
|
||
|
ret += "<span class=\"code-argument\">" + param.get_name() + "</span>"
|
||
|
ret += ')'
|
||
|
if element['node'].get_virtual_pure() == True:
|
||
|
ret += ' = 0'
|
||
|
if element['node'].get_constant() == True:
|
||
|
ret += ' const'
|
||
|
|
||
|
ret += ';'
|
||
|
ret += '<br/>'
|
||
|
return ret
|
||
|
|
||
|
def generate_stupid_index_page(outFolder, header, footer, myLutinDoc):
|
||
|
# create index.hml :
|
||
|
filename = outFolder + "/index.html"
|
||
|
monkTools.create_directory_of_file(filename);
|
||
|
file = open(filename, "w")
|
||
|
file.write(header)
|
||
|
file.write("<h1>" + myLutinDoc.get_base_doc_node().get_name() + "</h1>");
|
||
|
file.write("<br/>");
|
||
|
file.write("TODO : Main page ...");
|
||
|
file.write("<br/>");
|
||
|
file.write("<br/>");
|
||
|
file.write(footer)
|
||
|
file.close();
|
||
|
|
||
|
def generate_page(outFolder, header, footer, element, namespaceStack=[]):
|
||
|
if element.get_node_type() in ['library', 'application', 'namespace', 'class', 'struct', 'enum', 'union']:
|
||
|
listBase = element.get_all_sub_type(['library', 'application', 'namespace', 'class', 'struct', 'enum', 'union'])
|
||
|
for elem in listBase:
|
||
|
if element.get_node_type() in ['namespace', 'class', 'struct']:
|
||
|
namespaceStack.append(element.get_name())
|
||
|
generate_page(outFolder, header, footer, elem['node'], namespaceStack)
|
||
|
namespaceStack.pop()
|
||
|
else:
|
||
|
generate_page(outFolder, header, footer, elem['node'], namespaceStack)
|
||
|
|
||
|
|
||
|
filename = outFolder + '/' + generate_html_page_name(element, namespaceStack)
|
||
|
monkTools.create_directory_of_file(filename);
|
||
|
file = open(filename, "w")
|
||
|
file.write(header)
|
||
|
file.write("<h1>" + generate_name(element, namespaceStack) + "</h1>");
|
||
|
file.write("<hr/>");
|
||
|
if element.get_node_type() == 'library':
|
||
|
file.write("TODO : the page ...");
|
||
|
elif element.get_node_type() == 'application':
|
||
|
file.write("TODO : the page ...");
|
||
|
elif element.get_node_type() == 'namespace':
|
||
|
file.write("TODO : the page ...");
|
||
|
elif element.get_node_type() == 'class':
|
||
|
# calculate element size :
|
||
|
listBase = element.get_all_sub_type(['methode', 'constructor', 'destructor'])
|
||
|
displayLen = calculate_methode_size(listBase)
|
||
|
|
||
|
file.write("<h2>Constructor and Destructor:</h2>\n")
|
||
|
file.write("<pre>\n");
|
||
|
listBaseConstructor = element.get_all_sub_type(['constructor'])
|
||
|
for elem in listBaseConstructor:
|
||
|
ret = write_methode(elem, namespaceStack, displayLen)
|
||
|
file.write(ret)
|
||
|
listBaseDestructor = element.get_all_sub_type(['destructor'])
|
||
|
for elem in listBaseDestructor:
|
||
|
ret = write_methode(elem, namespaceStack, displayLen)
|
||
|
file.write(ret)
|
||
|
file.write("</pre>\n");
|
||
|
file.write("<br/>\n")
|
||
|
|
||
|
file.write("<h2>Synopsis:</h2>\n")
|
||
|
file.write("<pre>\n");
|
||
|
listBaseMethode = element.get_all_sub_type(['methode'])
|
||
|
displayLen = calculate_methode_size(listBaseMethode)
|
||
|
for elem in listBaseMethode:
|
||
|
ret = write_methode(elem, namespaceStack, displayLen)
|
||
|
file.write(ret)
|
||
|
file.write("</pre>\n")
|
||
|
file.write("<br/>\n")
|
||
|
|
||
|
file.write("<h2>Description:</h2>\n")
|
||
|
|
||
|
# display all functions :
|
||
|
file.write("<h2>Detail:<h2>\n")
|
||
|
for element in listBase:
|
||
|
file.write('<h3><a id="' + str(element['node'].get_uid()) + '">' + element['node'].get_name() + '</a></h3>')
|
||
|
file.write("<pre>\n");
|
||
|
file.write(write_methode(element, namespaceStack, link = False))
|
||
|
file.write("</pre>\n");
|
||
|
#debug.info(str(element['node'].get_doc()));
|
||
|
file.write(parse_doxygen(element['node'].get_doc()));
|
||
|
file.write("<br/>\n");
|
||
|
file.write("<hr/>\n");
|
||
|
|
||
|
elif element.get_node_type() == 'struct':
|
||
|
file.write("TODO : the page ...");
|
||
|
elif element.get_node_type() == 'enum':
|
||
|
file.write("TODO : the page ...");
|
||
|
elif element.get_node_type() == 'union':
|
||
|
file.write("TODO : the page ...");
|
||
|
else:
|
||
|
# not in a specific file ...
|
||
|
debug.warning("might not appear here :'" + element.get_node_type() + "' = '" + element.get_name() + "'")
|
||
|
pass
|
||
|
file.write(footer)
|
||
|
file.close();
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
def generate(myLutinDoc, outFolder) :
|
||
|
myDoc = myLutinDoc.get_base_doc_node()
|
||
|
monkTools.copy_file(monkTools.get_current_path(__file__)+"/theme/base.css", outFolder+"/base.css")
|
||
|
monkTools.copy_file(monkTools.get_current_path(__file__)+"/theme/menu.css", outFolder+"/menu.css")
|
||
|
# create common header
|
||
|
genericHeader = '<!DOCTYPE html>\n'
|
||
|
genericHeader += '<html>\n'
|
||
|
genericHeader += '<head>\n'
|
||
|
genericHeader += ' <meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">\n'
|
||
|
genericHeader += ' <title>' + myDoc.get_name() + ' Library</title>\n'
|
||
|
genericHeader += ' <link rel="stylesheet" href="base.css">\n'
|
||
|
genericHeader += ' <link rel="stylesheet" href="menu.css">\n'
|
||
|
genericHeader += '</head>\n'
|
||
|
genericHeader += '<body>\n'
|
||
|
genericHeader += ' <div class="navbar navbar-fixed-top">\n'
|
||
|
genericHeader += ' <div class="container">\n'
|
||
|
genericHeader += ' <h1>' + myDoc.get_name() + ' Library</h1>\n'
|
||
|
genericHeader += ' <div id="menu">\n'
|
||
|
#genericHeader += ' <h2>' + myDoc.moduleName + '</h2>\n'
|
||
|
genericHeader += generate_menu(myDoc)
|
||
|
#genericHeader += ' <h3> </h3>\n'
|
||
|
genericHeader += ' </div>\n'
|
||
|
genericHeader += " </div>\n"
|
||
|
genericHeader += " </div>\n"
|
||
|
genericHeader += " <div class=\"container\" id=\"content\">\n"
|
||
|
|
||
|
genericFooter = " </div>\n"
|
||
|
genericFooter += "</body>\n"
|
||
|
genericFooter += "</html>\n"
|
||
|
|
||
|
# create index.hml :
|
||
|
generate_stupid_index_page(outFolder, genericHeader, genericFooter, myLutinDoc)
|
||
|
|
||
|
# create the namespace index properties :
|
||
|
generate_page(outFolder, genericHeader, genericFooter, myDoc)
|
||
|
|
||
|
for docInputName,outpath in myLutinDoc.listDocFile :
|
||
|
debug.print_element("doc", myLutinDoc.name, "<==", docInputName)
|
||
|
outputFileName = outFolder + "/" + outpath.replace('/','_') +".html"
|
||
|
debug.debug("output file : " + outputFileName)
|
||
|
monkTools.create_directory_of_file(outputFileName)
|
||
|
inData = monkTools.file_read_data(docInputName)
|
||
|
if inData == "":
|
||
|
continue
|
||
|
outData = genericHeader + codeBB.transcode(inData) + genericFooter
|
||
|
monkTools.file_write_data(outputFileName, outData)
|
||
|
|
||
|
for docInputName,outpath in myLutinDoc.listTutorialFile :
|
||
|
debug.print_element("tutorial", myLutinDoc.name, "<==", docInputName)
|
||
|
outputFileName = outFolder + "/" + outpath+".html"
|
||
|
debug.debug("output file : " + outputFileName)
|
||
|
monkTools.create_directory_of_file(outputFileName)
|
||
|
inData = monkTools.file_read_data(docInputName)
|
||
|
if inData == "":
|
||
|
continue
|
||
|
outData = genericHeader + codeBB.transcode(inData) + genericFooter
|
||
|
monkTools.file_write_data(outputFileName, outData)
|
||
|
|
||
|
|
||
|
|