for SVG images,
// which do not support the alt attribute
{
*pAltValue = att.value;
}
else
{
result+=" ";
result+=att.name;
result+="=\""+convertToXML(att.value)+"\"";
}
}
else if (att.name=="nowrap") // In XHTML, attribute minimization is forbidden, and the nowrap attribute must be defined as .
{
result+=" ";
result+=att.name;
result+="=\"nowrap\"";
}
}
return result;
}
//-------------------------------------------------------------------------
HtmlDocVisitor::HtmlDocVisitor(TextStream &t,CodeOutputInterface &ci,
const Definition *ctx)
: DocVisitor(DocVisitor_Html), m_t(t), m_ci(ci), m_insidePre(FALSE),
m_hide(FALSE), m_ctx(ctx)
{
if (ctx) m_langExt=ctx->getDefFileExtension();
}
//--------------------------------------
// visitor functions for leaf nodes
//--------------------------------------
void HtmlDocVisitor::visit(DocWord *w)
{
//printf("word: %s\n",qPrint(w->word()));
if (m_hide) return;
filter(w->word());
}
void HtmlDocVisitor::visit(DocLinkedWord *w)
{
if (m_hide) return;
//printf("linked word: %s\n",qPrint(w->word()));
startLink(w->ref(),w->file(),w->relPath(),w->anchor(),w->tooltip());
filter(w->word());
endLink();
}
void HtmlDocVisitor::visit(DocWhiteSpace *w)
{
if (m_hide) return;
if (m_insidePre)
{
m_t << w->chars();
}
else
{
m_t << " ";
}
}
void HtmlDocVisitor::visit(DocSymbol *s)
{
if (m_hide) return;
const char *res = HtmlEntityMapper::instance()->html(s->symbol());
if (res)
{
m_t << res;
}
else
{
err("HTML: non supported HTML-entity found: %s\n",HtmlEntityMapper::instance()->html(s->symbol(),TRUE));
}
}
void HtmlDocVisitor::visit(DocEmoji *s)
{
if (m_hide) return;
const char *res = EmojiEntityMapper::instance()->unicode(s->index());
if (res)
{
m_t << ""<";
}
else
{
m_t << s->name();
}
}
void HtmlDocVisitor::writeObfuscatedMailAddress(const QCString &url)
{
m_t << "";
}
void HtmlDocVisitor::visit(DocURL *u)
{
if (m_hide) return;
if (u->isEmail()) // mail address
{
QCString url = u->url();
writeObfuscatedMailAddress(url);
uint size=5,i;
for (i=0;i.nosp@m. ";
i+=size;
if (size==5) size=4; else size=5;
}
m_t << "";
}
else // web address
{
m_t << "url() << "\">";
filter(u->url());
m_t << " ";
}
}
void HtmlDocVisitor::visit(DocLineBreak *br)
{
if (m_hide) return;
m_t << " attribs()) << " />\n";
}
void HtmlDocVisitor::visit(DocHorRuler *hr)
{
if (m_hide) return;
forceEndParagraph(hr);
m_t << " attribs()) << " />\n";
forceStartParagraph(hr);
}
void HtmlDocVisitor::visit(DocStyleChange *s)
{
if (m_hide) return;
switch (s->style())
{
case DocStyleChange::Bold:
if (s->enable()) m_t << "attribs()) << ">"; else m_t << " ";
break;
case DocStyleChange::S:
if (s->enable()) m_t << "attribs()) << ">"; else m_t << " ";
break;
case DocStyleChange::Strike:
if (s->enable()) m_t << "attribs()) << ">"; else m_t << " ";
break;
case DocStyleChange::Del:
if (s->enable()) m_t << "attribs()) << ">"; else m_t << "";
break;
case DocStyleChange::Underline:
if (s->enable()) m_t << "attribs()) << ">"; else m_t << " ";
break;
case DocStyleChange::Ins:
if (s->enable()) m_t << "attribs()) << ">"; else m_t << " ";
break;
case DocStyleChange::Italic:
if (s->enable()) m_t << "attribs()) << ">"; else m_t << " ";
break;
case DocStyleChange::Code:
if (s->enable()) m_t << "attribs()) << ">"; else m_t << "
";
break;
case DocStyleChange::Subscript:
if (s->enable()) m_t << "attribs()) << ">"; else m_t << " ";
break;
case DocStyleChange::Superscript:
if (s->enable()) m_t << "attribs()) << ">"; else m_t << " ";
break;
case DocStyleChange::Center:
if (s->enable())
{
forceEndParagraph(s);
m_t << "attribs()) << ">";
}
else
{
m_t << " ";
forceStartParagraph(s);
}
break;
case DocStyleChange::Small:
if (s->enable()) m_t << "attribs()) << ">"; else m_t << " ";
break;
case DocStyleChange::Preformatted:
if (s->enable())
{
forceEndParagraph(s);
m_t << "attribs()) << ">";
m_insidePre=TRUE;
}
else
{
m_insidePre=FALSE;
m_t << " ";
forceStartParagraph(s);
}
break;
case DocStyleChange::Div:
if (s->enable())
{
forceEndParagraph(s);
m_t << "attribs()) << ">";
}
else
{
m_t << "
";
forceStartParagraph(s);
}
break;
case DocStyleChange::Span:
if (s->enable()) m_t << "attribs()) << ">"; else m_t << " ";
break;
}
}
static void visitPreCaption(TextStream &t, DocVerbatim *s)
{
if (s->hasCaption())
{
t << "\n";
}
}
static void visitPostCaption(TextStream &t, DocVerbatim *s)
{
if (s->hasCaption())
{
t << "
\n";
}
}
static void visitCaption(HtmlDocVisitor *parent, DocNodeList &children)
{
for (const auto &n : children) n->accept(parent);
}
void HtmlDocVisitor::visit(DocVerbatim *s)
{
if (m_hide) return;
QCString lang = m_langExt;
if (!s->language().isEmpty()) // explicit language setting
{
lang = s->language();
}
SrcLangExt langExt = getLanguageFromCodeLang(lang);
switch(s->type())
{
case DocVerbatim::Code:
forceEndParagraph(s);
m_ci.startCodeFragment("DoxyCode");
getCodeParser(lang).parseCode(m_ci,
s->context(),
s->text(),
langExt,
s->isExample(),
s->exampleFile(),
0, // fileDef
-1, // startLine
-1, // endLine
FALSE, // inlineFragment
0, // memberDef
TRUE, // show line numbers
m_ctx // search context
);
m_ci.endCodeFragment("DoxyCode");
forceStartParagraph(s);
break;
case DocVerbatim::Verbatim:
forceEndParagraph(s);
m_t << "";
filter(s->text());
m_t << " ";
forceStartParagraph(s);
break;
case DocVerbatim::HtmlOnly:
{
if (s->isBlock()) forceEndParagraph(s);
m_t << s->text();
if (s->isBlock()) forceStartParagraph(s);
}
break;
case DocVerbatim::ManOnly:
case DocVerbatim::LatexOnly:
case DocVerbatim::XmlOnly:
case DocVerbatim::RtfOnly:
case DocVerbatim::DocbookOnly:
/* nothing */
break;
case DocVerbatim::Dot:
{
static int dotindex = 1;
QCString fileName(4096);
forceEndParagraph(s);
fileName.sprintf("%s%d%s",
qPrint(Config_getString(HTML_OUTPUT)+"/inline_dotgraph_"),
dotindex++,
".dot"
);
std::ofstream file(fileName.str(),std::ofstream::out | std::ofstream::binary);
if (!file.is_open())
{
err("Could not open file %s for writing\n",qPrint(fileName));
}
else
{
QCString stext = s->text();
file.write( stext.data(), stext.length() );
file.close();
m_t << "\n";
writeDotFile(fileName,s->relPath(),s->context(),s->srcFile(),s->srcLine());
visitPreCaption(m_t, s);
visitCaption(this, s->children());
visitPostCaption(m_t, s);
m_t << "
\n";
if (Config_getBool(DOT_CLEANUP)) Dir().remove(fileName.str());
}
forceStartParagraph(s);
}
break;
case DocVerbatim::Msc:
{
forceEndParagraph(s);
static int mscindex = 1;
QCString baseName(4096);
baseName.sprintf("%s%d",
qPrint(Config_getString(HTML_OUTPUT)+"/inline_mscgraph_"),
mscindex++
);
std::ofstream file(baseName.str()+".msc",std::ofstream::out | std::ofstream::binary);
if (!file.is_open())
{
err("Could not open file %s.msc for writing\n",qPrint(baseName));
}
else
{
QCString text = "msc {";
text+=s->text();
text+="}";
file.write( text.data(), text.length() );
file.close();
m_t << "\n";
writeMscFile(baseName+".msc",s->relPath(),s->context(),s->srcFile(),s->srcLine());
visitPreCaption(m_t, s);
visitCaption(this, s->children());
visitPostCaption(m_t, s);
m_t << "
\n";
if (Config_getBool(DOT_CLEANUP)) Dir().remove(baseName.str()+".msc");
}
forceStartParagraph(s);
}
break;
case DocVerbatim::PlantUML:
{
forceEndParagraph(s);
static QCString htmlOutput = Config_getString(HTML_OUTPUT);
QCString imgExt = getDotImageExtension();
PlantumlManager::OutputFormat format = PlantumlManager::PUML_BITMAP; // default : PUML_BITMAP
if (imgExt=="svg")
{
format = PlantumlManager::PUML_SVG;
}
QCString baseName = PlantumlManager::instance().writePlantUMLSource(
htmlOutput,s->exampleFile(),
s->text(),format,s->engine(),s->srcFile(),s->srcLine());
m_t << "\n";
writePlantUMLFile(baseName,s->relPath(),s->context(),s->srcFile(),s->srcLine());
visitPreCaption(m_t, s);
visitCaption(this, s->children());
visitPostCaption(m_t, s);
m_t << "
\n";
forceStartParagraph(s);
}
break;
}
}
void HtmlDocVisitor::visit(DocAnchor *anc)
{
if (m_hide) return;
m_t << "anchor() << "\"" << htmlAttribsToString(anc->attribs()) << "> ";
}
void HtmlDocVisitor::visit(DocInclude *inc)
{
if (m_hide) return;
SrcLangExt langExt = getLanguageFromFileName(inc->extension());
switch(inc->type())
{
case DocInclude::Include:
forceEndParagraph(inc);
m_ci.startCodeFragment("DoxyCode");
getCodeParser(inc->extension()).parseCode(m_ci,
inc->context(),
inc->text(),
langExt,
inc->isExample(),
inc->exampleFile(),
0, // fileDef
-1, // startLine
-1, // endLine
TRUE, // inlineFragment
0, // memberDef
FALSE, // show line numbers
m_ctx // search context
);
m_ci.endCodeFragment("DoxyCode");
forceStartParagraph(inc);
break;
case DocInclude::IncWithLines:
{
forceEndParagraph(inc);
m_ci.startCodeFragment("DoxyCode");
FileInfo cfi( inc->file().str() );
FileDef *fd = createFileDef( cfi.dirPath(), cfi.fileName() );
getCodeParser(inc->extension()).parseCode(m_ci,
inc->context(),
inc->text(),
langExt,
inc->isExample(),
inc->exampleFile(),
fd, // fileDef,
-1, // start line
-1, // end line
FALSE, // inline fragment
0, // memberDef
TRUE, // show line numbers
m_ctx // search context
);
delete fd;
m_ci.endCodeFragment("DoxyCode");
forceStartParagraph(inc);
}
break;
case DocInclude::DontInclude:
case DocInclude::LatexInclude:
case DocInclude::RtfInclude:
case DocInclude::ManInclude:
case DocInclude::XmlInclude:
case DocInclude::DocbookInclude:
case DocInclude::DontIncWithLines:
break;
case DocInclude::HtmlInclude:
{
if (inc->isBlock()) forceEndParagraph(inc);
m_t << inc->text();
if (inc->isBlock()) forceStartParagraph(inc);
}
break;
case DocInclude::VerbInclude:
forceEndParagraph(inc);
m_t << "";
filter(inc->text());
m_t << " ";
forceStartParagraph(inc);
break;
case DocInclude::Snippet:
{
forceEndParagraph(inc);
m_ci.startCodeFragment("DoxyCode");
getCodeParser(inc->extension()).parseCode(m_ci,
inc->context(),
extractBlock(inc->text(),inc->blockId()),
langExt,
inc->isExample(),
inc->exampleFile(),
0,
-1, // startLine
-1, // endLine
TRUE, // inlineFragment
0, // memberDef
FALSE, // show line number
m_ctx // search context
);
m_ci.endCodeFragment("DoxyCode");
forceStartParagraph(inc);
}
break;
case DocInclude::SnipWithLines:
{
forceEndParagraph(inc);
m_ci.startCodeFragment("DoxyCode");
FileInfo cfi( inc->file().str() );
FileDef *fd = createFileDef( cfi.dirPath(), cfi.fileName() );
getCodeParser(inc->extension()).parseCode(m_ci,
inc->context(),
extractBlock(inc->text(),inc->blockId()),
langExt,
inc->isExample(),
inc->exampleFile(),
fd,
lineBlock(inc->text(),inc->blockId()),
-1, // endLine
FALSE, // inlineFragment
0, // memberDef
TRUE, // show line number
m_ctx // search context
);
delete fd;
m_ci.endCodeFragment("DoxyCode");
forceStartParagraph(inc);
}
break;
case DocInclude::SnippetDoc:
case DocInclude::IncludeDoc:
err("Internal inconsistency: found switch SnippetDoc / IncludeDoc in file: %s"
"Please create a bug report\n",__FILE__);
break;
}
}
void HtmlDocVisitor::visit(DocIncOperator *op)
{
//printf("DocIncOperator: type=%d first=%d, last=%d text='%s'\n",
// op->type(),op->isFirst(),op->isLast(),qPrint(op->text()));
if (op->isFirst())
{
forceEndParagraph(op);
if (!m_hide) m_ci.startCodeFragment("DoxyCode");
pushHidden(m_hide);
m_hide=TRUE;
}
QCString locLangExt = getFileNameExtension(op->includeFileName());
if (locLangExt.isEmpty()) locLangExt = m_langExt;
SrcLangExt langExt = getLanguageFromFileName(locLangExt);
if (op->type()!=DocIncOperator::Skip)
{
m_hide = popHidden();
if (!m_hide)
{
FileDef *fd = 0;
if (!op->includeFileName().isEmpty())
{
FileInfo cfi( op->includeFileName().str() );
fd = createFileDef( cfi.dirPath(), cfi.fileName() );
}
getCodeParser(locLangExt).parseCode(
m_ci,
op->context(),
op->text(),
langExt,
op->isExample(),
op->exampleFile(),
fd, // fileDef
op->line(), // startLine
-1, // endLine
FALSE, // inline fragment
0, // memberDef
op->showLineNo(), // show line numbers
m_ctx // search context
);
if (fd) delete fd;
}
pushHidden(m_hide);
m_hide=TRUE;
}
if (op->isLast())
{
m_hide = popHidden();
if (!m_hide) m_ci.endCodeFragment("DoxyCode");
forceStartParagraph(op);
}
else
{
if (!m_hide) m_t << "\n";
}
}
void HtmlDocVisitor::visit(DocFormula *f)
{
if (m_hide) return;
bool bDisplay = !f->isInline();
if (bDisplay)
{
forceEndParagraph(f);
m_t << "\n";
}
if (Config_getBool(USE_MATHJAX))
{
QCString text = f->text();
bool closeInline = FALSE;
if (!bDisplay && !text.isEmpty() && text.at(0)=='$' &&
text.at(text.length()-1)=='$')
{
closeInline=TRUE;
text = text.mid(1,text.length()-2);
m_t << "\\(";
}
else if (!bDisplay && !text.isEmpty())
{
closeInline=TRUE;
m_t << "\\(";
}
m_t << convertToHtml(text);
if (closeInline)
{
m_t << "\\)";
}
}
else
{
m_t << " text());
m_t << "\"";
m_t << " src=\"" << f->relPath() << f->name();
if (Config_getEnum(HTML_FORMULA_FORMAT)=="svg")
{
m_t << ".svg";
}
else
{
m_t << ".png";
}
FormulaManager::DisplaySize size = FormulaManager::instance().displaySize(f->id());
if (size.width!=-1)
{
m_t << "\" width=\"" << size.width;
}
if (size.height!=-1)
{
m_t << "\" height=\"" << size.height;
}
m_t << "\"/>";
}
if (bDisplay)
{
m_t << "\n
\n";
forceStartParagraph(f);
}
}
void HtmlDocVisitor::visit(DocIndexEntry *e)
{
QCString anchor = convertIndexWordToAnchor(e->entry());
if (e->member())
{
anchor.prepend(e->member()->anchor()+"_");
}
m_t << " ";
//printf("*** DocIndexEntry: word='%s' scope='%s' member='%s'\n",
// qPrint(e->entry()),
// e->scope() ? qPrint(e->scope()->name()) : "",
// e->member() ? qPrint(e->member()->name()) : ""
// );
Doxygen::indexList->addIndexItem(e->scope(),e->member(),anchor,e->entry());
}
void HtmlDocVisitor::visit(DocSimpleSectSep *)
{
m_t << "\n";
m_t << "\n";
}
void HtmlDocVisitor::visit(DocCite *cite)
{
if (m_hide) return;
if (!cite->file().isEmpty())
{
startLink(cite->ref(),cite->file(),cite->relPath(),cite->anchor());
}
else
{
m_t << "[";
}
filter(cite->text());
if (!cite->file().isEmpty())
{
endLink();
}
else
{
m_t << "] ";
}
}
//--------------------------------------
// visitor functions for compound nodes
//--------------------------------------
void HtmlDocVisitor::visitPre(DocAutoList *l)
{
//printf("DocAutoList::visitPre\n");
if (m_hide) return;
forceEndParagraph(l);
if (l->isEnumList())
{
//
// Do list type based on depth:
// 1.
// a.
// i.
// A.
// 1. (repeat)...
//
m_t << "depth() % NUM_HTML_LIST_TYPES] << "\">";
}
else
{
m_t << "";
}
if (!l->isPreformatted()) m_t << "\n";
}
void HtmlDocVisitor::visitPost(DocAutoList *l)
{
//printf("DocAutoList::visitPost\n");
if (m_hide) return;
if (l->isEnumList())
{
m_t << " ";
}
else
{
m_t << "";
}
if (!l->isPreformatted()) m_t << "\n";
forceStartParagraph(l);
}
void HtmlDocVisitor::visitPre(DocAutoListItem *)
{
if (m_hide) return;
m_t << "";
}
void HtmlDocVisitor::visitPost(DocAutoListItem *li)
{
if (m_hide) return;
m_t << " ";
if (!li->isPreformatted()) m_t << "\n";
}
template
bool isFirstChildNode(T *parent, DocNode *node)
{
return !parent->children().empty() && parent->children().front().get()==node;
}
template
bool isLastChildNode(T *parent, DocNode *node)
{
return !parent->children().empty() && parent->children().back().get()==node;
}
bool isSeparatedParagraph(DocSimpleSect *parent,DocPara *par)
{
const DocNodeList &nodes = parent->children();
auto it = std::find_if(nodes.begin(),nodes.end(),[par](const auto &n) { return n.get()==par; });
if (it==nodes.end()) return FALSE;
size_t i = it - nodes.begin();
size_t count = parent->children().size();
if (count>1 && i==0) // first node
{
if (nodes.at(i+1)->kind()==DocNode::Kind_SimpleSectSep)
{
return TRUE;
}
}
else if (count>1 && i==count-1) // last node
{
if (nodes.at(i-1)->kind()==DocNode::Kind_SimpleSectSep)
{
return TRUE;
}
}
else if (count>2 && i>0 && ikind()==DocNode::Kind_SimpleSectSep &&
nodes.at(i+1)->kind()==DocNode::Kind_SimpleSectSep)
{
return TRUE;
}
}
return FALSE;
}
static int getParagraphContext(DocPara *p,bool &isFirst,bool &isLast)
{
int t=0;
isFirst=FALSE;
isLast=FALSE;
if (p && p->parent())
{
switch (p->parent()->kind())
{
case DocNode::Kind_ParBlock:
{ // hierarchy: node N -> para -> parblock -> para
// adapt return value to kind of N
DocNode::Kind kind = DocNode::Kind_Para;
if ( p->parent()->parent() && p->parent()->parent()->parent() )
{
kind = p->parent()->parent()->parent()->kind();
}
isFirst=isFirstChildNode((DocParBlock*)p->parent(),p);
isLast =isLastChildNode ((DocParBlock*)p->parent(),p);
t=NONE;
if (isFirst)
{
if (kind==DocNode::Kind_HtmlListItem ||
kind==DocNode::Kind_SecRefItem)
{
t=STARTLI;
}
else if (kind==DocNode::Kind_HtmlDescData ||
kind==DocNode::Kind_XRefItem ||
kind==DocNode::Kind_SimpleSect)
{
t=STARTDD;
}
else if (kind==DocNode::Kind_HtmlCell ||
kind==DocNode::Kind_ParamList)
{
t=STARTTD;
}
}
if (isLast)
{
if (kind==DocNode::Kind_HtmlListItem ||
kind==DocNode::Kind_SecRefItem)
{
t=ENDLI;
}
else if (kind==DocNode::Kind_HtmlDescData ||
kind==DocNode::Kind_XRefItem ||
kind==DocNode::Kind_SimpleSect)
{
t=ENDDD;
}
else if (kind==DocNode::Kind_HtmlCell ||
kind==DocNode::Kind_ParamList)
{
t=ENDTD;
}
}
if (!isFirst && !isLast)
{
if (kind==DocNode::Kind_HtmlListItem ||
kind==DocNode::Kind_SecRefItem)
{
t=INTERLI;
}
else if (kind==DocNode::Kind_HtmlDescData ||
kind==DocNode::Kind_XRefItem ||
kind==DocNode::Kind_SimpleSect)
{
t=INTERDD;
}
else if (kind==DocNode::Kind_HtmlCell ||
kind==DocNode::Kind_ParamList)
{
t=INTERTD;
}
}
break;
}
case DocNode::Kind_AutoListItem:
isFirst=isFirstChildNode((DocAutoListItem*)p->parent(),p);
isLast =isLastChildNode ((DocAutoListItem*)p->parent(),p);
t=STARTLI; // not used
break;
case DocNode::Kind_SimpleListItem:
isFirst=TRUE;
isLast =TRUE;
t=STARTLI; // not used
break;
case DocNode::Kind_ParamList:
isFirst=TRUE;
isLast =TRUE;
t=STARTLI; // not used
break;
case DocNode::Kind_HtmlListItem:
isFirst=isFirstChildNode((DocHtmlListItem*)p->parent(),p);
isLast =isLastChildNode ((DocHtmlListItem*)p->parent(),p);
if (isFirst) t=STARTLI;
if (isLast) t=ENDLI;
if (!isFirst && !isLast) t = INTERLI;
break;
case DocNode::Kind_SecRefItem:
isFirst=isFirstChildNode((DocSecRefItem*)p->parent(),p);
isLast =isLastChildNode ((DocSecRefItem*)p->parent(),p);
if (isFirst) t=STARTLI;
if (isLast) t=ENDLI;
if (!isFirst && !isLast) t = INTERLI;
break;
case DocNode::Kind_HtmlDescData:
isFirst=isFirstChildNode((DocHtmlDescData*)p->parent(),p);
isLast =isLastChildNode ((DocHtmlDescData*)p->parent(),p);
if (isFirst) t=STARTDD;
if (isLast) t=ENDDD;
if (!isFirst && !isLast) t = INTERDD;
break;
case DocNode::Kind_XRefItem:
isFirst=isFirstChildNode((DocXRefItem*)p->parent(),p);
isLast =isLastChildNode ((DocXRefItem*)p->parent(),p);
if (isFirst) t=STARTDD;
if (isLast) t=ENDDD;
if (!isFirst && !isLast) t = INTERDD;
break;
case DocNode::Kind_SimpleSect:
isFirst=isFirstChildNode((DocSimpleSect*)p->parent(),p);
isLast =isLastChildNode ((DocSimpleSect*)p->parent(),p);
if (isFirst) t=STARTDD;
if (isLast) t=ENDDD;
if (isSeparatedParagraph((DocSimpleSect*)p->parent(),p))
// if the paragraph is enclosed with separators it will
// be included in .. so avoid addition paragraph
// markers
{
isFirst=isLast=TRUE;
}
if (!isFirst && !isLast) t = INTERDD;
break;
case DocNode::Kind_HtmlCell:
isFirst=isFirstChildNode((DocHtmlCell*)p->parent(),p);
isLast =isLastChildNode ((DocHtmlCell*)p->parent(),p);
if (isFirst) t=STARTTD;
if (isLast) t=ENDTD;
if (!isFirst && !isLast) t = INTERTD;
break;
default:
break;
}
//printf("para=%p parent()->kind=%d isFirst=%d isLast=%d t=%d\n",
// p,p->parent()->kind(),isFirst,isLast,t);
}
return t;
}
void HtmlDocVisitor::visitPre(DocPara *p)
{
if (m_hide) return;
//printf("DocPara::visitPre: parent of kind %d ",
// p->parent() ? p->parent()->kind() : -1);
bool needsTag = FALSE;
if (p && p->parent())
{
switch (p->parent()->kind())
{
case DocNode::Kind_Section:
case DocNode::Kind_Internal:
case DocNode::Kind_HtmlListItem:
case DocNode::Kind_HtmlDescData:
case DocNode::Kind_HtmlCell:
case DocNode::Kind_SimpleListItem:
case DocNode::Kind_AutoListItem:
case DocNode::Kind_SimpleSect:
case DocNode::Kind_XRefItem:
case DocNode::Kind_Copy:
case DocNode::Kind_HtmlBlockQuote:
case DocNode::Kind_ParBlock:
needsTag = TRUE;
break;
case DocNode::Kind_Root:
needsTag = !((DocRoot*)p->parent())->singleLine();
break;
default:
needsTag = FALSE;
}
}
// if the first element of a paragraph is something that should be outside of
// the paragraph (,,,..) then that will already started the
// paragraph and we don't need to do it here
size_t nodeIndex = 0;
if (p && nodeIndexchildren().size())
{
while (nodeIndexchildren().size() && isInvisibleNode(p->children().at(nodeIndex).get()))
{
nodeIndex++;
}
if (nodeIndexchildren().size())
{
const DocNode *n = p->children().at(nodeIndex).get();
if (mustBeOutsideParagraph(n))
{
needsTag = FALSE;
}
}
}
// check if this paragraph is the first or last or intermediate child of a or .
// this allows us to mark the tag with a special class so we can
// fix the otherwise ugly spacing.
int t;
bool isFirst;
bool isLast;
t = getParagraphContext(p,isFirst,isLast);
//printf("startPara first=%d last=%d\n",isFirst,isLast);
if (isFirst && isLast) needsTag=FALSE;
//printf(" needsTag=%d\n",needsTag);
// write the paragraph tag (if needed)
if (needsTag)
m_t << "attribs()) << ">";
}
void HtmlDocVisitor::visitPost(DocPara *p)
{
//printf("DocPara::visitPost: parent of kind %d ",
// p->parent() ? p->parent()->kind() : -1);
bool needsTag = FALSE;
if (p->parent())
{
switch (p->parent()->kind())
{
case DocNode::Kind_Section:
case DocNode::Kind_Internal:
case DocNode::Kind_HtmlListItem:
case DocNode::Kind_HtmlDescData:
case DocNode::Kind_HtmlCell:
case DocNode::Kind_SimpleListItem:
case DocNode::Kind_AutoListItem:
case DocNode::Kind_SimpleSect:
case DocNode::Kind_XRefItem:
case DocNode::Kind_Copy:
case DocNode::Kind_HtmlBlockQuote:
case DocNode::Kind_ParBlock:
needsTag = TRUE;
break;
case DocNode::Kind_Root:
needsTag = !((DocRoot*)p->parent())->singleLine();
break;
default:
needsTag = FALSE;
}
}
// if the last element of a paragraph is something that should be outside of
// the paragraph (
,,) then that will already have ended the
// paragraph and we don't need to do it here
if (!p->children().empty())
{
int nodeIndex = static_cast(p->children().size()-1);
while (nodeIndex>=0 && isInvisibleNode(p->children().at(nodeIndex).get()))
{
nodeIndex--;
}
if (nodeIndex>=0)
{
const DocNode *n = p->children().at(nodeIndex).get();
if (mustBeOutsideParagraph(n))
{
needsTag = FALSE;
}
}
}
bool isFirst;
bool isLast;
getParagraphContext(p,isFirst,isLast);
//printf("endPara first=%d last=%d\n",isFirst,isLast);
if (isFirst && isLast) needsTag=FALSE;
//printf("DocPara::visitPost needsTag=%d\n",needsTag);
if (needsTag) m_t << "\n";
}
void HtmlDocVisitor::visitPre(DocRoot *)
{
}
void HtmlDocVisitor::visitPost(DocRoot *)
{
}
void HtmlDocVisitor::visitPre(DocSimpleSect *s)
{
if (m_hide) return;
forceEndParagraph(s);
m_t << "typeString() << "\">";
switch(s->type())
{
case DocSimpleSect::See:
m_t << theTranslator->trSeeAlso(); break;
case DocSimpleSect::Return:
m_t << theTranslator->trReturns(); break;
case DocSimpleSect::Author:
m_t << theTranslator->trAuthor(TRUE,TRUE); break;
case DocSimpleSect::Authors:
m_t << theTranslator->trAuthor(TRUE,FALSE); break;
case DocSimpleSect::Version:
m_t << theTranslator->trVersion(); break;
case DocSimpleSect::Since:
m_t << theTranslator->trSince(); break;
case DocSimpleSect::Date:
m_t << theTranslator->trDate(); break;
case DocSimpleSect::Note:
m_t << theTranslator->trNote(); break;
case DocSimpleSect::Warning:
m_t << theTranslator->trWarning(); break;
case DocSimpleSect::Pre:
m_t << theTranslator->trPrecondition(); break;
case DocSimpleSect::Post:
m_t << theTranslator->trPostcondition(); break;
case DocSimpleSect::Copyright:
m_t << theTranslator->trCopyright(); break;
case DocSimpleSect::Invar:
m_t << theTranslator->trInvariant(); break;
case DocSimpleSect::Remark:
m_t << theTranslator->trRemarks(); break;
case DocSimpleSect::Attention:
m_t << theTranslator->trAttention(); break;
case DocSimpleSect::User: break;
case DocSimpleSect::Rcs: break;
case DocSimpleSect::Unknown: break;
}
// special case 1: user defined title
if (s->type()!=DocSimpleSect::User && s->type()!=DocSimpleSect::Rcs)
{
m_t << " ";
}
}
void HtmlDocVisitor::visitPost(DocSimpleSect *s)
{
if (m_hide) return;
m_t << " \n";
forceStartParagraph(s);
}
void HtmlDocVisitor::visitPre(DocTitle *)
{
}
void HtmlDocVisitor::visitPost(DocTitle *)
{
if (m_hide) return;
m_t << "";
}
void HtmlDocVisitor::visitPre(DocSimpleList *sl)
{
if (m_hide) return;
forceEndParagraph(sl);
m_t << "";
if (!sl->isPreformatted()) m_t << "\n";
}
void HtmlDocVisitor::visitPost(DocSimpleList *sl)
{
if (m_hide) return;
m_t << " ";
if (!sl->isPreformatted()) m_t << "\n";
forceStartParagraph(sl);
}
void HtmlDocVisitor::visitPre(DocSimpleListItem *)
{
if (m_hide) return;
m_t << "";
}
void HtmlDocVisitor::visitPost(DocSimpleListItem *li)
{
if (m_hide) return;
m_t << " ";
if (!li->isPreformatted()) m_t << "\n";
}
void HtmlDocVisitor::visitPre(DocSection *s)
{
if (m_hide) return;
forceEndParagraph(s);
m_t << "level() << ">";
m_t << "anchor();
m_t << "\"> \n";
filter(convertCharEntitiesToUTF8(s->title()));
m_t << " level() << ">\n";
}
void HtmlDocVisitor::visitPost(DocSection *s)
{
forceStartParagraph(s);
}
void HtmlDocVisitor::visitPre(DocHtmlList *s)
{
if (m_hide) return;
forceEndParagraph(s);
if (s->type()==DocHtmlList::Ordered)
{
m_t << "attribs());
}
else
{
m_t << "attribs());
}
m_t << ">\n";
}
void HtmlDocVisitor::visitPost(DocHtmlList *s)
{
if (m_hide) return;
if (s->type()==DocHtmlList::Ordered)
{
m_t << " ";
}
else
{
m_t << "";
}
if (!s->isPreformatted()) m_t << "\n";
forceStartParagraph(s);
}
void HtmlDocVisitor::visitPre(DocHtmlListItem *i)
{
if (m_hide) return;
m_t << "attribs()) << ">";
if (!i->isPreformatted()) m_t << "\n";
}
void HtmlDocVisitor::visitPost(DocHtmlListItem *)
{
if (m_hide) return;
m_t << " \n";
}
void HtmlDocVisitor::visitPre(DocHtmlDescList *dl)
{
if (m_hide) return;
forceEndParagraph(dl);
m_t << "attribs()) << ">\n";
}
void HtmlDocVisitor::visitPost(DocHtmlDescList *dl)
{
if (m_hide) return;
m_t << " \n";
forceStartParagraph(dl);
}
void HtmlDocVisitor::visitPre(DocHtmlDescTitle *dt)
{
if (m_hide) return;
m_t << " attribs()) << ">";
}
void HtmlDocVisitor::visitPost(DocHtmlDescTitle *)
{
if (m_hide) return;
m_t << " \n";
}
void HtmlDocVisitor::visitPre(DocHtmlDescData *dd)
{
if (m_hide) return;
m_t << "attribs()) << ">";
}
void HtmlDocVisitor::visitPost(DocHtmlDescData *)
{
if (m_hide) return;
m_t << " \n";
}
void HtmlDocVisitor::visitPre(DocHtmlTable *t)
{
if (m_hide) return;
forceEndParagraph(t);
if (t->hasCaption())
{
QCString anc = t->caption()->anchor();
if (!anc.isEmpty())
{
m_t << " \n";
}
}
QCString attrs = htmlAttribsToString(t->attribs());
if (attrs.isEmpty())
{
m_t << "\n";
}
else
{
m_t << "attribs()) << ">\n";
}
}
void HtmlDocVisitor::visitPost(DocHtmlTable *t)
{
if (m_hide) return;
m_t << "
\n";
forceStartParagraph(t);
}
void HtmlDocVisitor::visitPre(DocHtmlRow *tr)
{
if (m_hide) return;
m_t << "attribs()) << ">\n";
}
void HtmlDocVisitor::visitPost(DocHtmlRow *)
{
if (m_hide) return;
m_t << " \n";
}
void HtmlDocVisitor::visitPre(DocHtmlCell *c)
{
if (m_hide) return;
if (c->isHeading())
{
m_t << "attribs()) << ">";
}
else
{
m_t << " attribs()) << ">";
}
}
void HtmlDocVisitor::visitPost(DocHtmlCell *c)
{
if (m_hide) return;
if (c->isHeading()) m_t << ""; else m_t << " ";
}
void HtmlDocVisitor::visitPre(DocHtmlCaption *c)
{
if (m_hide) return;
m_t << "attribs()) << ">";
}
void HtmlDocVisitor::visitPost(DocHtmlCaption *)
{
if (m_hide) return;
m_t << " \n";
}
void HtmlDocVisitor::visitPre(DocInternal *)
{
if (m_hide) return;
//forceEndParagraph(i);
//m_t << "" << theTranslator->trForInternalUseOnly() << "
\n";
}
void HtmlDocVisitor::visitPost(DocInternal *)
{
if (m_hide) return;
//forceStartParagraph(i);
}
void HtmlDocVisitor::visitPre(DocHRef *href)
{
if (m_hide) return;
if (href->url().left(7)=="mailto:")
{
writeObfuscatedMailAddress(href->url().mid(7));
}
else
{
QCString url = correctURL(href->url(),href->relPath());
m_t << "attribs()) << ">";
}
}
void HtmlDocVisitor::visitPost(DocHRef *)
{
if (m_hide) return;
m_t << " ";
}
void HtmlDocVisitor::visitPre(DocHtmlHeader *header)
{
if (m_hide) return;
forceEndParagraph(header);
m_t << "level() << htmlAttribsToString(header->attribs()) << ">";
}
void HtmlDocVisitor::visitPost(DocHtmlHeader *header)
{
if (m_hide) return;
m_t << " level() << ">\n";
forceStartParagraph(header);
}
void HtmlDocVisitor::visitPre(DocImage *img)
{
if (img->type()==DocImage::Html)
{
bool inlineImage = img->isInlineImage();
bool typeSVG = img->isSVG();
QCString url = img->url();
if (!inlineImage)
{
forceEndParagraph(img);
}
if (m_hide) return;
QCString baseName=img->name();
int i;
if ((i=baseName.findRev('/'))!=-1 || (i=baseName.findRev('\\'))!=-1)
{
baseName=baseName.right(baseName.length()-i-1);
}
if (!inlineImage) m_t << "\n";
QCString sizeAttribs;
if (!img->width().isEmpty())
{
sizeAttribs+=" width=\""+img->width()+"\"";
}
if (!img->height().isEmpty()) // link to local file
{
sizeAttribs+=" height=\""+img->height()+"\"";
}
// 16 cases: url.isEmpty() | typeSVG | inlineImage | img->hasCaption()
HtmlAttribList extraAttribs;
if (typeSVG)
{
HtmlAttrib opt;
opt.name = "style";
opt.value = "pointer-events: none;";
extraAttribs.push_back(opt);
}
QCString alt;
mergeHtmlAttributes(img->attribs(),extraAttribs);
QCString attrs = htmlAttribsToString(extraAttribs,&alt);
QCString src;
if (url.isEmpty())
{
src = img->relPath()+img->name();
}
else
{
src = correctURL(url,img->relPath());
}
if (typeSVG && !inlineImage)
{
m_t << "
" << alt << " \n";
}
}
else
{
m_t << "
\n";
}
}
if (img->hasCaption())
{
if (inlineImage)
{
m_t << " title=\"";
}
else
{
m_t << "
\n";
}
}
else if (inlineImage)
{
m_t << "/>";
}
}
else // other format -> skip
{
pushHidden(m_hide);
m_hide=TRUE;
}
}
void HtmlDocVisitor::visitPost(DocImage *img)
{
if (img->type()==DocImage::Html)
{
if (m_hide) return;
bool inlineImage = img->isInlineImage();
if (img->hasCaption())
{
if (inlineImage)
{
m_t << "\"/>";
}
else // end
{
m_t << "
";
}
}
if (!inlineImage) // end
{
m_t << "
\n";
forceStartParagraph(img);
}
}
else // other format
{
m_hide = popHidden();
}
}
void HtmlDocVisitor::visitPre(DocDotFile *df)
{
if (m_hide) return;
m_t << "
\n";
writeDotFile(df->file(),df->relPath(),df->context(),df->srcFile(),df->srcLine());
if (df->hasCaption())
{
m_t << "
\n";
}
}
void HtmlDocVisitor::visitPost(DocDotFile *df)
{
if (m_hide) return;
if (df->hasCaption())
{
m_t << "
\n";
}
m_t << "
\n";
}
void HtmlDocVisitor::visitPre(DocMscFile *df)
{
if (m_hide) return;
m_t << "
\n";
writeMscFile(df->file(),df->relPath(),df->context(),df->srcFile(),df->srcLine());
if (df->hasCaption())
{
m_t << "
\n";
}
}
void HtmlDocVisitor::visitPost(DocMscFile *df)
{
if (m_hide) return;
if (df->hasCaption())
{
m_t << "
\n";
}
m_t << "
\n";
}
void HtmlDocVisitor::visitPre(DocDiaFile *df)
{
if (m_hide) return;
m_t << "
\n";
writeDiaFile(df->file(),df->relPath(),df->context(),df->srcFile(),df->srcLine());
if (df->hasCaption())
{
m_t << "
\n";
}
}
void HtmlDocVisitor::visitPost(DocDiaFile *df)
{
if (m_hide) return;
if (df->hasCaption())
{
m_t << "
\n";
}
m_t << "
\n";
}
void HtmlDocVisitor::visitPre(DocLink *lnk)
{
if (m_hide) return;
startLink(lnk->ref(),lnk->file(),lnk->relPath(),lnk->anchor());
}
void HtmlDocVisitor::visitPost(DocLink *)
{
if (m_hide) return;
endLink();
}
void HtmlDocVisitor::visitPre(DocRef *ref)
{
if (m_hide) return;
if (!ref->file().isEmpty())
{
// when ref->isSubPage()==TRUE we use ref->file() for HTML and
// ref->anchor() for LaTeX/RTF
startLink(ref->ref(),ref->file(),ref->relPath(),ref->isSubPage() ? QCString() : ref->anchor());
}
if (!ref->hasLinkText()) filter(ref->targetTitle());
}
void HtmlDocVisitor::visitPost(DocRef *ref)
{
if (m_hide) return;
if (!ref->file().isEmpty()) endLink();
//m_t << " ";
}
void HtmlDocVisitor::visitPre(DocSecRefItem *ref)
{
if (m_hide) return;
QCString refName=addHtmlExtensionIfMissing(ref->file());
m_t << "
anchor() << "\">";
}
void HtmlDocVisitor::visitPost(DocSecRefItem *)
{
if (m_hide) return;
m_t << " \n";
}
void HtmlDocVisitor::visitPre(DocSecRefList *s)
{
if (m_hide) return;
forceEndParagraph(s);
m_t << "
\n";
m_t << "
\n";
}
void HtmlDocVisitor::visitPost(DocSecRefList *s)
{
if (m_hide) return;
m_t << " \n";
m_t << "
\n";
forceStartParagraph(s);
}
void HtmlDocVisitor::visitPre(DocParamSect *s)
{
if (m_hide) return;
forceEndParagraph(s);
QCString className;
QCString heading;
switch(s->type())
{
case DocParamSect::Param:
heading=theTranslator->trParameters();
className="params";
break;
case DocParamSect::RetVal:
heading=theTranslator->trReturnValues();
className="retval";
break;
case DocParamSect::Exception:
heading=theTranslator->trExceptions();
className="exception";
break;
case DocParamSect::TemplateParam:
heading=theTranslator->trTemplateParameters();
className="tparams";
break;
default:
ASSERT(0);
}
m_t << "
";
m_t << heading;
m_t << " \n";
m_t << " \n";
}
void HtmlDocVisitor::visitPost(DocParamSect *s)
{
if (m_hide) return;
m_t << "
\n";
m_t << " \n";
m_t << "\n";
forceStartParagraph(s);
}
void HtmlDocVisitor::visitPre(DocParamList *pl)
{
//printf("DocParamList::visitPre\n");
if (m_hide) return;
m_t << "
";
DocParamSect *sect = 0;
if (pl->parent()->kind()==DocNode::Kind_ParamSect)
{
sect=(DocParamSect*)pl->parent();
}
if (sect && sect->hasInOutSpecifier())
{
m_t << "";
if (pl->direction()!=DocParamSect::Unspecified)
{
m_t << "[";
if (pl->direction()==DocParamSect::In)
{
m_t << "in";
}
else if (pl->direction()==DocParamSect::Out)
{
m_t << "out";
}
else if (pl->direction()==DocParamSect::InOut)
{
m_t << "in,out";
}
m_t << "]";
}
m_t << " ";
}
if (sect && sect->hasTypeSpecifier())
{
m_t << "";
for (const auto &type : pl->paramTypes())
{
if (type->kind()==DocNode::Kind_Word)
{
visit((DocWord*)type.get());
}
else if (type->kind()==DocNode::Kind_LinkedWord)
{
visit((DocLinkedWord*)type.get());
}
else if (type->kind()==DocNode::Kind_Sep)
{
m_t << " " << ((DocSeparator *)type.get())->chars() << " ";
}
}
m_t << " ";
}
m_t << "";
bool first=TRUE;
for (const auto ¶m : pl->parameters())
{
if (!first) m_t << ","; else first=FALSE;
if (param->kind()==DocNode::Kind_Word)
{
visit((DocWord*)param.get());
}
else if (param->kind()==DocNode::Kind_LinkedWord)
{
visit((DocLinkedWord*)param.get());
}
}
m_t << " ";
}
void HtmlDocVisitor::visitPost(DocParamList *)
{
//printf("DocParamList::visitPost\n");
if (m_hide) return;
m_t << " \n";
}
void HtmlDocVisitor::visitPre(DocXRefItem *x)
{
if (m_hide) return;
if (x->title().isEmpty()) return;
forceEndParagraph(x);
bool anonymousEnum = x->file()=="@";
if (!anonymousEnum)
{
m_t << "
key() << "\">relPath() << addHtmlExtensionIfMissing(x->file())
<< "#" << x->anchor() << "\">";
}
else
{
m_t << "key() << "\">";
}
filter(x->title());
m_t << ":";
if (!anonymousEnum) m_t << " ";
m_t << "";
}
void HtmlDocVisitor::visitPost(DocXRefItem *x)
{
if (m_hide) return;
if (x->title().isEmpty()) return;
m_t << " \n";
forceStartParagraph(x);
}
void HtmlDocVisitor::visitPre(DocInternalRef *ref)
{
if (m_hide) return;
startLink(QCString(),ref->file(),ref->relPath(),ref->anchor());
}
void HtmlDocVisitor::visitPost(DocInternalRef *)
{
if (m_hide) return;
endLink();
m_t << " ";
}
void HtmlDocVisitor::visitPre(DocText *)
{
}
void HtmlDocVisitor::visitPost(DocText *)
{
}
void HtmlDocVisitor::visitPre(DocHtmlBlockQuote *b)
{
if (m_hide) return;
forceEndParagraph(b);
QCString attrs = htmlAttribsToString(b->attribs());
m_t << "
attribs()) << ">\n";
}
void HtmlDocVisitor::visitPost(DocHtmlBlockQuote *b)
{
if (m_hide) return;
m_t << " \n";
forceStartParagraph(b);
}
void HtmlDocVisitor::visitPre(DocVhdlFlow *vf)
{
if (m_hide) return;
if (VhdlDocGen::getFlowMember()) // use VHDL flow chart creator
{
forceEndParagraph(vf);
QCString fname=FlowChart::convertNameToFileName();
m_t << "
";
m_t << "flowchart: " ; // TODO: translate me
m_t << "";
m_t << VhdlDocGen::getFlowMember()->name();
m_t << " ";
if (vf->hasCaption())
{
m_t << " ";
}
}
}
void HtmlDocVisitor::visitPost(DocVhdlFlow *vf)
{
if (m_hide) return;
if (VhdlDocGen::getFlowMember()) // use VHDL flow chart creator
{
m_t << "
";
forceStartParagraph(vf);
}
}
void HtmlDocVisitor::visitPre(DocParBlock *)
{
if (m_hide) return;
}
void HtmlDocVisitor::visitPost(DocParBlock *)
{
if (m_hide) return;
}
void HtmlDocVisitor::filter(const QCString &str)
{
if (str.isEmpty()) return;
const char *p=str.data();
char c;
while (*p)
{
c=*p++;
switch(c)
{
case '<': m_t << "<"; break;
case '>': m_t << ">"; break;
case '&': m_t << "&"; break;
case '\\': if ((*p == '(') || (*p == ')'))
m_t << "\\" << *p++;
else
m_t << c;
break;
default:
{
uchar uc = static_cast
(c);
if (uc<32 && !isspace(c)) // non-printable control characters
{
m_t << "$" << hex[uc>>4] << hex[uc&0xF] << ";";
}
else
{
m_t << c;
}
}
break;
}
}
}
/// Escape basic entities to produce a valid CDATA attribute value,
/// assume that the outer quoting will be using the double quote "
void HtmlDocVisitor::filterQuotedCdataAttr(const QCString &str)
{
if (str.isEmpty()) return;
const char *p=str.data();
char c;
while (*p)
{
c=*p++;
switch(c)
{
case '&': m_t << "&"; break;
case '"': m_t << """; break;
case '<': m_t << "<"; break;
case '>': m_t << ">"; break;
case '\\': if ((*p == '(') || (*p == ')'))
m_t << "\\" << *p++;
else
m_t << c;
break;
default:
{
uchar uc = static_cast(c);
if (uc<32 && !isspace(c)) // non-printable control characters
{
m_t << "$" << hex[uc>>4] << hex[uc&0xF] << ";";
}
else
{
m_t << c;
}
}
break;
}
}
}
void HtmlDocVisitor::startLink(const QCString &ref,const QCString &file,
const QCString &relPath,const QCString &anchor,
const QCString &tooltip)
{
//printf("HtmlDocVisitor: file=%s anchor=%s\n",qPrint(file),qPrint(anchor));
if (!ref.isEmpty()) // link to entity imported via tag file
{
m_t << "";
}
void HtmlDocVisitor::endLink()
{
m_t << " ";
}
void HtmlDocVisitor::writeDotFile(const QCString &fn,const QCString &relPath,
const QCString &context,const QCString &srcFile,int srcLine)
{
QCString baseName=fn;
int i;
if ((i=baseName.findRev('/'))!=-1)
{
baseName=baseName.right(baseName.length()-i-1);
}
if ((i=baseName.find('.'))!=-1) // strip extension
{
baseName=baseName.left(i);
}
baseName.prepend("dot_");
QCString outDir = Config_getString(HTML_OUTPUT);
writeDotGraphFromFile(fn,outDir,baseName,GOF_BITMAP,srcFile,srcLine);
writeDotImageMapFromFile(m_t,fn,outDir,relPath,baseName,context,-1,srcFile,srcLine);
}
void HtmlDocVisitor::writeMscFile(const QCString &fileName,const QCString &relPath,
const QCString &context,const QCString &srcFile,int srcLine)
{
QCString baseName=fileName;
int i;
if ((i=baseName.findRev('/'))!=-1) // strip path
{
baseName=baseName.right(baseName.length()-i-1);
}
if ((i=baseName.find('.'))!=-1) // strip extension
{
baseName=baseName.left(i);
}
baseName.prepend("msc_");
QCString outDir = Config_getString(HTML_OUTPUT);
QCString imgExt = getDotImageExtension();
MscOutputFormat mscFormat = MSC_BITMAP;
if ("svg" == imgExt)
mscFormat = MSC_SVG;
writeMscGraphFromFile(fileName,outDir,baseName,mscFormat,srcFile,srcLine);
writeMscImageMapFromFile(m_t,fileName,outDir,relPath,baseName,context,mscFormat,srcFile,srcLine);
}
void HtmlDocVisitor::writeDiaFile(const QCString &fileName, const QCString &relPath,
const QCString &,const QCString &srcFile,int srcLine)
{
QCString baseName=fileName;
int i;
if ((i=baseName.findRev('/'))!=-1) // strip path
{
baseName=baseName.right(baseName.length()-i-1);
}
if ((i=baseName.find('.'))!=-1) // strip extension
{
baseName=baseName.left(i);
}
baseName.prepend("dia_");
QCString outDir = Config_getString(HTML_OUTPUT);
writeDiaGraphFromFile(fileName,outDir,baseName,DIA_BITMAP,srcFile,srcLine);
m_t << " \n";
}
void HtmlDocVisitor::writePlantUMLFile(const QCString &fileName, const QCString &relPath,
const QCString &,const QCString &srcFile,int srcLine)
{
QCString baseName=fileName;
int i;
if ((i=baseName.findRev('/'))!=-1) // strip path
{
baseName=baseName.right(baseName.length()-i-1);
}
if ((i=baseName.findRev('.'))!=-1) // strip extension
{
baseName=baseName.left(i);
}
static QCString outDir = Config_getString(HTML_OUTPUT);
QCString imgExt = getDotImageExtension();
if (imgExt=="svg")
{
PlantumlManager::instance().generatePlantUMLOutput(fileName,outDir,PlantumlManager::PUML_SVG);
//m_t << "\n";
//m_t << "This browser is not able to show SVG: try Firefox, Chrome, Safari, or Opera instead.
";
//m_t << "\n";
m_t << " \n";
}
else
{
PlantumlManager::instance().generatePlantUMLOutput(fileName,outDir,PlantumlManager::PUML_BITMAP);
m_t << " \n";
}
}
/** Returns TRUE if the child nodes in paragraph \a para until \a nodeIndex
contain a style change node that is still active and that style change is one that
must be located outside of a paragraph, i.e. it is a center, div, or pre tag.
See also bug746162.
*/
static bool insideStyleChangeThatIsOutsideParagraph(DocPara *para,int nodeIndex)
{
//printf("insideStyleChangeThatIsOutputParagraph(index=%d)\n",nodeIndex);
int styleMask=0;
bool styleOutsideParagraph=FALSE;
while (nodeIndex>=0 && !styleOutsideParagraph)
{
DocNode *n = para->children().at(nodeIndex).get();
if (n->kind()==DocNode::Kind_StyleChange)
{
DocStyleChange *sc = (DocStyleChange*)n;
if (!sc->enable()) // remember styles that has been closed already
{
styleMask|=(int)sc->style();
}
bool paraStyle = sc->style()==DocStyleChange::Center ||
sc->style()==DocStyleChange::Div ||
sc->style()==DocStyleChange::Preformatted;
//printf("Found style change %s enabled=%d\n",sc->styleString(),sc->enable());
if (sc->enable() && (styleMask&(int)sc->style())==0 && // style change that is still active
paraStyle
)
{
styleOutsideParagraph=TRUE;
}
}
nodeIndex--;
}
return styleOutsideParagraph;
}
/** Used for items found inside a paragraph, which due to XHTML restrictions
* have to be outside of the paragraph. This method will forcefully end
* the current paragraph and forceStartParagraph() will restart it.
*/
void HtmlDocVisitor::forceEndParagraph(DocNode *n)
{
//printf("forceEndParagraph(%p) %d\n",n,n->kind());
if (n->parent() && n->parent()->kind()==DocNode::Kind_Para)
{
DocPara *para = (DocPara*)n->parent();
const DocNodeList &children = para->children();
auto it = std::find_if(children.begin(),children.end(),[n](const auto &np) { return np.get()==n; });
if (it==children.end()) return;
int nodeIndex = static_cast(it - children.begin());
nodeIndex--;
if (nodeIndex<0) return; // first node in paragraph
while (nodeIndex>=0 && isInvisibleNode(children.at(nodeIndex).get()))
{
nodeIndex--;
}
if (nodeIndex<0) return; // first visible node in paragraph
n = children.at(nodeIndex).get();
if (mustBeOutsideParagraph(n)) return; // previous node already outside paragraph context
nodeIndex--;
bool styleOutsideParagraph=insideStyleChangeThatIsOutsideParagraph(para,nodeIndex);
bool isFirst;
bool isLast;
getParagraphContext(para,isFirst,isLast);
//printf("forceEnd first=%d last=%d styleOutsideParagraph=%d\n",isFirst,isLast,styleOutsideParagraph);
if (isFirst && isLast) return;
if (styleOutsideParagraph) return;
m_t << "";
}
}
/** Used for items found inside a paragraph, which due to XHTML restrictions
* have to be outside of the paragraph. This method will forcefully start
* the paragraph, that was previously ended by forceEndParagraph().
*/
void HtmlDocVisitor::forceStartParagraph(DocNode *n)
{
//printf("forceStartParagraph(%p) %d\n",n,n->kind());
if (n->parent() && n->parent()->kind()==DocNode::Kind_Para) // if we are inside a paragraph
{
DocPara *para = (DocPara*)n->parent();
const DocNodeList &children = para->children();
auto it = std::find_if(children.begin(),children.end(),[n](const auto &np) { return np.get()==n; });
if (it==children.end()) return;
int nodeIndex = static_cast(it - children.begin());
int numNodes = static_cast(para->children().size());
bool styleOutsideParagraph=insideStyleChangeThatIsOutsideParagraph(para,nodeIndex);
if (styleOutsideParagraph) return;
nodeIndex++;
if (nodeIndex==numNodes) return; // last node
while (nodeIndexchildren().at(nodeIndex).get()))
{
nodeIndex++;
}
if (nodeIndexchildren().at(nodeIndex).get();
if (mustBeOutsideParagraph(n)) return; // next element also outside paragraph
}
else
{
return; // only whitespace at the end!
}
bool needsTag = TRUE;
bool isFirst;
bool isLast;
getParagraphContext(para,isFirst,isLast);
if (isFirst && isLast) needsTag = FALSE;
//printf("forceStart first=%d last=%d needsTag=%d\n",isFirst,isLast,needsTag);
if (needsTag) m_t << "";
}
}