| /* |
| * Copyright 2017 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "bookmaker.h" |
| |
| #include "SkOSFile.h" |
| #include "SkOSPath.h" |
| |
| #define FPRINTF(...) \ |
| if (fDebugOut) { \ |
| SkDebugf(__VA_ARGS__); \ |
| } \ |
| fprintf(fOut, __VA_ARGS__) |
| |
| const char* SubtopicKeys::kGeneratedSubtopics[] = { |
| kClasses, kConstants, kConstructors, kDefines, |
| kMemberFunctions, kMembers, kOperators, kRelatedFunctions, kStructs, kTypedefs, |
| }; |
| |
| const char* kConstTableStyle = |
| "<style>" "\n" |
| ".td_const td, th { border: 2px solid #dddddd; text-align: left; padding: 8px; }" "\n" |
| ".tr_const tr:nth-child(even) { background-color: #f0f0f0; }" "\n" |
| ".td2_const td:first-child + td { text-align: center; }" "\n" |
| "</style>" "\n"; |
| |
| const char* kTableDeclaration = "<table style='border-collapse: collapse; width: 62.5em'>"; |
| |
| #define kTD_Base "border: 2px solid #dddddd; padding: 8px; " |
| #define kTH_Left "<th style='text-align: left; " kTD_Base "'>" |
| #define kTH_Center "<th style='text-align: center; " kTD_Base "'>" |
| |
| string kTD_Left = " <td style='text-align: left; " kTD_Base "'>"; |
| string kTD_Center = " <td style='text-align: center; " kTD_Base "'>"; |
| string kTR_Dark = " <tr style='background-color: #f0f0f0; '>"; |
| |
| const char* kAllConstTableHeader = " <tr>" kTH_Left "Const</th>" "\n" |
| kTH_Center "Value</th>" "\n" |
| kTH_Left "Description</th>" "</tr>"; |
| const char* kSubConstTableHeader = " <tr>" kTH_Left "Const</th>" "\n" |
| kTH_Center "Value</th>" "\n" |
| kTH_Left "Details</th>" "\n" |
| kTH_Left "Description</th>" "</tr>"; |
| const char* kAllMemberTableHeader = " <tr>" kTH_Left "Type</th>" "\n" |
| kTH_Left "Name</th>" "\n" |
| kTH_Left "Description</th>" "</tr>"; |
| const char* kSubMemberTableHeader = " <tr>" kTH_Left "Type</th>" "\n" |
| kTH_Left "Name</th>" "\n" |
| kTH_Left "Details</th>" "\n" |
| kTH_Left "Description</th>" "</tr>"; |
| const char* kTopicsTableHeader = " <tr>" kTH_Left "Topic</th>" "\n" |
| kTH_Left "Description</th>" "</tr>"; |
| |
| static string html_file_name(string bmhFileName) { |
| SkASSERT("docs" == bmhFileName.substr(0, 4)); |
| SkASSERT('\\' == bmhFileName[4] || '/' == bmhFileName[4]); |
| SkASSERT(".bmh" == bmhFileName.substr(bmhFileName.length() - 4)); |
| string result = bmhFileName.substr(5, bmhFileName.length() - 4 - 5); |
| return result; |
| } |
| |
| string MdOut::anchorDef(string str, string name) { |
| if (fValidate) { |
| string htmlName = html_file_name(fFileName); |
| vector<AnchorDef>& allDefs = fAllAnchorDefs[htmlName]; |
| if (!std::any_of(allDefs.begin(), allDefs.end(), |
| [str](AnchorDef compare) { return compare.fDef == str; } )) { |
| MarkType markType = fLastDef->fMarkType; |
| if (MarkType::kMethod == markType |
| && std::any_of(fLastDef->fChildren.begin(), fLastDef->fChildren.end(), |
| [](const Definition* compare) { |
| return IncompleteAllowed(compare->fMarkType); } )) { |
| markType = MarkType::kDeprecated; |
| } |
| if (MarkType::kMethod == markType && fLastDef->fClone) { |
| markType = MarkType::kDeprecated; // TODO: hack to allow missing reference |
| } |
| allDefs.push_back( { str, markType } ); |
| } |
| } |
| return "<a name='" + str + "'>" + name + "</a>"; |
| } |
| |
| string MdOut::anchorRef(string ref, string name) { |
| if (fValidate) { |
| string htmlName; |
| size_t hashIndex = ref.find('#'); |
| if (string::npos != hashIndex && "https://" != ref.substr(0, 8)) { |
| if (0 == hashIndex) { |
| htmlName = html_file_name(fFileName); |
| } else { |
| htmlName = ref.substr(0, hashIndex); |
| } |
| vector<string>& allRefs = fAllAnchorRefs[htmlName]; |
| string refPart = ref.substr(hashIndex + 1); |
| if (allRefs.end() == std::find(allRefs.begin(), allRefs.end(), refPart)) { |
| allRefs.push_back(refPart); |
| } |
| } |
| } |
| SkASSERT(string::npos != ref.find('#') || string::npos != ref.find("https://")); |
| return "<a href='" + ref + "'>" + name + "</a>"; |
| } |
| |
| string MdOut::anchorLocalRef(string ref, string name) { |
| return this->anchorRef("#" + ref, name); |
| } |
| |
| string MdOut::tableDataCodeRef(string ref, string name) { |
| return kTD_Left + this->anchorRef(ref, "<code>" + name + "</code>") + "</td>"; |
| } |
| |
| string MdOut::tableDataCodeLocalRef(string ref, string name) { |
| return this->tableDataCodeRef("#" + ref, name); |
| } |
| |
| string MdOut::tableDataCodeLocalRef(string name) { |
| return this->tableDataCodeLocalRef(name, name); |
| } |
| |
| string MdOut::tableDataCodeRef(const Definition* ref) { |
| return this->tableDataCodeLocalRef(ref->fFiddle, ref->fName); |
| } |
| |
| string MdOut::tableDataCodeDef(string def, string name) { |
| return kTD_Left + this->anchorDef(def, "<code>" + name + "</code>") + "</td>"; |
| } |
| |
| string MdOut::tableDataCodeDef(const Definition* def) { |
| return this->tableDataCodeDef(def->fFiddle, def->fName); |
| } |
| |
| static string table_data_const(const Definition* def, const char** textStartPtr) { |
| TextParser parser(def); |
| SkAssertResult(parser.skipToEndBracket('\n')); |
| string constant = string(def->fContentStart, (int) (parser.fChar - def->fContentStart)); |
| if (textStartPtr) { |
| *textStartPtr = parser.fChar; |
| } |
| return kTD_Center + constant + "</td>"; |
| } |
| |
| static string out_table_data_description_start() { |
| return kTD_Left; |
| } |
| |
| static string out_table_data_description(string str) { |
| return kTD_Left + str + "</td>"; |
| } |
| |
| static string out_table_data_description(const Definition* def) { |
| return out_table_data_description(string(def->fContentStart, |
| (int) (def->fContentEnd - def->fContentStart))); |
| } |
| |
| static string out_table_data_details(string details) { |
| return kTD_Left + details + "</td>"; |
| } |
| |
| #undef kConstTDBase |
| #undef kTH_Left |
| #undef kTH_Center |
| |
| static void add_ref(string leadingSpaces, string ref, string* result) { |
| *result += leadingSpaces + ref; |
| } |
| |
| static string preformat(string orig) { |
| string result; |
| for (auto c : orig) { |
| if ('<' == c) { |
| result += "<"; |
| } else if ('>' == c) { |
| result += ">"; |
| } else { |
| result += c; |
| } |
| } |
| return result; |
| } |
| |
| static bool all_lower(string ref) { |
| for (auto ch : ref) { |
| if (!islower(ch)) { |
| return false; |
| } |
| } |
| return true; |
| } |
| |
| // from https://stackoverflow.com/questions/3418231/replace-part-of-a-string-with-another-string |
| void replace_all(string& str, const string& from, const string& to) { |
| SkASSERT(!from.empty()); |
| size_t start_pos = 0; |
| while ((start_pos = str.find(from, start_pos)) != std::string::npos) { |
| str.replace(start_pos, from.length(), to); |
| start_pos += to.length(); // In case 'to' contains 'from', like replacing 'x' with 'yx' |
| } |
| } |
| |
| // detail strings are preceded by an example comment to check readability |
| void MdOut::addPopulators() { |
| fPopulators[SubtopicKeys::kClasses].fName = "Class Declarations"; |
| fPopulators[SubtopicKeys::kClasses].fOneLiner = "embedded class members"; |
| fPopulators[SubtopicKeys::kClasses].fDetails = |
| /* SkImageInfo */ "uses C++ classes to declare the public data" |
| " structures and interfaces."; |
| fPopulators[SubtopicKeys::kConstants].fName = "Constants"; |
| fPopulators[SubtopicKeys::kConstants].fOneLiner = "enum and enum class, and their const values"; |
| fPopulators[SubtopicKeys::kConstants].fDetails = |
| /* SkImageInfo */ "related constants are defined by <code>enum</code>," |
| " <code>enum class</code>, <code>#define</code>, <code>const</code>," |
| " and <code>constexpr</code>."; |
| fPopulators[SubtopicKeys::kConstructors].fName = "Constructors"; |
| fPopulators[SubtopicKeys::kConstructors].fOneLiner = "functions that construct"; |
| fPopulators[SubtopicKeys::kConstructors].fDetails = |
| /* SkImageInfo */ "can be constructed or initialized by these functions," |
| " including C++ class constructors."; |
| fPopulators[SubtopicKeys::kDefines].fName = "Defines"; |
| fPopulators[SubtopicKeys::kDefines].fOneLiner = "preprocessor definitions of functions, values"; |
| fPopulators[SubtopicKeys::kDefines].fDetails = |
| /* SkImageInfo */ "uses preprocessor definitions to inline code and constants," |
| " and to abstract platform-specific functionality."; |
| fPopulators[SubtopicKeys::kMemberFunctions].fName = "Functions"; |
| fPopulators[SubtopicKeys::kMemberFunctions].fOneLiner = "global and class member functions"; |
| fPopulators[SubtopicKeys::kMemberFunctions].fDetails = |
| /* SkImageInfo */ "member functions read and modify the structure properties."; |
| fPopulators[SubtopicKeys::kMembers].fName = "Members"; |
| fPopulators[SubtopicKeys::kMembers].fOneLiner = "member values"; |
| fPopulators[SubtopicKeys::kMembers].fDetails = |
| /* SkImageInfo */ "members may be read and written directly without using" |
| " a member function."; |
| fPopulators[SubtopicKeys::kOperators].fName = "Operators"; |
| fPopulators[SubtopicKeys::kOperators].fOneLiner = "operator overloading methods"; |
| fPopulators[SubtopicKeys::kOperators].fDetails = |
| /* SkImageInfo */ "operators inline class member functions with arithmetic" |
| " equivalents."; |
| fPopulators[SubtopicKeys::kRelatedFunctions].fName = "Related Functions"; |
| fPopulators[SubtopicKeys::kRelatedFunctions].fOneLiner = |
| "similar member functions grouped together"; |
| fPopulators[SubtopicKeys::kRelatedFunctions].fDetails = |
| /* SkImageInfo */ "global, <code>struct</code>, and <code>class</code> related member" |
| " functions share a topic."; |
| fPopulators[SubtopicKeys::kStructs].fName = "Struct Declarations"; |
| fPopulators[SubtopicKeys::kStructs].fOneLiner = "embedded struct members"; |
| fPopulators[SubtopicKeys::kStructs].fDetails = |
| /* SkImageInfo */ "uses C++ structs to declare the public data" |
| " structures and interfaces."; |
| fPopulators[SubtopicKeys::kTypedefs].fName = "Typedef Declarations"; |
| fPopulators[SubtopicKeys::kTypedefs].fOneLiner = "types defined by other types"; |
| fPopulators[SubtopicKeys::kTypedefs].fDetails = |
| /* SkImageInfo */ " <code>typedef</code> define a data type."; |
| } |
| |
| Definition* MdOut::checkParentsForMatch(Definition* test, string ref) const { |
| bool isSubtopic = MarkType::kSubtopic == test->fMarkType |
| || MarkType::kTopic == test->fMarkType; |
| do { |
| if (!test->isRoot()) { |
| continue; |
| } |
| bool localTopic = MarkType::kSubtopic == test->fMarkType |
| || MarkType::kTopic == test->fMarkType; |
| if (localTopic != isSubtopic) { |
| continue; |
| } |
| string prefix(isSubtopic ? "_" : "::"); |
| RootDefinition* root = test->asRoot(); |
| string prefixed = root->fName + prefix + ref; |
| if (Definition* def = root->find(prefixed, RootDefinition::AllowParens::kYes)) { |
| return def; |
| } |
| } while ((test = test->fParent)); |
| return nullptr; |
| } |
| |
| // FIXME: preserve inter-line spaces and don't add new ones |
| string MdOut::addReferences(const char* refStart, const char* refEnd, |
| BmhParser::Resolvable resolvable) { |
| string result; |
| MethodParser t(fRoot ? fRoot->fName : string(), fFileName, refStart, refEnd, fLineCount); |
| bool lineStart = true; |
| string ref; |
| string leadingSpaces; |
| int distFromParam = 99; |
| do { |
| ++distFromParam; |
| const char* base = t.fChar; |
| t.skipWhiteSpace(); |
| const char* wordStart = t.fChar; |
| if (BmhParser::Resolvable::kFormula == resolvable && !t.eof() && '"' == t.peek()) { |
| t.next(); |
| t.skipToEndBracket('"'); |
| t.next(); |
| continue; |
| } |
| t.skipToMethodStart(); |
| const char* start = t.fChar; |
| if (wordStart < start) { |
| if (lineStart) { |
| lineStart = false; |
| } else { |
| wordStart = base; |
| } |
| result += string(wordStart, start - wordStart); |
| if ('\n' != result.back()) { |
| while (start > wordStart && '\n' == start[-1]) { |
| result += '\n'; |
| --start; |
| } |
| } |
| } |
| if (lineStart) { |
| lineStart = false; |
| } else { |
| leadingSpaces = string(base, wordStart - base); |
| } |
| t.skipToMethodEnd(resolvable); |
| if (base == t.fChar) { |
| if (!t.eof() && '~' == base[0] && !isalnum(base[1])) { |
| t.next(); |
| } else { |
| break; |
| } |
| } |
| if (start >= t.fChar) { |
| continue; |
| } |
| if (!t.eof() && '"' == t.peek() && start > wordStart && '"' == start[-1]) { |
| continue; |
| } |
| ref = string(start, t.fChar - start); |
| if (const Definition* def = this->isDefined(t, ref, resolvable)) { |
| if (MarkType::kExternal == def->fMarkType) { |
| (void) this->anchorRef("undocumented#" + ref, ""); // for anchor validate |
| add_ref(leadingSpaces, ref, &result); |
| continue; |
| } |
| SkASSERT(def->fFiddle.length()); |
| if (BmhParser::Resolvable::kSimple != resolvable |
| && !t.eof() && '(' == t.peek() && t.strnchr(')', t.fEnd)) { |
| if (!t.skipToEndBracket(')')) { |
| t.reportError("missing close paren"); |
| fAddRefFailed = true; |
| return result; |
| } |
| t.next(); |
| string fullRef = string(start, t.fChar - start); |
| // if _2 etc alternates are defined, look for paren match |
| // may ignore () if ref is all lower case |
| // otherwise flag as error |
| int suffix = '2'; |
| bool foundMatch = false; |
| const Definition* altDef = def; |
| while (altDef && suffix <= '9') { |
| if ((foundMatch = altDef->paramsMatch(fullRef, ref))) { |
| def = altDef; |
| ref = fullRef; |
| break; |
| } |
| string altTest = ref + '_'; |
| altTest += suffix++; |
| altDef = this->isDefined(t, altTest, BmhParser::Resolvable::kOut); |
| } |
| if (suffix > '9') { |
| t.reportError("too many alts"); |
| fAddRefFailed = true; |
| return result; |
| } |
| if (!foundMatch) { |
| if (!(def = this->isDefined(t, fullRef, resolvable))) { |
| if (BmhParser::Resolvable::kFormula == resolvable) { |
| // TODO: look for looser mapping -- if methods name match, look for |
| // unique mapping based on number of parameters |
| // for now, just look for function name match |
| def = this->isDefined(t, ref, resolvable); |
| } |
| if (!def && !result.size()) { |
| t.reportError("missing method"); |
| fAddRefFailed = true; |
| return result; |
| } |
| } |
| ref = fullRef; |
| } |
| } else if (BmhParser::Resolvable::kClone != resolvable && |
| all_lower(ref) && (t.eof() || '(' != t.peek())) { |
| add_ref(leadingSpaces, ref, &result); |
| continue; |
| } |
| if (!def) { |
| t.reportError("missing method"); |
| fAddRefFailed = true; |
| return result; |
| } |
| result += linkRef(leadingSpaces, def, ref, resolvable); |
| continue; |
| } |
| if (!t.eof() && '(' == t.peek()) { |
| if (!t.skipToEndBracket(')')) { |
| t.reportError("missing close paren"); |
| fAddRefFailed = true; |
| return result; |
| } |
| t.next(); |
| ref = string(start, t.fChar - start); |
| if (const Definition* def = this->isDefined(t, ref, BmhParser::Resolvable::kYes)) { |
| SkASSERT(def->fFiddle.length()); |
| result += linkRef(leadingSpaces, def, ref, resolvable); |
| continue; |
| } |
| } |
| // class, struct, and enum start with capitals |
| // methods may start with upper (static) or lower (most) |
| |
| // see if this should have been a findable reference |
| |
| // look for Sk / sk / SK .. |
| if (!ref.compare(0, 2, "Sk") && ref != "Skew" && ref != "Skews" && ref != "Skewing" |
| && ref != "Skip" && ref != "Skips") { |
| if (BmhParser::Resolvable::kOut != resolvable && |
| BmhParser::Resolvable::kFormula != resolvable) { |
| t.reportError("missed Sk prefixed"); |
| fAddRefFailed = true; |
| return result; |
| } |
| } |
| if (!ref.compare(0, 2, "SK")) { |
| if (BmhParser::Resolvable::kOut != resolvable && |
| BmhParser::Resolvable::kFormula != resolvable) { |
| t.reportError("missed SK prefixed"); |
| fAddRefFailed = true; |
| return result; |
| } |
| } |
| if (!isupper(start[0])) { |
| // TODO: |
| // look for all lowercase w/o trailing parens as mistaken method matches |
| // will also need to see if Example Description matches var in example |
| const Definition* def = nullptr; |
| if (fMethod && (def = fMethod->hasParam(ref))) { |
| result += linkRef(leadingSpaces, def, ref, resolvable); |
| fLastParam = def; |
| distFromParam = 0; |
| continue; |
| } else if (!fInDescription && ref[0] != '0' |
| && string::npos != ref.find_first_of("ABCDEFGHIJKLMNOPQRSTUVWXYZ")) { |
| // FIXME: see isDefined(); check to see if fXX is a member of xx.fXX |
| if (('f' != ref[0] && string::npos == ref.find("()")) |
| // || '.' != t.backup(ref.c_str()) |
| && ('k' != ref[0] && string::npos == ref.find("_Private"))) { |
| if ('.' == wordStart[0] && (distFromParam >= 1 && distFromParam <= 16)) { |
| const Definition* paramType = this->findParamType(); |
| if (paramType) { |
| string fullName = paramType->fName + "::" + ref; |
| if (paramType->hasMatch(fullName)) { |
| result += linkRef(leadingSpaces, paramType, ref, resolvable); |
| continue; |
| } |
| } |
| } |
| if (BmhParser::Resolvable::kSimple != resolvable |
| && BmhParser::Resolvable::kOut != resolvable |
| && BmhParser::Resolvable::kFormula != resolvable) { |
| t.reportError("missed camelCase"); |
| fAddRefFailed = true; |
| return result; |
| } |
| } |
| } |
| add_ref(leadingSpaces, ref, &result); |
| continue; |
| } |
| auto topicIter = fBmhParser.fTopicMap.find(ref); |
| if (topicIter != fBmhParser.fTopicMap.end()) { |
| result += linkRef(leadingSpaces, topicIter->second, ref, resolvable); |
| continue; |
| } |
| bool startsSentence = t.sentenceEnd(start); |
| if (!t.eof() && ' ' != t.peek()) { |
| add_ref(leadingSpaces, ref, &result); |
| continue; |
| } |
| if (t.fChar + 1 >= t.fEnd || (!isupper(t.fChar[1]) && startsSentence)) { |
| add_ref(leadingSpaces, ref, &result); |
| continue; |
| } |
| if (isupper(t.fChar[1]) && startsSentence) { |
| TextParser next(t.fFileName, &t.fChar[1], t.fEnd, t.fLineCount); |
| string nextWord(next.fChar, next.wordEnd() - next.fChar); |
| if (this->isDefined(t, nextWord, BmhParser::Resolvable::kYes)) { |
| add_ref(leadingSpaces, ref, &result); |
| continue; |
| } |
| } |
| Definition* def = this->checkParentsForMatch(fSubtopic, ref); |
| if (!def) { |
| def = this->checkParentsForMatch(fRoot, ref); |
| } |
| if (def) { |
| result += this->linkRef(leadingSpaces, def, ref, resolvable); |
| continue; |
| } |
| if (BmhParser::Resolvable::kOut != resolvable && |
| BmhParser::Resolvable::kFormula != resolvable) { |
| t.reportError("undefined reference"); |
| fAddRefFailed = true; |
| } else { |
| add_ref(leadingSpaces, ref, &result); |
| } |
| } while (!t.eof()); |
| return result; |
| } |
| |
| bool MdOut::buildReferences(const IncludeParser& includeParser, const char* docDir, |
| const char* mdFileOrPath) { |
| if (!sk_isdir(mdFileOrPath)) { |
| SkDebugf("must pass directory %s\n", mdFileOrPath); |
| SkDebugf("pass -i SkXXX.h to build references for a single include\n"); |
| return false; |
| } |
| SkOSFile::Iter it(docDir, ".bmh"); |
| for (SkString file; it.next(&file); ) { |
| if (!includeParser.references(file)) { |
| continue; |
| } |
| SkString p = SkOSPath::Join(docDir, file.c_str()); |
| if (!this->buildRefFromFile(p.c_str(), mdFileOrPath)) { |
| SkDebugf("failed to parse %s\n", p.c_str()); |
| return false; |
| } |
| } |
| return true; |
| } |
| |
| bool MdOut::buildStatus(const char* statusFile, const char* outDir) { |
| StatusIter iter(statusFile, ".bmh", StatusFilter::kInProgress); |
| for (string file; iter.next(&file); ) { |
| SkString p = SkOSPath::Join(iter.baseDir().c_str(), file.c_str()); |
| const char* hunk = p.c_str(); |
| if (!this->buildRefFromFile(hunk, outDir)) { |
| SkDebugf("failed to parse %s\n", hunk); |
| return false; |
| } |
| } |
| return true; |
| } |
| |
| bool MdOut::buildRefFromFile(const char* name, const char* outDir) { |
| if (!SkStrEndsWith(name, ".bmh")) { |
| return true; |
| } |
| if (SkStrEndsWith(name, "markup.bmh")) { // don't look inside this for now |
| return true; |
| } |
| if (SkStrEndsWith(name, "illustrations.bmh")) { // don't look inside this for now |
| return true; |
| } |
| fFileName = string(name); |
| string filename(name); |
| if (filename.substr(filename.length() - 4) == ".bmh") { |
| filename = filename.substr(0, filename.length() - 4); |
| } |
| size_t start = filename.length(); |
| while (start > 0 && (isalnum(filename[start - 1]) || '_' == filename[start - 1])) { |
| --start; |
| } |
| string match = filename.substr(start); |
| string header = match; |
| filename = match + ".md"; |
| match += ".bmh"; |
| fOut = nullptr; |
| string fullName; |
| |
| vector<string> keys; |
| keys.reserve(fBmhParser.fTopicMap.size()); |
| for (const auto& it : fBmhParser.fTopicMap) { |
| keys.push_back(it.first); |
| } |
| std::sort(keys.begin(), keys.end()); |
| for (auto key : keys) { |
| string s(key); |
| auto topicDef = fBmhParser.fTopicMap.at(s); |
| if (topicDef->fParent) { |
| continue; |
| } |
| if (string::npos == topicDef->fFileName.rfind(match)) { |
| continue; |
| } |
| if (!fOut) { |
| fullName = outDir; |
| if ('/' != fullName.back()) { |
| fullName += '/'; |
| } |
| fullName += filename; |
| fOut = fopen(filename.c_str(), "wb"); |
| if (!fOut) { |
| SkDebugf("could not open output file %s\n", fullName.c_str()); |
| return false; |
| } |
| if (false) { // try inlining the style |
| FPRINTF("%s", kConstTableStyle); |
| } |
| size_t underscorePos = header.find('_'); |
| if (string::npos != underscorePos) { |
| header.replace(underscorePos, 1, " "); |
| } |
| SkASSERT(string::npos == header.find('_')); |
| FPRINTF("%s", header.c_str()); |
| this->lfAlways(1); |
| FPRINTF("==="); |
| } |
| const Definition* prior = nullptr; |
| this->markTypeOut(topicDef, &prior); |
| } |
| if (fOut) { |
| this->writePending(); |
| fclose(fOut); |
| fflush(fOut); |
| if (this->writtenFileDiffers(filename, fullName)) { |
| fOut = fopen(fullName.c_str(), "wb"); |
| int writtenSize; |
| const char* written = ReadToBuffer(filename, &writtenSize); |
| fwrite(written, 1, writtenSize, fOut); |
| fclose(fOut); |
| fflush(fOut); |
| SkDebugf("wrote updated %s\n", fullName.c_str()); |
| } |
| remove(filename.c_str()); |
| fOut = nullptr; |
| } |
| return !fAddRefFailed; |
| } |
| |
| static bool contains_referenced_child(const Definition* found, const vector<string>& refs) { |
| for (auto child : found->fChildren) { |
| if (refs.end() != std::find_if(refs.begin(), refs.end(), |
| [child](string def) { return child->fName == def; } )) { |
| return true; |
| } |
| if (contains_referenced_child(child, refs)) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| void MdOut::checkAnchors() { |
| int missing = 0; |
| for (auto bmhFile : fAllAnchorRefs) { |
| auto defIter = fAllAnchorDefs.find(bmhFile.first); |
| SkASSERT(fAllAnchorDefs.end() != defIter); |
| vector<AnchorDef>& allDefs = defIter->second; |
| std::sort(allDefs.begin(), allDefs.end(), |
| [](const AnchorDef& a, const AnchorDef& b) { return a.fDef < b.fDef; } ); |
| std::sort(bmhFile.second.begin(), bmhFile.second.end()); |
| auto allDefsIter = allDefs.begin(); |
| auto allRefsIter = bmhFile.second.begin(); |
| for (;;) { |
| bool allDefsEnded = allDefsIter == allDefs.end(); |
| bool allRefsEnded = allRefsIter == bmhFile.second.end(); |
| if (allDefsEnded && allRefsEnded) { |
| break; |
| } |
| if (allRefsEnded || (!allDefsEnded && allDefsIter->fDef < *allRefsIter)) { |
| if (MarkType::kParam != allDefsIter->fMarkType |
| && !IncompleteAllowed(allDefsIter->fMarkType)) { |
| // If undocumented but parent or child is referred to: good enough for now |
| bool goodEnough = false; |
| if ("undocumented" == defIter->first) { |
| auto iter = fBmhParser.fTopicMap.find(allDefsIter->fDef); |
| if (fBmhParser.fTopicMap.end() != iter) { |
| const Definition* found = iter->second; |
| if (string::npos != found->fFileName.find("undocumented")) { |
| const Definition* parent = found; |
| while ((parent = parent->fParent)) { |
| if (bmhFile.second.end() != std::find_if(bmhFile.second.begin(), |
| bmhFile.second.end(), |
| [parent](string def) { |
| return parent->fName == def; } )) { |
| goodEnough = true; |
| break; |
| } |
| } |
| if (!goodEnough) { |
| goodEnough = contains_referenced_child(found, bmhFile.second); |
| } |
| } |
| } |
| } |
| if (!goodEnough) { |
| SkDebugf("missing ref %s %s\n", defIter->first.c_str(), |
| allDefsIter->fDef.c_str()); |
| missing++; |
| } |
| } |
| allDefsIter++; |
| } else if (allDefsEnded || (!allRefsEnded && allDefsIter->fDef > *allRefsIter)) { |
| if (fBmhParser.fExternals.end() == std::find_if(fBmhParser.fExternals.begin(), |
| fBmhParser.fExternals.end(), [allRefsIter](const RootDefinition& root) { |
| return *allRefsIter != root.fName; } )) { |
| SkDebugf("missing def %s %s\n", bmhFile.first.c_str(), allRefsIter->c_str()); |
| missing++; |
| } |
| allRefsIter++; |
| } else { |
| SkASSERT(!allDefsEnded); |
| SkASSERT(!allRefsEnded); |
| SkASSERT(allDefsIter->fDef == *allRefsIter); |
| allDefsIter++; |
| allRefsIter++; |
| } |
| if (missing >= 10) { |
| missing = 0; |
| } |
| } |
| } |
| } |
| |
| bool MdOut::checkParamReturnBody(const Definition* def) { |
| TextParser paramBody(def); |
| const char* descriptionStart = paramBody.fChar; |
| if (!islower(descriptionStart[0]) && !isdigit(descriptionStart[0])) { |
| paramBody.skipToNonName(); |
| string ref = string(descriptionStart, paramBody.fChar - descriptionStart); |
| if (!this->isDefined(paramBody, ref, BmhParser::Resolvable::kYes)) { |
| string errorStr = MarkType::kReturn == def->fMarkType ? "return" : "param"; |
| errorStr += " description must start with lower case"; |
| paramBody.reportError(errorStr.c_str()); |
| fAddRefFailed = true; |
| return false; |
| } |
| } |
| if ('.' == paramBody.fEnd[-1]) { |
| paramBody.reportError("make param description a phrase; should not end with period"); |
| fAddRefFailed = true; |
| return false; |
| } |
| return true; |
| } |
| |
| void MdOut::childrenOut(Definition* def, const char* start) { |
| const char* end; |
| fLineCount = def->fLineCount; |
| if (MarkType::kEnumClass == def->fMarkType) { |
| fEnumClass = def; |
| } |
| BmhParser::Resolvable resolvable = this->resolvable(def); |
| const Definition* prior = nullptr; |
| for (auto& child : def->fChildren) { |
| if (MarkType::kPhraseParam == child->fMarkType) { |
| continue; |
| } |
| end = child->fStart; |
| if (BmhParser::Resolvable::kNo != resolvable) { |
| this->resolveOut(start, end, resolvable); |
| } |
| this->markTypeOut(child, &prior); |
| start = child->fTerminator; |
| } |
| if (BmhParser::Resolvable::kNo != resolvable) { |
| end = def->fContentEnd; |
| this->resolveOut(start, end, resolvable); |
| } |
| if (MarkType::kEnumClass == def->fMarkType) { |
| fEnumClass = nullptr; |
| } |
| } |
| |
| // output header for subtopic for all consts: name, value, short descriptions (#Line) |
| // output link to in context #Const with moderate description |
| void MdOut::summaryOut(const Definition* def, MarkType markType, string name) { |
| this->writePending(); |
| SkASSERT(TableState::kNone == fTableState); |
| this->mdHeaderOut(3); |
| FPRINTF("%s", name.c_str()); |
| this->lfAlways(2); |
| FPRINTF("%s", kTableDeclaration); // <table> with style info |
| this->lfAlways(1); |
| FPRINTF("%s", MarkType::kConst == markType ? kAllConstTableHeader : kAllMemberTableHeader); |
| this->lfAlways(1); |
| bool odd = true; |
| for (auto child : def->fChildren) { |
| if (markType != child->fMarkType) { |
| continue; |
| } |
| auto oneLiner = std::find_if(child->fChildren.begin(), child->fChildren.end(), |
| [](const Definition* test){ return MarkType::kLine == test->fMarkType; } ); |
| if (child->fChildren.end() == oneLiner) { |
| child->reportError<void>("missing #Line"); |
| continue; |
| } |
| FPRINTF("%s", odd ? kTR_Dark.c_str() : " <tr>"); |
| this->lfAlways(1); |
| if (MarkType::kConst == markType) { |
| FPRINTF("%s", tableDataCodeRef(child).c_str()); |
| this->lfAlways(1); |
| FPRINTF("%s", table_data_const(child, nullptr).c_str()); |
| } else { |
| string memberType; |
| string memberName = this->getMemberTypeName(child, &memberType); |
| SkASSERT(MarkType::kMember == markType); |
| FPRINTF("%s", out_table_data_description(memberType).c_str()); |
| this->lfAlways(1); |
| FPRINTF("%s", tableDataCodeLocalRef(memberName).c_str()); |
| } |
| this->lfAlways(1); |
| FPRINTF("%s", out_table_data_description(*oneLiner).c_str()); |
| this->lfAlways(1); |
| FPRINTF("%s", " </tr>"); |
| this->lfAlways(1); |
| odd = !odd; |
| } |
| FPRINTF("</table>"); |
| this->lfAlways(1); |
| } |
| |
| Definition* MdOut::csParent() { |
| if (!fRoot) { |
| return nullptr; |
| } |
| Definition* csParent = fRoot->csParent(); |
| if (!csParent) { |
| const Definition* topic = fRoot; |
| while (topic && MarkType::kTopic != topic->fMarkType) { |
| topic = topic->fParent; |
| } |
| for (auto child : topic->fChildren) { |
| if (child->isStructOrClass() || MarkType::kTypedef == child->fMarkType) { |
| csParent = child; |
| break; |
| } |
| } |
| SkASSERT(csParent || string::npos == fRoot->fFileName.find("Sk") |
| || string::npos != fRoot->fFileName.find("SkBlendMode_Reference.bmh")); |
| } |
| return csParent; |
| } |
| |
| const Definition* MdOut::findParamType() { |
| SkASSERT(fMethod); |
| TextParser parser(fMethod->fFileName, fMethod->fStart, fMethod->fContentStart, |
| fMethod->fLineCount); |
| string lastFull; |
| do { |
| parser.skipToAlpha(); |
| if (parser.eof()) { |
| return nullptr; |
| } |
| const char* word = parser.fChar; |
| parser.skipFullName(); |
| SkASSERT(!parser.eof()); |
| string name = string(word, parser.fChar - word); |
| if (fLastParam->fName == name) { |
| const Definition* paramType = this->isDefined(parser, lastFull, |
| BmhParser::Resolvable::kOut); |
| return paramType; |
| } |
| if (isupper(name[0])) { |
| lastFull = name; |
| } |
| } while (true); |
| return nullptr; |
| } |
| |
| string MdOut::getMemberTypeName(const Definition* def, string* memberType) { |
| TextParser parser(def->fFileName, def->fStart, def->fContentStart, |
| def->fLineCount); |
| parser.skipExact("#Member"); |
| parser.skipWhiteSpace(); |
| const char* typeStart = parser.fChar; |
| const char* typeEnd = nullptr; |
| const char* nameStart = nullptr; |
| const char* nameEnd = nullptr; |
| do { |
| parser.skipToWhiteSpace(); |
| if (nameStart) { |
| nameEnd = parser.fChar; |
| } |
| if (parser.eof()) { |
| break; |
| } |
| const char* spaceLoc = parser.fChar; |
| if (parser.skipWhiteSpace()) { |
| typeEnd = spaceLoc; |
| nameStart = parser.fChar; |
| } |
| } while (!parser.eof()); |
| SkASSERT(typeEnd); |
| *memberType = string(typeStart, (int) (typeEnd - typeStart)); |
| replace_all(*memberType, " ", " "); |
| SkASSERT(nameStart); |
| SkASSERT(nameEnd); |
| return string(nameStart, (int) (nameEnd - nameStart)); |
| } |
| |
| bool MdOut::HasDetails(const Definition* def) { |
| for (auto child : def->fChildren) { |
| if (MarkType::kDetails == child->fMarkType) { |
| return true; |
| } |
| if (MdOut::HasDetails(child)) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| void MdOut::htmlOut(string s) { |
| SkASSERT(string::npos != s.find('<')); |
| FPRINTF("%s", s.c_str()); |
| } |
| |
| const Definition* MdOut::isDefinedByParent(RootDefinition* root, string ref) { |
| if (ref == root->fName) { |
| return root; |
| } |
| if (const Definition* definition = root->find(ref, RootDefinition::AllowParens::kYes)) { |
| return definition; |
| } |
| Definition* test = root; |
| bool isSubtopic = MarkType::kSubtopic == root->fMarkType |
| || MarkType::kTopic == root->fMarkType; |
| do { |
| if (!test->isRoot()) { |
| continue; |
| } |
| bool testIsSubtopic = MarkType::kSubtopic == test->fMarkType |
| || MarkType::kTopic == test->fMarkType; |
| if (isSubtopic != testIsSubtopic) { |
| continue; |
| } |
| RootDefinition* root = test->asRoot(); |
| for (auto& leaf : root->fBranches) { |
| if (ref == leaf.first) { |
| return leaf.second; |
| } |
| const Definition* definition = leaf.second->find(ref, |
| RootDefinition::AllowParens::kYes); |
| if (definition) { |
| return definition; |
| } |
| } |
| string prefix = isSubtopic ? "_" : "::"; |
| string prefixed = root->fName + prefix + ref; |
| if (Definition* definition = root->find(prefixed, RootDefinition::AllowParens::kYes)) { |
| return definition; |
| } |
| if (isSubtopic && isupper(prefixed[0])) { |
| auto topicIter = fBmhParser.fTopicMap.find(prefixed); |
| if (topicIter != fBmhParser.fTopicMap.end()) { |
| return topicIter->second; |
| } |
| } |
| if (isSubtopic) { |
| string fiddlePrefixed = root->fFiddle + "_" + ref; |
| auto topicIter = fBmhParser.fTopicMap.find(fiddlePrefixed); |
| if (topicIter != fBmhParser.fTopicMap.end()) { |
| return topicIter->second; |
| } |
| } |
| } while ((test = test->fParent)); |
| return nullptr; |
| } |
| |
| const Definition* MdOut::isDefined(const TextParser& parser, string ref, |
| BmhParser::Resolvable resolvable) { |
| auto rootIter = fBmhParser.fClassMap.find(ref); |
| if (rootIter != fBmhParser.fClassMap.end()) { |
| return &rootIter->second; |
| } |
| auto typedefIter = fBmhParser.fTypedefMap.find(ref); |
| if (typedefIter != fBmhParser.fTypedefMap.end()) { |
| return &typedefIter->second; |
| } |
| auto enumIter = fBmhParser.fEnumMap.find(ref); |
| if (enumIter != fBmhParser.fEnumMap.end()) { |
| return &enumIter->second; |
| } |
| auto constIter = fBmhParser.fConstMap.find(ref); |
| if (constIter != fBmhParser.fConstMap.end()) { |
| return &constIter->second; |
| } |
| auto methodIter = fBmhParser.fMethodMap.find(ref); |
| if (methodIter != fBmhParser.fMethodMap.end()) { |
| return &methodIter->second; |
| } |
| auto aliasIter = fBmhParser.fAliasMap.find(ref); |
| if (aliasIter != fBmhParser.fAliasMap.end()) { |
| return aliasIter->second; |
| } |
| auto defineIter = fBmhParser.fDefineMap.find(ref); |
| if (defineIter != fBmhParser.fDefineMap.end()) { |
| return &defineIter->second; |
| } |
| for (const auto& external : fBmhParser.fExternals) { |
| if (external.fName == ref) { |
| return &external; |
| } |
| } |
| if (const Definition* definition = this->isDefinedByParent(fRoot, ref)) { |
| return definition; |
| } |
| if (const Definition* definition = this->isDefinedByParent(fSubtopic, ref)) { |
| return definition; |
| } |
| size_t doubleColon = ref.find("::"); |
| if (string::npos != doubleColon) { |
| string className = ref.substr(0, doubleColon); |
| auto classIter = fBmhParser.fClassMap.find(className); |
| if (classIter != fBmhParser.fClassMap.end()) { |
| RootDefinition& classDef = classIter->second; |
| const Definition* result = classDef.find(ref, RootDefinition::AllowParens::kYes); |
| if (result) { |
| return result; |
| } |
| } |
| |
| } |
| if (!ref.compare(0, 2, "SK") || !ref.compare(0, 3, "sk_") |
| || (('k' == ref[0] || 'g' == ref[0] || 'f' == ref[0]) && |
| ref.length() > 1 && isupper(ref[1]))) { |
| // try with a prefix |
| if ('k' == ref[0]) { |
| for (auto& iter : fBmhParser.fEnumMap) { |
| auto def = iter.second.find(ref, RootDefinition::AllowParens::kYes); |
| if (def) { |
| return def; |
| } |
| } |
| if (fEnumClass) { |
| string fullName = fEnumClass->fName + "::" + ref; |
| for (auto child : fEnumClass->fChildren) { |
| if (fullName == child->fName) { |
| return child; |
| } |
| } |
| } |
| if (string::npos != ref.find("_Private")) { |
| return nullptr; |
| } |
| } |
| if ('f' == ref[0]) { |
| // FIXME : find def associated with prior, e.g.: r.fX where 'SkPoint r' was earlier |
| // need to have pushed last resolve on stack to do this |
| // for now, just try to make sure that it's there and error if not |
| if ('.' != parser.backup(ref.c_str())) { |
| parser.reportError("fX member undefined"); |
| fAddRefFailed = true; |
| return nullptr; |
| } |
| } else { |
| if (BmhParser::Resolvable::kOut != resolvable && |
| BmhParser::Resolvable::kFormula != resolvable) { |
| parser.reportError("SK undefined"); |
| fAddRefFailed = true; |
| } |
| return nullptr; |
| } |
| } |
| if (isupper(ref[0])) { |
| auto topicIter = fBmhParser.fTopicMap.find(ref); |
| if (topicIter != fBmhParser.fTopicMap.end()) { |
| return topicIter->second; |
| } |
| size_t pos = ref.find('_'); |
| if (string::npos != pos) { |
| // see if it is defined by another base class |
| string className(ref, 0, pos); |
| auto classIter = fBmhParser.fClassMap.find(className); |
| if (classIter != fBmhParser.fClassMap.end()) { |
| if (const Definition* definition = classIter->second.find(ref, |
| RootDefinition::AllowParens::kYes)) { |
| return definition; |
| } |
| } |
| auto enumIter = fBmhParser.fEnumMap.find(className); |
| if (enumIter != fBmhParser.fEnumMap.end()) { |
| if (const Definition* definition = enumIter->second.find(ref, |
| RootDefinition::AllowParens::kYes)) { |
| return definition; |
| } |
| } |
| if (BmhParser::Resolvable::kOut != resolvable && |
| BmhParser::Resolvable::kFormula != resolvable) { |
| parser.reportError("_ undefined"); |
| fAddRefFailed = true; |
| } |
| return nullptr; |
| } |
| } |
| return nullptr; |
| } |
| |
| string MdOut::linkName(const Definition* ref) const { |
| string result = ref->fName; |
| size_t under = result.find('_'); |
| if (string::npos != under) { |
| string classPart = result.substr(0, under); |
| string namePart = result.substr(under + 1, result.length()); |
| if (fRoot && (fRoot->fName == classPart |
| || (fRoot->fParent && fRoot->fParent->fName == classPart))) { |
| result = namePart; |
| } |
| } |
| replace_all(result, "::", "_"); |
| return result; |
| } |
| |
| // for now, hard-code to html links |
| // def should not include SkXXX_ |
| string MdOut::linkRef(string leadingSpaces, const Definition* def, |
| string ref, BmhParser::Resolvable resolvable) { |
| string buildup; |
| string refName; |
| const string* str = &def->fFiddle; |
| SkASSERT(str->length() > 0); |
| string classPart = *str; |
| bool globalEnumMember = false; |
| if (MarkType::kAlias == def->fMarkType) { |
| def = def->fParent; |
| SkASSERT(def); |
| SkASSERT(MarkType::kSubtopic == def->fMarkType |
| || MarkType::kTopic == def->fMarkType |
| || MarkType::kConst == def->fMarkType); |
| } |
| if (MarkType::kSubtopic == def->fMarkType) { |
| const Definition* topic = def->topicParent(); |
| SkASSERT(topic); |
| classPart = topic->fName; |
| refName = def->fName; |
| } else if (MarkType::kTopic == def->fMarkType) { |
| refName = def->fName; |
| } else { |
| if ('k' == (*str)[0] && string::npos != str->find("_Sk")) { |
| globalEnumMember = true; |
| } else { |
| SkASSERT("Sk" == str->substr(0, 2) || "SK" == str->substr(0, 2) |
| // FIXME: kitchen sink catch below, need to do better |
| || string::npos != def->fFileName.find("undocumented")); |
| size_t under = str->find('_'); |
| classPart = string::npos != under ? str->substr(0, under) : *str; |
| } |
| refName = def->fFiddle; |
| } |
| bool classMatch = fRoot->fFileName == def->fFileName; |
| SkASSERT(fRoot); |
| SkASSERT(fRoot->fFileName.length()); |
| if (!classMatch) { |
| string filename = def->fFileName; |
| if (filename.substr(filename.length() - 4) == ".bmh") { |
| filename = filename.substr(0, filename.length() - 4); |
| } |
| size_t start = filename.length(); |
| while (start > 0 && (isalnum(filename[start - 1]) || '_' == filename[start - 1])) { |
| --start; |
| } |
| buildup = filename.substr(start); |
| } |
| buildup += "#" + refName; |
| if (MarkType::kParam == def->fMarkType) { |
| const Definition* parent = def->fParent; |
| SkASSERT(MarkType::kMethod == parent->fMarkType); |
| buildup = '#' + parent->fFiddle + '_' + ref; |
| } |
| string refOut(ref); |
| if (!globalEnumMember) { |
| std::replace(refOut.begin(), refOut.end(), '_', ' '); |
| } |
| if (ref.length() > 2 && islower(ref[0]) && "()" == ref.substr(ref.length() - 2)) { |
| refOut = refOut.substr(0, refOut.length() - 2); |
| } |
| string result = leadingSpaces + this->anchorRef(buildup, refOut); |
| if (BmhParser::Resolvable::kClone == resolvable && MarkType::kMethod == def->fMarkType && |
| def->fCloned && !def->fClone) { |
| bool found = false; |
| string match = def->fName; |
| if ("()" == match.substr(match.length() - 2)) { |
| match = match.substr(0, match.length() - 2); |
| } |
| match += '_'; |
| auto classIter = fBmhParser.fClassMap.find(classPart); |
| if (fBmhParser.fClassMap.end() != classIter) { |
| for (char num = '2'; num <= '9'; ++num) { |
| string clone = match + num; |
| const auto& leafIter = classIter->second.fLeaves.find(clone); |
| if (leafIter != classIter->second.fLeaves.end()) { |
| result += "<sup>" + this->anchorRef(buildup + "_" + num, |
| string("[") + num + "]") + "</sup>"; |
| found = true; |
| } |
| } |
| } |
| if (!found) { |
| SkDebugf(""); // convenient place to set a breakpoint |
| } |
| } |
| return result; |
| } |
| |
| static bool writeTableEnd(MarkType markType, Definition* def, const Definition** prior) { |
| return markType != def->fMarkType && *prior && markType == (*prior)->fMarkType; |
| } |
| |
| void MdOut::markTypeOut(Definition* def, const Definition** prior) { |
| string printable = def->printableName(); |
| const char* textStart = def->fContentStart; |
| bool lookForOneLiner = false; |
| // #Param and #Const don't have markers to say when the last is seen, so detect that by looking |
| // for a change in type. |
| if (writeTableEnd(MarkType::kParam, def, prior) || writeTableEnd(MarkType::kConst, def, prior) |
| || writeTableEnd(MarkType::kMember, def, prior)) { |
| this->writePending(); |
| FPRINTF("</table>"); |
| this->lf(2); |
| fTableState = TableState::kNone; |
| } |
| fLastDef = def; |
| switch (def->fMarkType) { |
| case MarkType::kAlias: |
| break; |
| case MarkType::kAnchor: { |
| if (fColumn > 0) { |
| this->writeSpace(); |
| } |
| this->writePending(); |
| TextParser parser(def); |
| const char* start = parser.fChar; |
| parser.skipToEndBracket((string(" ") + def->fMC + " ").c_str()); |
| string anchorText(start, parser.fChar - start); |
| parser.skipExact((string(" ") + def->fMC + " ").c_str()); |
| string anchorLink(parser.fChar, parser.fEnd - parser.fChar); |
| this->htmlOut(anchorRef(anchorLink, anchorText)); |
| } break; |
| case MarkType::kBug: |
| break; |
| case MarkType::kClass: |
| case MarkType::kStruct: { |
| fRoot = def->asRoot(); |
| this->mdHeaderOut(1); |
| if (MarkType::kStruct == def->fMarkType) { |
| this->htmlOut(anchorDef(def->fFiddle, "Struct " + def->fName)); |
| } else { |
| this->htmlOut(anchorDef(this->linkName(def), "Class " + def->fName)); |
| } |
| this->lf(1); |
| if (string::npos != fRoot->fFileName.find("undocumented")) { |
| break; |
| } |
| // if class or struct contains constants, and doesn't contain subtopic kConstant, add it |
| // and add a child populate |
| const Definition* subtopic = def->subtopicParent(); |
| const Definition* topic = def->topicParent(); |
| for (auto item : SubtopicKeys::kGeneratedSubtopics) { |
| string subname; |
| if (subtopic != topic) { |
| subname = subtopic->fName + '_'; |
| } |
| subname += item; |
| if (fRoot->populator(item).fMembers.size() |
| && !std::any_of(fRoot->fChildren.begin(), fRoot->fChildren.end(), |
| [subname](const Definition* child) { |
| return MarkType::kSubtopic == child->fMarkType |
| && subname == child->fName; |
| } )) { |
| // generate subtopic |
| this->mdHeaderOut(2); |
| this->htmlOut(anchorDef(subname, item)); |
| this->lf(2); |
| // generate populate |
| this->subtopicOut(item); |
| } |
| } |
| } break; |
| case MarkType::kCode: |
| this->lfAlways(2); |
| FPRINTF("<pre style=\"padding: 1em 1em 1em 1em;" |
| "width: 62.5em; background-color: #f0f0f0\">"); |
| this->lf(1); |
| fResolveAndIndent = true; |
| break; |
| case MarkType::kColumn: |
| this->writePending(); |
| if (fInList) { |
| FPRINTF(" <td>"); |
| } else { |
| FPRINTF("| "); |
| } |
| break; |
| case MarkType::kComment: |
| break; |
| case MarkType::kMember: |
| case MarkType::kConst: { |
| bool isConst = MarkType::kConst == def->fMarkType; |
| lookForOneLiner = false; |
| fWroteSomething = false; |
| // output consts for one parent with moderate descriptions |
| // optional link to subtopic with longer descriptions, examples |
| if (TableState::kNone == fTableState) { |
| SkASSERT(!*prior || (isConst && MarkType::kConst != (*prior)->fMarkType) |
| || (!isConst && MarkType::kMember != (*prior)->fMarkType)); |
| this->mdHeaderOut(3); |
| FPRINTF("%s", this->fPopulators[isConst ? SubtopicKeys::kConstants : |
| SubtopicKeys::kMembers].fName.c_str()); |
| this->lfAlways(2); |
| FPRINTF("%s", kTableDeclaration); |
| fTableState = TableState::kRow; |
| fOddRow = true; |
| this->lfAlways(1); |
| // look ahead to see if the details column has data or not |
| fHasDetails = MdOut::HasDetails(def->fParent); |
| FPRINTF("%s", fHasDetails ? \ |
| (isConst ? kSubConstTableHeader : kSubMemberTableHeader) : \ |
| (isConst ? kAllConstTableHeader : kAllMemberTableHeader)); |
| this->lfAlways(1); |
| } |
| if (TableState::kRow == fTableState) { |
| this->writePending(); |
| FPRINTF("%s", fOddRow ? kTR_Dark.c_str() : " <tr>"); |
| fOddRow = !fOddRow; |
| this->lfAlways(1); |
| fTableState = TableState::kColumn; |
| } |
| this->writePending(); |
| if (isConst) { |
| // TODO: if fHasDetails is true, could defer def and issue a ref instead |
| // unclear if this is a good idea or not |
| FPRINTF("%s", this->tableDataCodeDef(def).c_str()); |
| this->lfAlways(1); |
| FPRINTF("%s", table_data_const(def, &textStart).c_str()); |
| } else { |
| string memberType; |
| string memberName = this->getMemberTypeName(def, &memberType); |
| FPRINTF("%s", out_table_data_description(memberType).c_str()); |
| this->lfAlways(1); |
| FPRINTF("%s", tableDataCodeDef(def->fFiddle, memberName).c_str()); |
| } |
| this->lfAlways(1); |
| if (fHasDetails) { |
| string details; |
| auto subtopic = std::find_if(def->fChildren.begin(), def->fChildren.end(), |
| [](const Definition* test){ |
| return MarkType::kDetails == test->fMarkType; } ); |
| if (def->fChildren.end() != subtopic) { |
| string subtopicName = string((*subtopic)->fContentStart, |
| (int) ((*subtopic)->fContentEnd - (*subtopic)->fContentStart)); |
| const Definition* parentSubtopic = def->subtopicParent(); |
| SkASSERT(parentSubtopic); |
| string fullName = parentSubtopic->fFiddle + '_' + subtopicName; |
| if (fBmhParser.fTopicMap.end() == fBmhParser.fTopicMap.find(fullName)) { |
| (*subtopic)->reportError<void>("missing #Details subtopic"); |
| } |
| // subtopicName = parentSubtopic->fName + '_' + subtopicName; |
| string noUnderscores = subtopicName; |
| replace_all(noUnderscores, "_", " "); |
| details = this->anchorLocalRef(subtopicName, noUnderscores) + " "; |
| } |
| FPRINTF("%s", out_table_data_details(details).c_str()); |
| this->lfAlways(1); |
| } |
| lookForOneLiner = true; // if description is empty, use oneLiner data |
| FPRINTF("%s", out_table_data_description_start().c_str()); // start of Description |
| this->lfAlways(1); |
| } break; |
| case MarkType::kDefine: |
| this->mdHeaderOut(2); |
| this->htmlOut(anchorDef(def->fFiddle, "Define " + def->fName)); |
| this->lf(2); |
| break; |
| case MarkType::kDeprecated: |
| this->writeString("Deprecated."); |
| this->lf(2); |
| break; |
| case MarkType::kDescription: |
| fInDescription = true; |
| this->writePending(); |
| FPRINTF("%s", "<div>"); |
| break; |
| case MarkType::kDetails: |
| break; |
| case MarkType::kDuration: |
| break; |
| case MarkType::kEnum: |
| case MarkType::kEnumClass: |
| this->mdHeaderOut(2); |
| this->htmlOut(anchorDef(def->fFiddle, "Enum " + def->fName)); |
| this->lf(2); |
| break; |
| case MarkType::kExample: { |
| this->mdHeaderOut(3); |
| FPRINTF("%s", "Example\n" |
| "\n"); |
| fHasFiddle = true; |
| bool showGpu = false; |
| bool gpuAndCpu = false; |
| const Definition* platform = def->hasChild(MarkType::kPlatform); |
| if (platform) { |
| TextParser platParse(platform); |
| fHasFiddle = !platParse.strnstr("!fiddle", platParse.fEnd); |
| showGpu = platParse.strnstr("gpu", platParse.fEnd); |
| if (showGpu) { |
| gpuAndCpu = platParse.strnstr("cpu", platParse.fEnd); |
| } |
| } |
| if (fHasFiddle) { |
| SkASSERT(def->fHash.length() > 0); |
| FPRINTF("<div><fiddle-embed name=\"%s\"", def->fHash.c_str()); |
| if (showGpu) { |
| FPRINTF("%s", " gpu=\"true\""); |
| if (gpuAndCpu) { |
| FPRINTF("%s", " cpu=\"true\""); |
| } |
| } |
| FPRINTF("%s", ">"); |
| } else { |
| SkASSERT(def->fHash.length() == 0); |
| FPRINTF("%s", "<pre style=\"padding: 1em 1em 1em 1em; font-size: 13px" |
| " width: 62.5em; background-color: #f0f0f0\">"); |
| this->lfAlways(1); |
| if (def->fWrapper.length() > 0) { |
| FPRINTF("%s", def->fWrapper.c_str()); |
| } |
| fLiteralAndIndent = true; |
| } |
| } break; |
| case MarkType::kExperimental: |
| writeString("Experimental."); |
| this->lf(2); |
| break; |
| case MarkType::kExternal: |
| break; |
| case MarkType::kFile: |
| break; |
| case MarkType::kFormula: |
| break; |
| case MarkType::kFunction: |
| break; |
| case MarkType::kHeight: |
| break; |
| case MarkType::kIllustration: { |
| string illustName = "Illustrations_" + def->fParent->fFiddle; |
| string number = string(def->fContentStart, def->length()); |
| if (number.length() && "1" != number) { |
| illustName += "_" + number; |
| } |
| auto illustIter = fBmhParser.fTopicMap.find(illustName); |
| SkASSERT(fBmhParser.fTopicMap.end() != illustIter); |
| Definition* illustDef = illustIter->second; |
| SkASSERT(MarkType::kSubtopic == illustDef->fMarkType); |
| SkASSERT(1 == illustDef->fChildren.size()); |
| Definition* illustExample = illustDef->fChildren[0]; |
| SkASSERT(MarkType::kExample == illustExample->fMarkType); |
| string hash = illustExample->fHash; |
| SkASSERT("" != hash); |
| string title; |
| this->writePending(); |
| FPRINTF("![%s](https://fiddle.skia.org/i/%s_raster.png \"%s\")", |
| def->fName.c_str(), hash.c_str(), title.c_str()); |
| this->lf(2); |
| } break; |
| case MarkType::kImage: |
| break; |
| case MarkType::kIn: |
| break; |
| case MarkType::kLegend: |
| break; |
| case MarkType::kLine: |
| break; |
| case MarkType::kLink: |
| break; |
| case MarkType::kList: |
| fInList = true; |
| fTableState = TableState::kRow; |
| this->lfAlways(2); |
| FPRINTF("%s", "<table>"); |
| this->lf(1); |
| break; |
| case MarkType::kLiteral: |
| break; |
| case MarkType::kMarkChar: |
| fBmhParser.fMC = def->fContentStart[0]; |
| break; |
| case MarkType::kMethod: { |
| string method_name = def->methodName(); |
| string formattedStr = def->formatFunction(Definition::Format::kIncludeReturn); |
| this->lfAlways(2); |
| this->htmlOut(anchorDef(def->fFiddle, "")); |
| if (!def->isClone()) { |
| this->mdHeaderOutLF(2, 1); |
| FPRINTF("%s", method_name.c_str()); |
| } |
| this->lf(2); |
| |
| // TODO: put in css spec that we can define somewhere else (if markup supports that) |
| // TODO: 50em below should match limit = 80 in formatFunction() |
| this->writePending(); |
| string preformattedStr = preformat(formattedStr); |
| string references = this->addReferences(&preformattedStr.front(), |
| &preformattedStr.back() + 1, BmhParser::Resolvable::kSimple); |
| preformattedStr = references; |
| this->htmlOut("<pre style=\"padding: 1em 1em 1em 1em; width: 62.5em;" |
| "background-color: #f0f0f0\">\n" + preformattedStr + "\n" + "</pre>"); |
| this->lf(2); |
| fTableState = TableState::kNone; |
| fMethod = def; |
| } break; |
| case MarkType::kNoExample: |
| break; |
| case MarkType::kNoJustify: |
| break; |
| case MarkType::kOutdent: |
| break; |
| case MarkType::kParam: { |
| if (TableState::kNone == fTableState) { |
| SkASSERT(!*prior || MarkType::kParam != (*prior)->fMarkType); |
| this->mdHeaderOut(3); |
| this->htmlOut( |
| "Parameters\n" |
| "\n" |
| "<table>" |
| ); |
| this->lf(1); |
| fTableState = TableState::kRow; |
| } |
| if (TableState::kRow == fTableState) { |
| FPRINTF(" <tr>"); |
| this->lf(1); |
| fTableState = TableState::kColumn; |
| } |
| TextParser paramParser(def->fFileName, def->fStart, def->fContentStart, |
| def->fLineCount); |
| paramParser.skipWhiteSpace(); |
| SkASSERT(paramParser.startsWith("#Param")); |
| paramParser.next(); // skip hash |
| paramParser.skipToNonName(); // skip Param |
| paramParser.skipSpace(); |
| const char* paramName = paramParser.fChar; |
| paramParser.skipToSpace(); |
| string paramNameStr(paramName, (int) (paramParser.fChar - paramName)); |
| if (!this->checkParamReturnBody(def)) { |
| *prior = def; |
| return; |
| } |
| string refNameStr = def->fParent->fFiddle + "_" + paramNameStr; |
| this->htmlOut(" <td>" + anchorDef(refNameStr, |
| "<code><strong>" + paramNameStr + "</strong></code>") + "</td>"); |
| this->lfAlways(1); |
| FPRINTF(" <td>"); |
| } break; |
| case MarkType::kPhraseDef: |
| // skip text and children |
| *prior = def; |
| return; |
| case MarkType::kPhraseParam: |
| SkDebugf(""); // convenient place to set a breakpoint |
| break; |
| case MarkType::kPhraseRef: |
| if (fPhraseParams.end() != fPhraseParams.find(def->fName)) { |
| if (fColumn > 0) { |
| this->writeSpace(); |
| } |
| this->writeString(fPhraseParams[def->fName]); |
| if (isspace(def->fContentStart[0])) { |
| this->writeSpace(); |
| } |
| } else if (fBmhParser.fPhraseMap.end() == fBmhParser.fPhraseMap.find(def->fName)) { |
| def->reportError<void>("missing phrase definition"); |
| fAddRefFailed = true; |
| } else { |
| if (fColumn) { |
| SkASSERT(' ' >= def->fStart[0]); |
| this->writeSpace(); |
| } |
| Definition* phraseRef = fBmhParser.fPhraseMap.find(def->fName)->second; |
| // def->fChildren are parameters to substitute phraseRef->fChildren, |
| // phraseRef->fChildren has both param defines and references |
| // def->fChildren must have the same number of entries as phaseRef->fChildren |
| // which are kPhraseParam, and substitute one for one |
| // Then, each kPhraseRef in phaseRef looks up the key and value |
| fPhraseParams.clear(); |
| auto refKidsIter = phraseRef->fChildren.begin(); |
| for (auto child : def->fChildren) { |
| if (MarkType::kPhraseParam != child->fMarkType) { |
| // more work to do to support other types |
| this->reportError("phrase ref child must be param"); |
| } |
| do { |
| if (refKidsIter == phraseRef->fChildren.end()) { |
| this->reportError("phrase def missing param"); |
| break; |
| } |
| if (MarkType::kPhraseRef == (*refKidsIter)->fMarkType) { |
| continue; |
| } |
| if (MarkType::kPhraseParam != (*refKidsIter)->fMarkType) { |
| this->reportError("unexpected type in phrase def children"); |
| break; |
| } |
| fPhraseParams[(*refKidsIter)->fName] = child->fName; |
| break; |
| } while (true); |
| } |
| this->childrenOut(phraseRef, phraseRef->fContentStart); |
| fPhraseParams.clear(); |
| if (' ' >= def->fContentStart[0] && !fPendingLF) { |
| this->writeSpace(); |
| } |
| } |
| break; |
| case MarkType::kPlatform: |
| break; |
| case MarkType::kPopulate: { |
| SkASSERT(MarkType::kSubtopic == def->fParent->fMarkType); |
| string name = def->fParent->fName; |
| if (string::npos != name.find(SubtopicKeys::kOverview)) { |
| this->subtopicsOut(def->fParent); |
| } else { |
| this->subtopicOut(name); |
| } |
| } break; |
| case MarkType::kPrivate: |
| break; |
| case MarkType::kReturn: |
| this->mdHeaderOut(3); |
| FPRINTF("Return Value"); |
| if (!this->checkParamReturnBody(def)) { |
| *prior = def; |
| return; |
| } |
| this->lf(2); |
| break; |
| case MarkType::kRow: |
| if (fInList) { |
| FPRINTF(" <tr>"); |
| this->lf(1); |
| } |
| break; |
| case MarkType::kSeeAlso: |
| this->mdHeaderOut(3); |
| FPRINTF("See Also"); |
| this->lf(2); |
| break; |
| case MarkType::kSet: |
| break; |
| case MarkType::kStdOut: { |
| TextParser code(def); |
| this->mdHeaderOut(4); |
| FPRINTF( |
| "Example Output\n" |
| "\n" |
| "~~~~"); |
| this->lfAlways(1); |
| code.skipSpace(); |
| while (!code.eof()) { |
| const char* end = code.trimmedLineEnd(); |
| FPRINTF("%.*s\n", (int) (end - code.fChar), code.fChar); |
| code.skipToLineStart(); |
| } |
| FPRINTF("~~~~"); |
| this->lf(2); |
| } break; |
| case MarkType::kSubstitute: |
| break; |
| case MarkType::kSubtopic: |
| fSubtopic = def->asRoot(); |
| this->mdHeaderOut(2); |
| if (SubtopicKeys::kOverview == def->fName) { |
| this->writeString(def->fName); |
| } else { |
| this->htmlOut(anchorDef(def->fName, printable)); |
| } |
| this->lf(2); |
| // if a subtopic child is const, generate short table of const name, value, line desc |
| if (std::any_of(def->fChildren.begin(), def->fChildren.end(), |
| [](Definition* child){return MarkType::kConst == child->fMarkType;})) { |
| this->summaryOut(def, MarkType::kConst, fPopulators[SubtopicKeys::kConstants].fName); |
| } |
| // if a subtopic child is member, generate short table of const name, value, line desc |
| if (std::any_of(def->fChildren.begin(), def->fChildren.end(), |
| [](Definition* child){return MarkType::kMember == child->fMarkType;})) { |
| this->summaryOut(def, MarkType::kMember, fPopulators[SubtopicKeys::kMembers].fName); |
| } |
| break; |
| case MarkType::kTable: |
| this->lf(2); |
| break; |
| case MarkType::kTemplate: |
| break; |
| case MarkType::kText: |
| break; |
| case MarkType::kToDo: |
| break; |
| case MarkType::kTopic: { |
| auto found = std::find_if(def->fChildren.begin(), def->fChildren.end(), |
| [](Definition* test) { return test->isStructOrClass(); } ); |
| bool hasClassOrStruct = def->fChildren.end() != found; |
| fRoot = hasClassOrStruct ? (*found)->asRoot() : def->asRoot(); |
| fSubtopic = def->asRoot(); |
| bool isUndocumented = string::npos != def->fFileName.find("undocumented"); |
| if (!isUndocumented) { |
| this->populateTables(def, fRoot); |
| } |
| this->mdHeaderOut(1); |
| this->htmlOut(anchorDef(this->linkName(def), printable)); |
| this->lf(1); |
| } break; |
| case MarkType::kTypedef: |
| this->mdHeaderOut(2); |
| this->htmlOut(anchorDef(def->fFiddle, "Typedef " + def->fName)); |
| this->lf(1); |
| break; |
| case MarkType::kUnion: |
| break; |
| case MarkType::kVolatile: |
| break; |
| case MarkType::kWidth: |
| break; |
| default: |
| SkDebugf("fatal error: MarkType::k%s unhandled in %s()\n", |
| BmhParser::kMarkProps[(int) def->fMarkType].fName, __func__); |
| SkASSERT(0); // handle everything |
| break; |
| } |
| this->childrenOut(def, textStart); |
| switch (def->fMarkType) { // post child work, at least for tables |
| case MarkType::kAnchor: |
| if (fColumn > 0) { |
| this->writeSpace(); |
| } |
| break; |
| case MarkType::kClass: |
| case MarkType::kStruct: |
| if (TableState::kNone != fTableState) { |
| this->writePending(); |
| FPRINTF("</table>"); |
| this->lf(2); |
| fTableState = TableState::kNone; |
| } |
| if (def->csParent()) { |
| fRoot = def->csParent()->asRoot(); |
| } |
| break; |
| case MarkType::kCode: |
| fIndent = 0; |
| this->lf(1); |
| this->writePending(); |
| FPRINTF("</pre>"); |
| this->lf(2); |
| fResolveAndIndent = false; |
| break; |
| case MarkType::kColumn: |
| if (fInList) { |
| this->writePending(); |
| FPRINTF("</td>"); |
| this->lfAlways(1); |
| } else { |
| FPRINTF(" "); |
| } |
| break; |
| case MarkType::kDescription: |
| this->writePending(); |
| FPRINTF("</div>"); |
| fInDescription = false; |
| break; |
| case MarkType::kEnum: |
| case MarkType::kEnumClass: |
| if (TableState::kNone != fTableState) { |
| this->writePending(); |
| FPRINTF("</table>"); |
| this->lf(2); |
| fTableState = TableState::kNone; |
| } |
| break; |
| case MarkType::kExample: |
| this->writePending(); |
| if (fHasFiddle) { |
| FPRINTF("</fiddle-embed></div>"); |
| } else { |
| this->lfAlways(1); |
| if (def->fWrapper.length() > 0) { |
| FPRINTF("}"); |
| this->lfAlways(1); |
| } |
| FPRINTF("</pre>"); |
| } |
| this->lf(2); |
| fLiteralAndIndent = false; |
| break; |
| case MarkType::kLink: |
| this->writeString("</a>"); |
| this->writeSpace(); |
| break; |
| case MarkType::kList: |
| fInList = false; |
| this->writePending(); |
| SkASSERT(TableState::kNone != fTableState); |
| FPRINTF("</table>"); |
| this->lf(2); |
| fTableState = TableState::kNone; |
| break; |
| case MarkType::kLegend: { |
| SkASSERT(def->fChildren.size() == 1); |
| const Definition* row = def->fChildren[0]; |
| SkASSERT(MarkType::kRow == row->fMarkType); |
| size_t columnCount = row->fChildren.size(); |
| SkASSERT(columnCount > 0); |
| this->writePending(); |
| for (size_t index = 0; index < columnCount; ++index) { |
| FPRINTF("| --- "); |
| } |
| FPRINTF(" |"); |
| this->lf(1); |
| } break; |
| case MarkType::kMethod: |
| fMethod = nullptr; |
| this->lfAlways(2); |
| FPRINTF("---"); |
| this->lf(2); |
| break; |
| case MarkType::kConst: |
| case MarkType::kMember: |
| if (lookForOneLiner && !fWroteSomething) { |
| auto oneLiner = std::find_if(def->fChildren.begin(), def->fChildren.end(), |
| [](const Definition* test){ return MarkType::kLine == test->fMarkType; } ); |
| if (def->fChildren.end() != oneLiner) { |
| TextParser parser(*oneLiner); |
| parser.skipWhiteSpace(); |
| parser.trimEnd(); |
| FPRINTF("%.*s", (int) (parser.fEnd - parser.fChar), parser.fChar); |
| } |
| lookForOneLiner = false; |
| } |
| case MarkType::kParam: |
| SkASSERT(TableState::kColumn == fTableState); |
| fTableState = TableState::kRow; |
| this->writePending(); |
| FPRINTF("</td>"); |
| this->lfAlways(1); |
| FPRINTF(" </tr>"); |
| this->lfAlways(1); |
| break; |
| case MarkType::kReturn: |
| case MarkType::kSeeAlso: |
| this->lf(2); |
| break; |
| case MarkType::kRow: |
| if (fInList) { |
| FPRINTF(" </tr>"); |
| } else { |
| FPRINTF("|"); |
| } |
| this->lf(1); |
| break; |
| case MarkType::kTable: |
| this->lf(2); |
| break; |
| case MarkType::kPhraseDef: |
| break; |
| case MarkType::kPrivate: |
| break; |
| case MarkType::kSubtopic: |
| SkASSERT(def); |
| do { |
| def = def->fParent; |
| } while (def && MarkType::kTopic != def->fMarkType |
| && MarkType::kSubtopic != def->fMarkType); |
| SkASSERT(def); |
| fSubtopic = def->asRoot(); |
| break; |
| case MarkType::kTopic: |
| fSubtopic = nullptr; |
| break; |
| default: |
| break; |
| } |
| *prior = def; |
| } |
| |
| void MdOut::mdHeaderOutLF(int depth, int lf) { |
| this->lfAlways(lf); |
| for (int index = 0; index < depth; ++index) { |
| FPRINTF("#"); |
| } |
| FPRINTF(" "); |
| } |
| |
| void MdOut::populateOne(Definition* def, |
| unordered_map<string, RootDefinition::SubtopicContents>& populator) { |
| if (MarkType::kConst == def->fMarkType) { |
| populator[SubtopicKeys::kConstants].fMembers.push_back(def); |
| return; |
| } |
| if (MarkType::kEnum == def->fMarkType || MarkType::kEnumClass == def->fMarkType) { |
| populator[SubtopicKeys::kConstants].fMembers.push_back(def); |
| return; |
| } |
| if (MarkType::kDefine == def->fMarkType) { |
| populator[SubtopicKeys::kDefines].fMembers.push_back(def); |
| return; |
| } |
| if (MarkType::kMember == def->fMarkType) { |
| populator[SubtopicKeys::kMembers].fMembers.push_back(def); |
| return; |
| } |
| if (MarkType::kTypedef == def->fMarkType) { |
| populator[SubtopicKeys::kTypedefs].fMembers.push_back(def); |
| return; |
| } |
| if (MarkType::kMethod != def->fMarkType) { |
| return; |
| } |
| if (def->fClone) { |
| return; |
| } |
| if (Definition::MethodType::kConstructor == def->fMethodType |
| || Definition::MethodType::kDestructor == def->fMethodType) { |
| populator[SubtopicKeys::kConstructors].fMembers.push_back(def); |
| return; |
| } |
| if (Definition::MethodType::kOperator == def->fMethodType) { |
| populator[SubtopicKeys::kOperators].fMembers.push_back(def); |
| return; |
| } |
| populator[SubtopicKeys::kMemberFunctions].fMembers.push_back(def); |
| const Definition* csParent = this->csParent(); |
| if (csParent) { |
| if (0 == def->fName.find(csParent->fName + "::Make") |
| || 0 == def->fName.find(csParent->fName + "::make")) { |
| populator[SubtopicKeys::kConstructors].fMembers.push_back(def); |
| return; |
| } |
| } |
| for (auto item : def->fChildren) { |
| if (MarkType::kIn == item->fMarkType) { |
| string name(item->fContentStart, item->fContentEnd - item->fContentStart); |
| populator[name].fMembers.push_back(def); |
| populator[name].fShowClones = true; |
| break; |
| } |
| } |
| } |
| |
| void MdOut::populateTables(const Definition* def, RootDefinition* root) { |
| for (auto child : def->fChildren) { |
| if (MarkType::kSubtopic == child->fMarkType) { |
| string name = child->fName; |
| bool builtInTopic = name == SubtopicKeys::kOverview; |
| for (auto item : SubtopicKeys::kGeneratedSubtopics) { |
| builtInTopic |= name == item; |
| } |
| if (!builtInTopic) { |
| string subname; |
| const Definition* subtopic = child->subtopicParent(); |
| if (subtopic) { |
| subname = subtopic->fName + '_'; |
| } |
| builtInTopic = name == subname + SubtopicKeys::kOverview; |
| for (auto item : SubtopicKeys::kGeneratedSubtopics) { |
| builtInTopic |= name == subname + item; |
| } |
| if (!builtInTopic) { |
| root->populator(SubtopicKeys::kRelatedFunctions).fMembers.push_back(child); |
| } |
| } |
| this->populateTables(child, root); |
| continue; |
| } |
| if (child->isStructOrClass()) { |
| if (fClassStack.size() > 0) { |
| root->populator(MarkType::kStruct != child->fMarkType ? SubtopicKeys::kClasses : |
| SubtopicKeys::kStructs).fMembers.push_back(child); |
| } |
| fClassStack.push_back(child); |
| this->populateTables(child, child->asRoot()); |
| fClassStack.pop_back(); |
| continue; |
| } |
| if (MarkType::kEnum == child->fMarkType || MarkType::kEnumClass == child->fMarkType) { |
| this->populateTables(child, root); |
| } |
| this->populateOne(child, root->fPopulators); |
| } |
| } |
| |
| void MdOut::resolveOut(const char* start, const char* end, BmhParser::Resolvable resolvable) { |
| if ((BmhParser::Resolvable::kLiteral == resolvable || fLiteralAndIndent || |
| fResolveAndIndent) && end > start) { |
| int linefeeds = 0; |
| while ('\n' == *start) { |
| ++linefeeds; |
| ++start; |
| } |
| if (fResolveAndIndent && linefeeds) { |
| this->lf(linefeeds); |
| } |
| const char* spaceStart = start; |
| while (' ' == *start) { |
| ++start; |
| } |
| if (start > spaceStart) { |
| fIndent = start - spaceStart; |
| } |
| } |
| if (BmhParser::Resolvable::kLiteral == resolvable || fLiteralAndIndent) { |
| this->writeBlockTrim(end - start, start); |
| if ('\n' == end[-1]) { |
| this->lf(1); |
| } |
| fIndent = 0; |
| return; |
| } |
| // FIXME: this needs the markdown character present when the def was defined, |
| // not the last markdown character the parser would have seen... |
| while (fBmhParser.fMC == end[-1]) { |
| --end; |
| } |
| if (start >= end) { |
| return; |
| } |
| string resolved = this->addReferences(start, end, resolvable); |
| trim_end_spaces(resolved); |
| if (resolved.length()) { |
| TextParser paragraph(fFileName, &*resolved.begin(), &*resolved.end(), fLineCount); |
| while (!paragraph.eof()) { |
| while ('\n' == paragraph.peek()) { |
| paragraph.next(); |
| if (paragraph.eof()) { |
| return; |
| } |
| } |
| const char* lineStart = paragraph.fChar; |
| paragraph.skipWhiteSpace(); |
| const char* contentStart = paragraph.fChar; |
| if (fResolveAndIndent && contentStart > lineStart) { |
| this->writePending(); |
| this->indentToColumn(contentStart - lineStart); |
| } |
| paragraph.skipToEndBracket('\n'); |
| ptrdiff_t lineLength = paragraph.fChar - contentStart; |
| if (lineLength) { |
| while (lineLength && contentStart[lineLength - 1] <= ' ') { |
| --lineLength; |
| } |
| string str(contentStart, lineLength); |
| this->writeString(str.c_str()); |
| fWroteSomething = !!lineLength; |
| } |
| if (paragraph.eof()) { |
| break; |
| } |
| if ('\n' == paragraph.next()) { |
| int linefeeds = 1; |
| if (!paragraph.eof() && '\n' == paragraph.peek()) { |
| linefeeds = 2; |
| } |
| this->lf(linefeeds); |
| } |
| } |
| } |
| } |
| |
| void MdOut::rowOut(const char* name, string description, bool literalName) { |
| FPRINTF("%s", fOddRow ? kTR_Dark.c_str() : " <tr>"); |
| this->lfAlways(1); |
| FPRINTF("%s", kTD_Left.c_str()); |
| if (literalName) { |
| if (strlen(name)) { |
| this->writeString(name); |
| } |
| } else { |
| this->resolveOut(name, name + strlen(name), BmhParser::Resolvable::kYes); |
| } |
| FPRINTF("</td>"); |
| this->lfAlways(1); |
| FPRINTF("%s", kTD_Left.c_str()); |
| this->resolveOut(&description.front(), &description.back() + 1, BmhParser::Resolvable::kYes); |
| FPRINTF("</td>"); |
| this->lfAlways(1); |
| FPRINTF(" </tr>"); |
| this->lfAlways(1); |
| fOddRow = !fOddRow; |
| } |
| |
| void MdOut::subtopicsOut(Definition* def) { |
| Definition* csParent = def->csParent(); |
| const Definition* subtopicParent = def->subtopicParent(); |
| const Definition* topicParent = def->topicParent(); |
| SkASSERT(subtopicParent); |
| this->lfAlways(1); |
| FPRINTF("%s", kTableDeclaration); |
| this->lfAlways(1); |
| FPRINTF("%s", kTopicsTableHeader); |
| this->lfAlways(1); |
| fOddRow = true; |
| for (auto item : SubtopicKeys::kGeneratedSubtopics) { |
| for (auto entry : fRoot->populator(item).fMembers) { |
| if ((csParent && entry->csParent() == csParent) |
| || entry->subtopicParent() == subtopicParent) { |
| auto popItem = fPopulators.find(item); |
| string description = popItem->second.fOneLiner; |
| if (SubtopicKeys::kConstructors == item) { |
| description += " " + fRoot->fName; |
| } |
| string subtopic; |
| if (subtopicParent != topicParent) { |
| subtopic = subtopicParent->fName + '_'; |
| } |
| string link = this->anchorLocalRef(subtopic + item, popItem->second.fName); |
| this->rowOut(link.c_str(), description, true); |
| break; |
| } |
| } |
| } |
| FPRINTF("</table>"); |
| this->lfAlways(1); |
| } |
| |
| void MdOut::subtopicOut(string name) { |
| const Definition* topicParent = fSubtopic ? fSubtopic->topicParent() : nullptr; |
| Definition* csParent = this->csParent(); |
| if (!csParent) { |
| auto csIter = std::find_if(topicParent->fChildren.begin(), topicParent->fChildren.end(), |
| [](const Definition* def){ return MarkType::kEnum == def->fMarkType |
| || MarkType::kEnumClass == def->fMarkType; } ); |
| SkASSERT(topicParent->fChildren.end() != csIter); |
| csParent = *csIter; |
| } |
| SkASSERT(csParent); |
| this->lfAlways(1); |
| if (fPopulators.end() != fPopulators.find(name)) { |
| const SubtopicDescriptions& tableDescriptions = this->populator(name); |
| this->anchorDef(name, tableDescriptions.fName); |
| this->lfAlways(1); |
| if (tableDescriptions.fDetails.length()) { |
| string details = csParent->fName; |
| details += " " + tableDescriptions.fDetails; |
| this->writeString(details); |
| this->lfAlways(1); |
| } |
| } else { |
| this->anchorDef(name, name); |
| this->lfAlways(1); |
| } |
| FPRINTF("%s", kTableDeclaration); |
| this->lfAlways(1); |
| FPRINTF("%s", kTopicsTableHeader); |
| this->lfAlways(1); |
| fOddRow = true; |
| std::map<string, const Definition*> items; |
| const RootDefinition::SubtopicContents& tableContents = fRoot->populator(name.c_str()); |
| auto& data = tableContents.fMembers; |
| for (auto entry : data) { |
| if (entry->csParent() != csParent && entry->topicParent() != topicParent) { |
| continue; |
| } |
| size_t start = entry->fName.find_last_of("::"); |
| if (MarkType::kConst == entry->fMarkType && entry->fParent |
| && MarkType::kEnumClass == entry->fParent->fMarkType |
| && string::npos != start && start > 1) { |
| start = entry->fName.substr(0, start - 1).rfind("::"); |
| } |
| string entryName = entry->fName.substr(string::npos == start ? 0 : start + 1); |
| items[entryName] = entry; |
| } |
| for (auto entry : items) { |
| if (entry.second->fDeprecated) { |
| continue; |
| } |
| const Definition* oneLiner = nullptr; |
| for (auto child : entry.second->fChildren) { |
| if (MarkType::kLine == child->fMarkType) { |
| oneLiner = child; |
| break; |
| } |
| } |
| if (!oneLiner) { |
| TextParser parser(entry.second->fFileName, entry.second->fStart, |
| entry.second->fContentStart, entry.second->fLineCount); |
| parser.reportError("missing #Line"); |
| continue; |
| } |
| string keyName = entry.first; |
| TextParser dummy(entry.second); // for reporting errors, which we won't do |
| if (!this->isDefined(dummy, keyName, BmhParser::Resolvable::kOut)) { |
| keyName = entry.second->fName; |
| size_t doubleColon = keyName.find("::"); |
| SkASSERT(string::npos != doubleColon); |
| keyName = keyName.substr(doubleColon + 2); |
| } |
| this->rowOut(keyName.c_str(), string(oneLiner->fContentStart, |
| oneLiner->fContentEnd - oneLiner->fContentStart), false); |
| if (tableContents.fShowClones && entry.second->fCloned) { |
| int cloneNo = 2; |
| string builder = entry.second->fName; |
| if ("()" == builder.substr(builder.length() - 2)) { |
| builder = builder.substr(0, builder.length() - 2); |
| } |
| builder += '_'; |
| this->rowOut("", |
| preformat(entry.second->formatFunction(Definition::Format::kOmitReturn)), true); |
| do { |
| string match = builder + to_string(cloneNo); |
| auto child = csParent->findClone(match); |
| if (!child) { |
| break; |
| } |
| this->rowOut("", |
| preformat(child->formatFunction(Definition::Format::kOmitReturn)), true); |
| } while (++cloneNo); |
| } |
| } |
| FPRINTF("</table>"); |
| this->lf(2); |
| } |