diff -r 000000000000 -r 42188c7ea2d9 Orb/Doxygen/src/perlmodgen.cpp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/Orb/Doxygen/src/perlmodgen.cpp Thu Jan 21 17:29:01 2010 +0000 @@ -0,0 +1,2883 @@ +/****************************************************************************** + * + * + * + * + * Copyright (C) 1997-2008 by Dimitri van Heesch. + * Authors: Dimitri van Heesch, Miguel Lobo. + * + * Permission to use, copy, modify, and distribute this software and its + * documentation under the terms of the GNU General Public License is hereby + * granted. No representations are made about the suitability of this software + * for any purpose. It is provided "as is" without express or implied warranty. + * See the GNU General Public License for more details. + * + * Documents produced by Doxygen are derivative works derived from the + * input used in their production; they are not affected by this license. + * + */ + +#include + +#include "perlmodgen.h" +#include "docparser.h" +#include "message.h" +#include "doxygen.h" +#include "pagedef.h" + +#include +#include +#include +#include +#include + +#define PERLOUTPUT_MAX_INDENTATION 40 + +class PerlModOutputStream +{ +public: + + QString m_s; + QTextStream *m_t; + + PerlModOutputStream(QTextStream *t = 0) : m_t(t) { } + + void add(char c); + void add(const char *s); + void add(QCString &s); + void add(QString &s); + void add(int n); + void add(unsigned int n); +}; + +void PerlModOutputStream::add(char c) +{ + if (m_t != 0) + (*m_t) << c; + else + m_s += c; +} + +void PerlModOutputStream::add(const char *s) +{ + if (m_t != 0) + (*m_t) << s; + else + m_s += s; +} + +void PerlModOutputStream::add(QCString &s) +{ + if (m_t != 0) + (*m_t) << s; + else + m_s += s; +} + +void PerlModOutputStream::add(QString &s) +{ + if (m_t != 0) + (*m_t) << s; + else + m_s += s; +} + +void PerlModOutputStream::add(int n) +{ + if (m_t != 0) + (*m_t) << n; + else + m_s += n; +} + +void PerlModOutputStream::add(unsigned int n) +{ + if (m_t != 0) + (*m_t) << n; + else + m_s += n; +} + +class PerlModOutput +{ +public: + + bool m_pretty; + + inline PerlModOutput(bool pretty) + : m_pretty(pretty), m_stream(0), m_indentation(false), m_blockstart(true) + { + m_spaces[0] = 0; + } + + virtual ~PerlModOutput() { } + + inline void setPerlModOutputStream(PerlModOutputStream *os) { m_stream = os; } + + inline PerlModOutput &openSave() { iopenSave(); return *this; } + inline PerlModOutput &closeSave(QString &s) { icloseSave(s); return *this; } + + inline PerlModOutput &continueBlock() + { + if (m_blockstart) + m_blockstart = false; + else + m_stream->add(','); + indent(); + return *this; + } + + inline PerlModOutput &add(char c) { m_stream->add(c); return *this; } + inline PerlModOutput &add(const char *s) { m_stream->add(s); return *this; } + inline PerlModOutput &add(QCString &s) { m_stream->add(s); return *this; } + inline PerlModOutput &add(QString &s) { m_stream->add(s); return *this; } + inline PerlModOutput &add(int n) { m_stream->add(n); return *this; } + inline PerlModOutput &add(unsigned int n) { m_stream->add(n); return *this; } + + PerlModOutput &addQuoted(const char *s) { iaddQuoted(s); return *this; } + + inline PerlModOutput &indent() + { + if (m_pretty) { + m_stream->add('\n'); + m_stream->add(m_spaces); + } + return *this; + } + + inline PerlModOutput &open(char c, const char *s = 0) { iopen(c, s); return *this; } + inline PerlModOutput &close(char c = 0) { iclose(c); return *this; } + + inline PerlModOutput &addField(const char *s) { iaddField(s); return *this; } + inline PerlModOutput &addFieldQuotedChar(const char *field, char content) + { + iaddFieldQuotedChar(field, content); return *this; + } + inline PerlModOutput &addFieldQuotedString(const char *field, const char *content) + { + iaddFieldQuotedString(field, content); return *this; + } + inline PerlModOutput &addFieldBoolean(const char *field, bool content) + { + return addFieldQuotedString(field, content ? "yes" : "no"); + } + inline PerlModOutput &openList(const char *s = 0) { open('[', s); return *this; } + inline PerlModOutput &closeList() { close(']'); return *this; } + inline PerlModOutput &openHash(const char *s = 0 ) { open('{', s); return *this; } + inline PerlModOutput &closeHash() { close('}'); return *this; } + +protected: + + void iopenSave(); + void icloseSave(QString &); + + void incIndent(); + void decIndent(); + + void iaddQuoted(const char *); + void iaddFieldQuotedChar(const char *, char); + void iaddFieldQuotedString(const char *, const char *); + void iaddField(const char *); + + void iopen(char, const char *); + void iclose(char); + +private: + + PerlModOutputStream *m_stream; + int m_indentation; + bool m_blockstart; + + QStack m_saved; + char m_spaces[PERLOUTPUT_MAX_INDENTATION * 2 + 2]; +}; + +void PerlModOutput::iopenSave() +{ + m_saved.push(m_stream); + m_stream = new PerlModOutputStream(); +} + +void PerlModOutput::icloseSave(QString &s) +{ + s = m_stream->m_s; + delete m_stream; + m_stream = m_saved.pop(); +} + +void PerlModOutput::incIndent() +{ + if (m_indentation < PERLOUTPUT_MAX_INDENTATION) + { + char *s = &m_spaces[m_indentation * 2]; + *s++ = ' '; *s++ = ' '; *s = 0; + } + m_indentation++; +} + +void PerlModOutput::decIndent() +{ + m_indentation--; + if (m_indentation < PERLOUTPUT_MAX_INDENTATION) + m_spaces[m_indentation * 2] = 0; +} + +void PerlModOutput::iaddQuoted(const char *s) +{ + char c; + while ((c = *s++) != 0) { + if ((c == '\'') || (c == '\\')) + m_stream->add('\\'); + m_stream->add(c); + } +} + +void PerlModOutput::iaddField(const char *s) +{ + continueBlock(); + m_stream->add(s); + m_stream->add(m_pretty ? " => " : "=>"); +} + +void PerlModOutput::iaddFieldQuotedChar(const char *field, char content) +{ + iaddField(field); + m_stream->add('\''); + if ((content == '\'') || (content == '\\')) + m_stream->add('\\'); + m_stream->add(content); + m_stream->add('\''); +} + +void PerlModOutput::iaddFieldQuotedString(const char *field, const char *content) +{ + if (content == 0) + return; + iaddField(field); + m_stream->add('\''); + iaddQuoted(content); + m_stream->add('\''); +} + +void PerlModOutput::iopen(char c, const char *s) +{ + if (s != 0) + iaddField(s); + else + continueBlock(); + m_stream->add(c); + incIndent(); + m_blockstart = true; +} + +void PerlModOutput::iclose(char c) +{ + decIndent(); + indent(); + if (c != 0) + m_stream->add(c); + m_blockstart = false; +} + +/*! @brief Concrete visitor implementation for PerlMod output. */ +class PerlModDocVisitor : public DocVisitor +{ +public: + PerlModDocVisitor(PerlModOutput &); + virtual ~PerlModDocVisitor() { } + + void finish(); + + //-------------------------------------- + // visitor functions for leaf nodes + //-------------------------------------- + + void visit(DocWord *); + void visit(DocLinkedWord *); + void visit(DocWhiteSpace *); + void visit(DocSymbol *); + void visit(DocURL *); + void visit(DocLineBreak *); + void visit(DocHorRuler *); + void visit(DocStyleChange *); + void visit(DocVerbatim *); + void visit(DocAnchor *); + void visit(DocInclude *); + void visit(DocIncOperator *); + void visit(DocFormula *); + void visit(DocIndexEntry *); + void visit(DocSimpleSectSep *); + + //-------------------------------------- + // visitor functions for compound nodes + //-------------------------------------- + + void visitPre(DocAutoList *); + void visitPost(DocAutoList *); + void visitPre(DocAutoListItem *); + void visitPost(DocAutoListItem *); + void visitPre(DocPara *) ; + void visitPost(DocPara *); + void visitPre(DocRoot *); + void visitPost(DocRoot *); + void visitPre(DocSimpleSect *); + void visitPost(DocSimpleSect *); + void visitPre(DocTitle *); + void visitPost(DocTitle *); + void visitPre(DocSimpleList *); + void visitPost(DocSimpleList *); + void visitPre(DocSimpleListItem *); + void visitPost(DocSimpleListItem *); + void visitPre(DocSection *); + void visitPost(DocSection *); + void visitPre(DocHtmlList *); + void visitPost(DocHtmlList *) ; + void visitPre(DocHtmlListItem *); + void visitPost(DocHtmlListItem *); + //void visitPre(DocHtmlPre *); + //void visitPost(DocHtmlPre *); + void visitPre(DocHtmlDescList *); + void visitPost(DocHtmlDescList *); + void visitPre(DocHtmlDescTitle *); + void visitPost(DocHtmlDescTitle *); + void visitPre(DocHtmlDescData *); + void visitPost(DocHtmlDescData *); + void visitPre(DocHtmlTable *); + void visitPost(DocHtmlTable *); + void visitPre(DocHtmlRow *); + void visitPost(DocHtmlRow *) ; + void visitPre(DocHtmlCell *); + void visitPost(DocHtmlCell *); + void visitPre(DocHtmlCaption *); + void visitPost(DocHtmlCaption *); + void visitPre(DocInternal *); + void visitPost(DocInternal *); + void visitPre(DocHRef *); + void visitPost(DocHRef *); + void visitPre(DocHtmlHeader *); + void visitPost(DocHtmlHeader *); + void visitPre(DocImage *); + void visitPost(DocImage *); + void visitPre(DocDotFile *); + void visitPost(DocDotFile *); + void visitPre(DocLink *); + void visitPost(DocLink *); + void visitPre(DocRef *); + void visitPost(DocRef *); + void visitPre(DocSecRefItem *); + void visitPost(DocSecRefItem *); + void visitPre(DocSecRefList *); + void visitPost(DocSecRefList *); + //void visitPre(DocLanguage *); + //void visitPost(DocLanguage *); + void visitPre(DocParamSect *); + void visitPost(DocParamSect *); + void visitPre(DocParamList *); + void visitPost(DocParamList *); + void visitPre(DocXRefItem *); + void visitPost(DocXRefItem *); + void visitPre(DocInternalRef *); + void visitPost(DocInternalRef *); + void visitPre(DocCopy *); + void visitPost(DocCopy *); + void visitPre(DocText *); + void visitPost(DocText *); + +private: + + //-------------------------------------- + // helper functions + //-------------------------------------- + + void addLink(const QString &ref, const QString &file, + const QString &anchor); + + void enterText(); + void leaveText(); + + void openItem(const char *); + void closeItem(); + void singleItem(const char *); + void openSubBlock(const char * = 0); + void closeSubBlock(); + void openOther(); + void closeOther(); + + //-------------------------------------- + // state variables + //-------------------------------------- + + PerlModOutput &m_output; + bool m_textmode; + bool m_textblockstart; + QString m_other; +}; + +PerlModDocVisitor::PerlModDocVisitor(PerlModOutput &output) + : DocVisitor(DocVisitor_Other), m_output(output), m_textmode(false) +{ + m_output.openList("doc"); +} + +void PerlModDocVisitor::finish() +{ + leaveText(); + m_output.closeList() + .add(m_other); +} + +void PerlModDocVisitor::addLink(const QString &,const QString &file,const QString &anchor) +{ + QString link = file; + if (!anchor.isEmpty()) + (link += "_1") += anchor; + m_output.addFieldQuotedString("link", link); +} + +void PerlModDocVisitor::openItem(const char *name) +{ + leaveText(); + m_output.openHash().addFieldQuotedString("type", name); +} + +void PerlModDocVisitor::closeItem() +{ + leaveText(); + m_output.closeHash(); +} + +void PerlModDocVisitor::enterText() +{ + if (m_textmode) + return; + openItem("text"); + m_output.addField("content").add('\''); + m_textmode = true; +} + +void PerlModDocVisitor::leaveText() +{ + if (!m_textmode) + return; + m_textmode = false; + m_output + .add('\'') + .closeHash(); +} + +void PerlModDocVisitor::singleItem(const char *name) +{ + openItem(name); + closeItem(); +} + +void PerlModDocVisitor::openSubBlock(const char *s) +{ + leaveText(); + m_output.openList(s); + m_textblockstart = true; +} + +void PerlModDocVisitor::closeSubBlock() +{ + leaveText(); + m_output.closeList(); +} + +void PerlModDocVisitor::openOther() +{ + // Using a secondary text stream will corrupt the perl file. Instead of + // printing doc => [ data => [] ], it will print doc => [] data => []. + /* + leaveText(); + m_output.openSave(); + */ +} + +void PerlModDocVisitor::closeOther() +{ + // Using a secondary text stream will corrupt the perl file. Instead of + // printing doc => [ data => [] ], it will print doc => [] data => []. + /* + QString other; + leaveText(); + m_output.closeSave(other); + m_other += other; + */ +} + +void PerlModDocVisitor::visit(DocWord *w) +{ + enterText(); + m_output.addQuoted(w->word()); +} + +void PerlModDocVisitor::visit(DocLinkedWord *w) +{ + openItem("url"); + addLink(w->ref(), w->file(), w->anchor()); + m_output.addFieldQuotedString("content", w->word()); + closeItem(); +} + +void PerlModDocVisitor::visit(DocWhiteSpace *) +{ + enterText(); + m_output.add(' '); +} + +void PerlModDocVisitor::visit(DocSymbol *sy) +{ + char c = 0; + const char *s = 0; + const char *accent = 0; + const char *symbol = 0; + switch(sy->symbol()) + { + case DocSymbol::At: c = '@'; break; + case DocSymbol::Less: c = '<'; break; + case DocSymbol::Greater: c = '>'; break; + case DocSymbol::Amp: c = '&'; break; + case DocSymbol::Dollar: c = '$'; break; + case DocSymbol::Hash: c = '#'; break; + case DocSymbol::Percent: c = '%'; break; + case DocSymbol::Quot: c = '"'; break; + case DocSymbol::Lsquo: s = "\\\'"; break; + case DocSymbol::Rsquo: s = "\\\'"; break; + case DocSymbol::Ldquo: c = '"'; break; + case DocSymbol::Rdquo: c = '"'; break; + case DocSymbol::Ndash: c = '-'; break; + case DocSymbol::Mdash: s = "--"; break; + case DocSymbol::Nbsp: c = ' '; break; + case DocSymbol::Uml: accent = "umlaut"; break; + case DocSymbol::Acute: accent = "acute"; break; + case DocSymbol::Grave: accent = "grave"; break; + case DocSymbol::Circ: accent = "circ"; break; + case DocSymbol::Slash: accent = "slash"; break; + case DocSymbol::Tilde: accent = "tilde"; break; + case DocSymbol::Cedil: accent = "cedilla"; break; + case DocSymbol::Ring: accent = "ring"; break; + case DocSymbol::BSlash: s = "\\\\"; break; + case DocSymbol::Copy: symbol = "copyright"; break; + case DocSymbol::Tm: symbol = "trademark"; break; + case DocSymbol::Reg: symbol = "registered"; break; + case DocSymbol::Szlig: symbol = "szlig"; break; + case DocSymbol::Apos: s = "\\\'"; break; + case DocSymbol::Aelig: symbol = "aelig"; break; + case DocSymbol::AElig: symbol = "AElig"; break; + case DocSymbol::Unknown: + err("Error: unknown symbol found\n"); + break; + } + if (c != 0) { + enterText(); + m_output.add(c); + } else if (s != 0) { + enterText(); + m_output.add(s); + } else if (symbol != 0) { + leaveText(); + openItem("symbol"); + m_output.addFieldQuotedString("symbol", symbol); + closeItem(); + } else if (accent != 0) { + leaveText(); + openItem("accent"); + m_output + .addFieldQuotedString("accent", accent) + .addFieldQuotedChar("letter", sy->letter()); + closeItem(); + } +} + +void PerlModDocVisitor::visit(DocURL *u) +{ + openItem("url"); + m_output.addFieldQuotedString("content", u->url()); + closeItem(); +} + +void PerlModDocVisitor::visit(DocLineBreak *) { singleItem("linebreak"); } +void PerlModDocVisitor::visit(DocHorRuler *) { singleItem("hruler"); } + +void PerlModDocVisitor::visit(DocStyleChange *s) +{ + const char *style = 0; + switch (s->style()) + { + case DocStyleChange::Bold: style = "bold"; break; + case DocStyleChange::Italic: style = "italic"; break; + case DocStyleChange::Code: style = "code"; break; + case DocStyleChange::Subscript: style = "subscript"; break; + case DocStyleChange::Superscript: style = "superscript"; break; + case DocStyleChange::Center: style = "center"; break; + case DocStyleChange::Small: style = "small"; break; + case DocStyleChange::Preformatted: style = "preformatted"; break; + case DocStyleChange::Div: style = "div"; break; + case DocStyleChange::Span: style = "span"; break; + + } + openItem("style"); + m_output.addFieldQuotedString("style", style) + .addFieldBoolean("enable", s->enable()); + closeItem(); +} + +void PerlModDocVisitor::visit(DocVerbatim *s) +{ + const char *type = 0; + switch(s->type()) + { + case DocVerbatim::Code: +#if 0 + m_output.add(""); + parseCode(m_ci,s->context(),s->text(),FALSE,0); + m_output.add(""); +#endif + return; + case DocVerbatim::Verbatim: type = "preformatted"; break; + case DocVerbatim::HtmlOnly: type = "htmlonly"; break; + case DocVerbatim::ManOnly: type = "manonly"; break; + case DocVerbatim::LatexOnly: type = "latexonly"; break; + case DocVerbatim::XmlOnly: type = "xmlonly"; break; + case DocVerbatim::Dot: type = "dot"; break; + case DocVerbatim::Msc: type = "msc"; break; + } + openItem(type); + m_output.addFieldQuotedString("content", s->text()); + closeItem(); +} + +void PerlModDocVisitor::visit(DocAnchor *anc) +{ + QString anchor = anc->file() + "_1" + anc->anchor(); + openItem("anchor"); + m_output.addFieldQuotedString("id", anchor); + closeItem(); +} + +void PerlModDocVisitor::visit(DocInclude *inc) +{ + const char *type = 0; + switch(inc->type()) + { + case DocInclude::IncWithLines: + #if 0 + { + m_t << "
";
+         QFileInfo cfi( inc->file() );
+         FileDef fd( cfi.dirPath(), cfi.fileName() );
+         parseCode(m_ci,inc->context(),inc->text().latin1(),inc->isExample(),inc->exampleFile(), &fd);
+         m_t << "
"; + } + break; + #endif + return; + case DocInclude::Include: +#if 0 + m_output.add(""); + parseCode(m_ci,inc->context(),inc->text(),FALSE,0); + m_output.add(""); +#endif + return; + case DocInclude::DontInclude: return; + case DocInclude::HtmlInclude: type = "htmlonly"; break; + case DocInclude::VerbInclude: type = "preformatted"; break; + } + openItem(type); + m_output.addFieldQuotedString("content", inc->text()); + closeItem(); +} + +void PerlModDocVisitor::visit(DocIncOperator *) +{ +#if 0 + //printf("DocIncOperator: type=%d first=%d, last=%d text=`%s'\n", + // op->type(),op->isFirst(),op->isLast(),op->text().data()); + if (op->isFirst()) + { + m_output.add(""); + } + if (op->type()!=DocIncOperator::Skip) + { + parseCode(m_ci,op->context(),op->text(),FALSE,0); + } + if (op->isLast()) + { + m_output.add(""); + } + else + { + m_output.add('\n'); + } +#endif +} + +void PerlModDocVisitor::visit(DocFormula *f) +{ + openItem("formula"); + QString id; + id += f->id(); + m_output.addFieldQuotedString("id", id).addFieldQuotedString("content", f->text()); + closeItem(); +} + +void PerlModDocVisitor::visit(DocIndexEntry *) +{ +#if 0 + m_output.add("" + ""); + m_output.addQuoted(ie->entry()); + m_output.add("" + "" + ""); +#endif +} + +void PerlModDocVisitor::visit(DocSimpleSectSep *) +{ +} + +//-------------------------------------- +// visitor functions for compound nodes +//-------------------------------------- + +void PerlModDocVisitor::visitPre(DocAutoList *l) +{ + openItem("list"); + m_output.addFieldQuotedString("style", l->isEnumList() ? "ordered" : "itemized"); + openSubBlock("content"); +} + +void PerlModDocVisitor::visitPost(DocAutoList *) +{ + closeSubBlock(); + closeItem(); +} + +void PerlModDocVisitor::visitPre(DocAutoListItem *) +{ + openSubBlock(); +} + +void PerlModDocVisitor::visitPost(DocAutoListItem *) +{ + closeSubBlock(); +} + +void PerlModDocVisitor::visitPre(DocPara *) +{ + if (m_textblockstart) + m_textblockstart = false; + else + singleItem("parbreak"); + /* + openItem("para"); + openSubBlock("content"); + */ +} + +void PerlModDocVisitor::visitPost(DocPara *) +{ + /* + closeSubBlock(); + closeItem(); + */ +} + +void PerlModDocVisitor::visitPre(DocRoot *) +{ +} + +void PerlModDocVisitor::visitPost(DocRoot *) +{ +} + +void PerlModDocVisitor::visitPre(DocSimpleSect *s) +{ + const char *type = 0; + switch (s->type()) + { + case DocSimpleSect::See: type = "see"; break; + case DocSimpleSect::Return: type = "return"; break; + case DocSimpleSect::Author: type = "author"; break; + case DocSimpleSect::Authors: type = "authors"; break; + case DocSimpleSect::Version: type = "version"; break; + case DocSimpleSect::Since: type = "since"; break; + case DocSimpleSect::Date: type = "date"; break; + case DocSimpleSect::Note: type = "bug"; break; + case DocSimpleSect::Warning: type = "warning"; break; + case DocSimpleSect::Pre: type = "pre"; break; + case DocSimpleSect::Post: type = "post"; break; + case DocSimpleSect::Invar: type = "invariant"; break; + case DocSimpleSect::Remark: type = "remark"; break; + case DocSimpleSect::Attention: type = "attention"; break; + case DocSimpleSect::User: type = "par"; break; + case DocSimpleSect::Rcs: type = "rcs"; break; + case DocSimpleSect::Unknown: + err("Error: unknown simple section found\n"); + break; + } + openOther(); + openSubBlock(type); +} + +void PerlModDocVisitor::visitPost(DocSimpleSect *) +{ + closeSubBlock(); + closeOther(); +} + +void PerlModDocVisitor::visitPre(DocTitle *) +{ + openItem("title"); + openSubBlock("content"); +} + +void PerlModDocVisitor::visitPost(DocTitle *) +{ + closeSubBlock(); + closeItem(); +} + +void PerlModDocVisitor::visitPre(DocSimpleList *) +{ + openItem("list"); + m_output.addFieldQuotedString("style", "itemized"); + openSubBlock("content"); +} + +void PerlModDocVisitor::visitPost(DocSimpleList *) +{ + closeSubBlock(); + closeItem(); +} + +void PerlModDocVisitor::visitPre(DocSimpleListItem *) { openSubBlock(); } +void PerlModDocVisitor::visitPost(DocSimpleListItem *) { closeSubBlock(); } + +void PerlModDocVisitor::visitPre(DocSection *s) +{ + QString sect = QString("sect%1").arg(s->level()); + openItem(sect); + openSubBlock("content"); +} + +void PerlModDocVisitor::visitPost(DocSection *) +{ + closeSubBlock(); + closeItem(); +} + +void PerlModDocVisitor::visitPre(DocHtmlList *l) +{ + openItem("list"); + m_output.addFieldQuotedString("style", (l->type() == DocHtmlList::Ordered) ? "ordered" : "itemized"); + openSubBlock("content"); +} + +void PerlModDocVisitor::visitPost(DocHtmlList *) +{ + closeSubBlock(); + closeItem(); +} + +void PerlModDocVisitor::visitPre(DocHtmlListItem *) { openSubBlock(); } +void PerlModDocVisitor::visitPost(DocHtmlListItem *) { closeSubBlock(); } + +//void PerlModDocVisitor::visitPre(DocHtmlPre *) +//{ +// openItem("preformatted"); +// openSubBlock("content"); +// //m_insidePre=TRUE; +//} + +//void PerlModDocVisitor::visitPost(DocHtmlPre *) +//{ +// //m_insidePre=FALSE; +// closeSubBlock(); +// closeItem(); +//} + +void PerlModDocVisitor::visitPre(DocHtmlDescList *) +{ +#if 0 + m_output.add("\n"); +#endif +} + +void PerlModDocVisitor::visitPost(DocHtmlDescList *) +{ +#if 0 + m_output.add("\n"); +#endif +} + +void PerlModDocVisitor::visitPre(DocHtmlDescTitle *) +{ +#if 0 + m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPost(DocHtmlDescTitle *) +{ +#if 0 + m_output.add("\n"); +#endif +} + +void PerlModDocVisitor::visitPre(DocHtmlDescData *) +{ +#if 0 + m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPost(DocHtmlDescData *) +{ +#if 0 + m_output.add("\n"); +#endif +} + +void PerlModDocVisitor::visitPre(DocHtmlTable *) +{ +#if 0 + m_output.add("numRows()); + m_output.add("\" cols=\""); m_output.add(t->numCols()); m_output.add("\">"); +#endif +} + +void PerlModDocVisitor::visitPost(DocHtmlTable *) +{ +#if 0 + m_output.add("
\n"); +#endif +} + +void PerlModDocVisitor::visitPre(DocHtmlRow *) +{ +#if 0 + m_output.add("\n"); +#endif +} + +void PerlModDocVisitor::visitPost(DocHtmlRow *) +{ +#if 0 + m_output.add("\n"); +#endif +} + +void PerlModDocVisitor::visitPre(DocHtmlCell *) +{ +#if 0 + if (c->isHeading()) m_output.add(""); else m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPost(DocHtmlCell *) +{ +#if 0 + m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPre(DocHtmlCaption *) +{ +#if 0 + m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPost(DocHtmlCaption *) +{ +#if 0 + m_output.add("\n"); +#endif +} + +void PerlModDocVisitor::visitPre(DocInternal *) +{ +#if 0 + m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPost(DocInternal *) +{ +#if 0 + m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPre(DocHRef *) +{ +#if 0 + m_output.add("url()); m_output.add("\">"); +#endif +} + +void PerlModDocVisitor::visitPost(DocHRef *) +{ +#if 0 + m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPre(DocHtmlHeader *) +{ +#if 0 + m_output.add("level()); m_output.add(">"); +#endif +} + +void PerlModDocVisitor::visitPost(DocHtmlHeader *) +{ +#if 0 + m_output.add("level()); m_output.add(">\n"); +#endif +} + +void PerlModDocVisitor::visitPre(DocImage *) +{ +#if 0 + m_output.add("type()) + { + case DocImage::Html: m_output.add("html"); break; + case DocImage::Latex: m_output.add("latex"); break; + case DocImage::Rtf: m_output.add("rtf"); break; + } + m_output.add("\""); + + QCString baseName=img->name(); + int i; + if ((i=baseName.findRev('/'))!=-1 || (i=baseName.findRev('\\'))!=-1) + { + baseName=baseName.right(baseName.length()-i-1); + } + m_output.add(" name=\""); m_output.add(baseName); m_output.add("\""); + if (!img->width().isEmpty()) + { + m_output.add(" width=\""); + m_output.addQuoted(img->width()); + m_output.add("\""); + } + else if (!img->height().isEmpty()) + { + m_output.add(" height=\""); + m_output.addQuoted(img->height()); + m_output.add("\""); + } + m_output.add(">"); +#endif +} + +void PerlModDocVisitor::visitPost(DocImage *) +{ +#if 0 + m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPre(DocDotFile *) +{ +#if 0 + m_output.add("file()); m_output.add("\">"); +#endif +} + +void PerlModDocVisitor::visitPost(DocDotFile *) +{ +#if 0 + m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPre(DocLink *lnk) +{ + openItem("link"); + addLink(lnk->ref(), lnk->file(), lnk->anchor()); +} + +void PerlModDocVisitor::visitPost(DocLink *) +{ + closeItem(); +} + +void PerlModDocVisitor::visitPre(DocRef *ref) +{ + openItem("ref"); + if (!ref->hasLinkText()) + m_output.addFieldQuotedString("text", ref->targetTitle()); + openSubBlock("content"); +} + +void PerlModDocVisitor::visitPost(DocRef *) +{ + closeSubBlock(); + closeItem(); +} + +void PerlModDocVisitor::visitPre(DocSecRefItem *) +{ +#if 0 + m_output.add("file()); m_output.add("_1"); m_output.add(ref->anchor()); m_output.add("\">"); +#endif +} + +void PerlModDocVisitor::visitPost(DocSecRefItem *) +{ +#if 0 + m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPre(DocSecRefList *) +{ +#if 0 + m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPost(DocSecRefList *) +{ +#if 0 + m_output.add(""); +#endif +} + +//void PerlModDocVisitor::visitPre(DocLanguage *l) +//{ +// openItem("language"); +// m_output.addFieldQuotedString("id", l->id()); +//} +// +//void PerlModDocVisitor::visitPost(DocLanguage *) +//{ +// closeItem(); +//} + +void PerlModDocVisitor::visitPre(DocParamSect *s) +{ + leaveText(); + const char *type = 0; + switch(s->type()) + { + case DocParamSect::Param: type = "params"; break; + case DocParamSect::RetVal: type = "retvals"; break; + case DocParamSect::Exception: type = "exceptions"; break; + case DocParamSect::TemplateParam: type = "templateparam"; break; + case DocParamSect::Unknown: + err("Error: unknown parameter section found\n"); + break; + } + openOther(); + openSubBlock(type); +} + +void PerlModDocVisitor::visitPost(DocParamSect *) +{ + closeSubBlock(); + closeOther(); +} + +void PerlModDocVisitor::visitPre(DocParamList *pl) +{ + leaveText(); + m_output.openHash() + .openList("parameters"); + //QStrListIterator li(pl->parameters()); + //const char *s; + QListIterator li(pl->parameters()); + DocNode *param; + for (li.toFirst();(param=li.current());++li) + { + QCString s; + if (param->kind()==DocNode::Kind_Word) + { + s = ((DocWord*)param)->word(); + } + else if (param->kind()==DocNode::Kind_LinkedWord) + { + s = ((DocLinkedWord*)param)->word(); + } + m_output.openHash() + .addFieldQuotedString("name", s) + .closeHash(); + } + m_output.closeList() + .openList("doc"); +} + +void PerlModDocVisitor::visitPost(DocParamList *) +{ + leaveText(); + m_output.closeList() + .closeHash(); +} + +void PerlModDocVisitor::visitPre(DocXRefItem *) +{ +#if 0 + m_output.add("file()); m_output.add("_1"); m_output.add(x->anchor()); + m_output.add("\">"); + m_output.add(""); + m_output.addQuoted(x->title()); + m_output.add(""); + m_output.add(""); +#endif + openItem("xrefitem"); + openSubBlock("content"); +} + +void PerlModDocVisitor::visitPost(DocXRefItem *) +{ + closeSubBlock(); + closeItem(); +#if 0 + m_output.add(""); + m_output.add(""); +#endif +} + +void PerlModDocVisitor::visitPre(DocInternalRef *ref) +{ + openItem("ref"); + addLink(0,ref->file(),ref->anchor()); + openSubBlock("content"); +} + +void PerlModDocVisitor::visitPost(DocInternalRef *) +{ + closeSubBlock(); + closeItem(); +} + +void PerlModDocVisitor::visitPre(DocCopy *) +{ +} + +void PerlModDocVisitor::visitPost(DocCopy *) +{ +} + +void PerlModDocVisitor::visitPre(DocText *) +{ +} + +void PerlModDocVisitor::visitPost(DocText *) +{ +} + +static void addTemplateArgumentList(ArgumentList *al,PerlModOutput &output,const char *) +{ + QCString indentStr; + if (!al) + return; + output.openList("template_parameters"); + ArgumentListIterator ali(*al); + Argument *a; + for (ali.toFirst();(a=ali.current());++ali) + { + output.openHash(); + if (!a->type.isEmpty()) + output.addFieldQuotedString("type", a->type); + if (!a->name.isEmpty()) + output.addFieldQuotedString("declaration_name", a->name) + .addFieldQuotedString("definition_name", a->name); + if (!a->defval.isEmpty()) + output.addFieldQuotedString("default", a->defval); + output.closeHash(); + } + output.closeList(); +} + +#if 0 +static void addMemberTemplateLists(MemberDef *md,PerlModOutput &output) +{ + ClassDef *cd = md->getClassDef(); + const char *cname = cd ? cd->name().data() : 0; + if (md->templateArguments()) // function template prefix + addTemplateArgumentList(md->templateArguments(),output,cname); +} +#endif + +static void addTemplateList(ClassDef *cd,PerlModOutput &output) +{ + addTemplateArgumentList(cd->templateArguments(),output,cd->name()); +} + +static void addPerlModDocBlock(PerlModOutput &output, + const char *name, + const QCString &fileName, + int lineNr, + Definition *scope, + MemberDef *md, + const QCString &text) +{ + QCString stext = text.stripWhiteSpace(); + if (stext.isEmpty()) + output.addField(name).add("{}"); + else { + DocNode *root = validatingParseDoc(fileName,lineNr,scope,md,stext,FALSE,0); + output.openHash(name); + PerlModDocVisitor *visitor = new PerlModDocVisitor(output); + root->accept(visitor); + visitor->finish(); + output.closeHash(); + delete visitor; + delete root; + } +} + +static const char *getProtectionName(Protection prot) +{ + switch (prot) + { + case Public: return "public"; + case Protected: return "protected"; + case Private: return "private"; + case Package: return "package"; + } + return 0; +} + +static const char *getVirtualnessName(Specifier virt) +{ + switch(virt) + { + case Normal: return "non_virtual"; + case Virtual: return "virtual"; + case Pure: return "pure_virtual"; + } + return 0; +} + +static QString pathDoxyfile; +static QString pathDoxyExec; + +void setPerlModDoxyfile(const QString &qs) +{ + pathDoxyfile = qs; + pathDoxyExec = QDir::currentDirPath(); +} + +class PerlModGenerator +{ +public: + + PerlModOutput m_output; + + QString pathDoxyStructurePM; + QString pathDoxyDocsTex; + QString pathDoxyFormatTex; + QString pathDoxyLatexTex; + QString pathDoxyLatexDVI; + QString pathDoxyLatexPDF; + QString pathDoxyStructureTex; + QString pathDoxyDocsPM; + QString pathDoxyLatexPL; + QString pathDoxyLatexStructurePL; + QString pathDoxyRules; + QString pathMakefile; + + inline PerlModGenerator(bool pretty) : m_output(pretty) { } + + void generatePerlModForMember(MemberDef *md, Definition *); + void generatePerlModSection(Definition *d, MemberList *ml, + const char *name, const char *header=0); + void addListOfAllMembers(ClassDef *cd); + void generatePerlModForClass(ClassDef *cd); + void generatePerlModForNamespace(NamespaceDef *nd); + void generatePerlModForFile(FileDef *fd); + void generatePerlModForGroup(GroupDef *gd); + void generatePerlModForPage(PageDef *pi); + + bool createOutputFile(QFile &f, const char *s); + bool createOutputDir(QDir &perlModDir); + bool generateDoxyLatexTex(); + bool generateDoxyFormatTex(); + bool generateDoxyStructurePM(); + bool generateDoxyLatexPL(); + bool generateDoxyLatexStructurePL(); + bool generateDoxyRules(); + bool generateMakefile(); + bool generatePerlModOutput(); + + void generate(); +}; + +void PerlModGenerator::generatePerlModForMember(MemberDef *md,Definition *) +{ + // + declaration/definition arg lists + // + reimplements + // + reimplementedBy + // + exceptions + // + const/volatile specifiers + // - examples + // - source definition + // - source references + // - source referenced by + // - body code + // - template arguments + // (templateArguments(), definitionTemplateParameterLists()) + + QCString memType; + bool isFunc=FALSE; + switch (md->memberType()) + { + case MemberDef::Define: memType="define"; break; + case MemberDef::EnumValue: memType="enumvalue"; break; + case MemberDef::Property: memType="property"; break; + case MemberDef::Variable: memType="variable"; break; + case MemberDef::Typedef: memType="typedef"; break; + case MemberDef::Enumeration: memType="enum"; break; + case MemberDef::Function: memType="function"; isFunc=TRUE; break; + case MemberDef::Signal: memType="signal"; isFunc=TRUE; break; + //case MemberDef::Prototype: memType="prototype"; isFunc=TRUE; break; + case MemberDef::Friend: memType="friend"; isFunc=TRUE; break; + case MemberDef::DCOP: memType="dcop"; isFunc=TRUE; break; + case MemberDef::Slot: memType="slot"; isFunc=TRUE; break; + case MemberDef::Event: memType="event"; break; + } + + m_output.openHash() + .addFieldQuotedString("kind", memType) + .addFieldQuotedString("name", md->name()) + .addFieldQuotedString("virtualness", getVirtualnessName(md->virtualness())) + .addFieldQuotedString("protection", getProtectionName(md->protection())) + .addFieldBoolean("static", md->isStatic()); + + addPerlModDocBlock(m_output,"brief",md->getDefFileName(),md->getDefLine(),md->getOuterScope(),md,md->briefDescription()); + addPerlModDocBlock(m_output,"detailed",md->getDefFileName(),md->getDefLine(),md->getOuterScope(),md,md->documentation()); + if (md->memberType()!=MemberDef::Define && + md->memberType()!=MemberDef::Enumeration) + m_output.addFieldQuotedString("type", md->typeString()); + + LockingPtr al = md->argumentList(); + if (isFunc) //function + { + m_output.addFieldBoolean("const", al!=0 && al->constSpecifier) + .addFieldBoolean("volatile", al!=0 && al->volatileSpecifier); + + m_output.openList("parameters"); + LockingPtr declAl = md->declArgumentList(); + LockingPtr defAl = md->argumentList(); + if (declAl!=0 && declAl->count()>0) + { + ArgumentListIterator declAli(*declAl); + ArgumentListIterator defAli(*defAl); + Argument *a; + for (declAli.toFirst();(a=declAli.current());++declAli) + { + Argument *defArg = defAli.current(); + m_output.openHash(); + + if (!a->name.isEmpty()) + m_output.addFieldQuotedString("declaration_name", a->name); + + if (defArg && !defArg->name.isEmpty() && defArg->name!=a->name) + m_output.addFieldQuotedString("definition_name", defArg->name); + + if (!a->type.isEmpty()) + m_output.addFieldQuotedString("type", a->type); + + if (!a->array.isEmpty()) + m_output.addFieldQuotedString("array", a->array); + + if (!a->defval.isEmpty()) + m_output.addFieldQuotedString("default_value", a->defval); + + if (!a->attrib.isEmpty()) + m_output.addFieldQuotedString("attributes", a->attrib); + + m_output.closeHash(); + if (defArg) ++defAli; + } + } + m_output.closeList(); + } + else if (md->memberType()==MemberDef::Define && + md->argsString()!=0) // define + { + m_output.openList("parameters"); + ArgumentListIterator ali(*al); + Argument *a; + for (ali.toFirst();(a=ali.current());++ali) + { + m_output.openHash() + .addFieldQuotedString("name", a->type) + .closeHash(); + } + m_output.closeList(); + } + if (!md->initializer().isEmpty()) + m_output.addFieldQuotedString("initializer", md->initializer()); + + if (md->excpString()) + m_output.addFieldQuotedString("exceptions", md->excpString()); + + if (md->memberType()==MemberDef::Enumeration) // enum + { + LockingPtr enumFields = md->enumFieldList(); + if (enumFields!=0) + { + m_output.openList("values"); + MemberListIterator emli(*enumFields); + MemberDef *emd; + for (emli.toFirst();(emd=emli.current());++emli) + { + m_output.openHash() + .addFieldQuotedString("name", emd->name()); + + if (!emd->initializer().isEmpty()) + m_output.addFieldQuotedString("initializer", emd->initializer()); + + addPerlModDocBlock(m_output,"brief",emd->getDefFileName(),emd->getDefLine(),emd->getOuterScope(),emd,emd->briefDescription()); + + addPerlModDocBlock(m_output,"detailed",emd->getDefFileName(),emd->getDefLine(),emd->getOuterScope(),emd,emd->documentation()); + + m_output.closeHash(); + } + m_output.closeList(); + } + } + + MemberDef *rmd = md->reimplements(); + if (rmd) + m_output.openHash("reimplements") + .addFieldQuotedString("name", rmd->name()) + .closeHash(); + + LockingPtr rbml = md->reimplementedBy(); + if (rbml!=0) + { + MemberListIterator mli(*rbml); + m_output.openList("reimplemented_by"); + for (mli.toFirst();(rmd=mli.current());++mli) + m_output.openHash() + .addFieldQuotedString("name", rmd->name()) + .closeHash(); + m_output.closeList(); + } + + m_output.closeHash(); +} + +void PerlModGenerator::generatePerlModSection(Definition *d, + MemberList *ml,const char *name,const char *header) +{ + if (ml==0) return; // empty list + + m_output.openHash(name); + + if (header) + m_output.addFieldQuotedString("header", header); + + m_output.openList("members"); + MemberListIterator mli(*ml); + MemberDef *md; + for (mli.toFirst();(md=mli.current());++mli) + { + generatePerlModForMember(md,d); + } + m_output.closeList() + .closeHash(); +} + +void PerlModGenerator::addListOfAllMembers(ClassDef *cd) +{ + m_output.openList("all_members"); + if (cd->memberNameInfoSDict()) + { + MemberNameInfoSDict::Iterator mnii(*cd->memberNameInfoSDict()); + MemberNameInfo *mni; + for (mnii.toFirst();(mni=mnii.current());++mnii) + { + MemberNameInfoIterator mii(*mni); + MemberInfo *mi; + for (mii.toFirst();(mi=mii.current());++mii) + { + MemberDef *md=mi->memberDef; + ClassDef *cd=md->getClassDef(); + Definition *d=md->getGroupDef(); + if (d==0) d = cd; + + m_output.openHash() + .addFieldQuotedString("name", md->name()) + .addFieldQuotedString("virtualness", getVirtualnessName(md->virtualness())) + .addFieldQuotedString("protection", getProtectionName(mi->prot)); + + if (!mi->ambiguityResolutionScope.isEmpty()) + m_output.addFieldQuotedString("ambiguity_scope", mi->ambiguityResolutionScope); + + m_output.addFieldQuotedString("scope", cd->name()) + .closeHash(); + } + } + } + m_output.closeList(); +} + +void PerlModGenerator::generatePerlModForClass(ClassDef *cd) +{ + // + brief description + // + detailed description + // + template argument list(s) + // - include file + // + member groups + // + inheritance diagram + // + list of direct super classes + // + list of direct sub classes + // + list of inner classes + // + collaboration diagram + // + list of all members + // + user defined member sections + // + standard member sections + // + detailed member documentation + // - examples using the class + + if (cd->isReference()) return; // skip external references. + if (cd->name().find('@')!=-1) return; // skip anonymous compounds. + if (cd->templateMaster()!=0) return; // skip generated template instances. + + m_output.openHash() + .addFieldQuotedString("name", cd->name()); + + if (cd->baseClasses()) + { + m_output.openList("base"); + BaseClassListIterator bcli(*cd->baseClasses()); + BaseClassDef *bcd; + for (bcli.toFirst();(bcd=bcli.current());++bcli) + m_output.openHash() + .addFieldQuotedString("name", bcd->classDef->displayName()) + .addFieldQuotedString("virtualness", getVirtualnessName(bcd->virt)) + .addFieldQuotedString("protection", getProtectionName(bcd->prot)) + .closeHash(); + m_output.closeList(); + } + + if (cd->subClasses()) + { + m_output.openList("derived"); + BaseClassListIterator bcli(*cd->subClasses()); + BaseClassDef *bcd; + for (bcli.toFirst();(bcd=bcli.current());++bcli) + m_output.openHash() + .addFieldQuotedString("name", bcd->classDef->displayName()) + .addFieldQuotedString("virtualness", getVirtualnessName(bcd->virt)) + .addFieldQuotedString("protection", getProtectionName(bcd->prot)) + .closeHash(); + m_output.closeList(); + } + + ClassSDict *cl = cd->getInnerClasses(); + if (cl) + { + m_output.openList("inner"); + ClassSDict::Iterator cli(*cl); + ClassDef *cd; + for (cli.toFirst();(cd=cli.current());++cli) + m_output.openHash() + .addFieldQuotedString("name", cd->name()) + .closeHash(); + m_output.closeList(); + } + + IncludeInfo *ii=cd->includeInfo(); + if (ii) + { + QCString nm = ii->includeName; + if (nm.isEmpty() && ii->fileDef) nm = ii->fileDef->docName(); + if (!nm.isEmpty()) + { + m_output.openHash("includes"); +#if 0 + if (ii->fileDef && !ii->fileDef->isReference()) // TODO: support external references + t << " id=\"" << ii->fileDef->getOutputFileBase() << "\""; +#endif + m_output.addFieldBoolean("local", ii->local) + .addFieldQuotedString("name", nm) + .closeHash(); + } + } + + addTemplateList(cd,m_output); + addListOfAllMembers(cd); + if (cd->getMemberGroupSDict()) + { + MemberGroupSDict::Iterator mgli(*cd->getMemberGroupSDict()); + MemberGroup *mg; + for (;(mg=mgli.current());++mgli) + generatePerlModSection(cd,mg->members(),"user_defined",mg->header()); + } + + generatePerlModSection(cd,cd->getMemberList(MemberList::pubTypes),"public_typedefs"); + generatePerlModSection(cd,cd->getMemberList(MemberList::pubMethods),"public_methods"); + generatePerlModSection(cd,cd->getMemberList(MemberList::pubAttribs),"public_members"); + generatePerlModSection(cd,cd->getMemberList(MemberList::pubSlots),"public_slots"); + generatePerlModSection(cd,cd->getMemberList(MemberList::signals),"signals"); + generatePerlModSection(cd,cd->getMemberList(MemberList::dcopMethods),"dcop_methods"); + generatePerlModSection(cd,cd->getMemberList(MemberList::properties),"properties"); + generatePerlModSection(cd,cd->getMemberList(MemberList::pubStaticMethods),"public_static_methods"); + generatePerlModSection(cd,cd->getMemberList(MemberList::pubStaticAttribs),"public_static_members"); + generatePerlModSection(cd,cd->getMemberList(MemberList::proTypes),"protected_typedefs"); + generatePerlModSection(cd,cd->getMemberList(MemberList::proMethods),"protected_methods"); + generatePerlModSection(cd,cd->getMemberList(MemberList::proAttribs),"protected_members"); + generatePerlModSection(cd,cd->getMemberList(MemberList::proSlots),"protected_slots"); + generatePerlModSection(cd,cd->getMemberList(MemberList::proStaticMethods),"protected_static_methods"); + generatePerlModSection(cd,cd->getMemberList(MemberList::proStaticAttribs),"protected_static_members"); + generatePerlModSection(cd,cd->getMemberList(MemberList::priTypes),"private_typedefs"); + generatePerlModSection(cd,cd->getMemberList(MemberList::priMethods),"private_methods"); + generatePerlModSection(cd,cd->getMemberList(MemberList::priAttribs),"private_members"); + generatePerlModSection(cd,cd->getMemberList(MemberList::priSlots),"private_slots"); + generatePerlModSection(cd,cd->getMemberList(MemberList::priStaticMethods),"private_static_methods"); + generatePerlModSection(cd,cd->getMemberList(MemberList::priStaticAttribs),"private_static_members"); + generatePerlModSection(cd,cd->getMemberList(MemberList::friends),"friend_methods"); + generatePerlModSection(cd,cd->getMemberList(MemberList::related),"related_methods"); + + addPerlModDocBlock(m_output,"brief",cd->getDefFileName(),cd->getDefLine(),cd,0,cd->briefDescription()); + addPerlModDocBlock(m_output,"detailed",cd->getDefFileName(),cd->getDefLine(),cd,0,cd->documentation()); + +#if 0 + DotClassGraph inheritanceGraph(cd,DotClassGraph::Inheritance); + if (!inheritanceGraph.isTrivial()) + { + t << " " << endl; + inheritanceGraph.writePerlMod(t); + t << " " << endl; + } + DotClassGraph collaborationGraph(cd,DotClassGraph::Implementation); + if (!collaborationGraph.isTrivial()) + { + t << " " << endl; + collaborationGraph.writePerlMod(t); + t << " " << endl; + } + t << " getDefFileName() << "\" line=\"" + << cd->getDefLine() << "\""; + if (cd->getStartBodyLine()!=-1) + { + t << " bodystart=\"" << cd->getStartBodyLine() << "\" bodyend=\"" + << cd->getEndBodyLine() << "\""; + } + t << "/>" << endl; +#endif + + m_output.closeHash(); +} + +void PerlModGenerator::generatePerlModForNamespace(NamespaceDef *nd) +{ + // + contained class definitions + // + contained namespace definitions + // + member groups + // + normal members + // + brief desc + // + detailed desc + // + location + // - files containing (parts of) the namespace definition + + if (nd->isReference()) return; // skip external references + + m_output.openHash() + .addFieldQuotedString("name", nd->name()); + + ClassSDict *cl = nd->getClassSDict(); + if (cl) + { + m_output.openList("classes"); + ClassSDict::Iterator cli(*cl); + ClassDef *cd; + for (cli.toFirst();(cd=cli.current());++cli) + m_output.openHash() + .addFieldQuotedString("name", cd->name()) + .closeHash(); + m_output.closeList(); + } + + NamespaceSDict *nl = nd->getNamespaceSDict(); + if (nl) + { + m_output.openList("namespaces"); + NamespaceSDict::Iterator nli(*nl); + NamespaceDef *nd; + for (nli.toFirst();(nd=nli.current());++nli) + m_output.openHash() + .addFieldQuotedString("name", nd->name()) + .closeHash(); + m_output.closeList(); + } + + if (nd->getMemberGroupSDict()) + { + MemberGroupSDict::Iterator mgli(*nd->getMemberGroupSDict()); + MemberGroup *mg; + for (;(mg=mgli.current());++mgli) + generatePerlModSection(nd,mg->members(),"user-defined",mg->header()); + } + + generatePerlModSection(nd,nd->getMemberList(MemberList::decDefineMembers),"defines"); + generatePerlModSection(nd,nd->getMemberList(MemberList::decProtoMembers),"prototypes"); + generatePerlModSection(nd,nd->getMemberList(MemberList::decTypedefMembers),"typedefs"); + generatePerlModSection(nd,nd->getMemberList(MemberList::decEnumMembers),"enums"); + generatePerlModSection(nd,nd->getMemberList(MemberList::decFuncMembers),"functions"); + generatePerlModSection(nd,nd->getMemberList(MemberList::decVarMembers),"variables"); + + addPerlModDocBlock(m_output,"brief",nd->getDefFileName(),nd->getDefLine(),0,0,nd->briefDescription()); + addPerlModDocBlock(m_output,"detailed",nd->getDefFileName(),nd->getDefLine(),0,0,nd->documentation()); + + m_output.closeHash(); +} + +void PerlModGenerator::generatePerlModForFile(FileDef *fd) +{ + // + includes files + // + includedby files + // - include graph + // - included by graph + // - contained class definitions + // - contained namespace definitions + // - member groups + // + normal members + // + brief desc + // + detailed desc + // - source code + // - location + // - number of lines + + if (fd->isReference()) return; + + m_output.openHash() + .addFieldQuotedString("name", fd->name()); + + IncludeInfo *inc; + m_output.openList("includes"); + if (fd->includeFileList()) + { + QListIterator ili1(*fd->includeFileList()); + for (ili1.toFirst();(inc=ili1.current());++ili1) + { + m_output.openHash() + .addFieldQuotedString("name", inc->includeName); + if (inc->fileDef && !inc->fileDef->isReference()) + { + m_output.addFieldQuotedString("ref", inc->fileDef->getOutputFileBase()); + } + m_output.closeHash(); + } + } + m_output.closeList(); + + m_output.openList("included_by"); + if (fd->includedByFileList()) + { + QListIterator ili2(*fd->includedByFileList()); + for (ili2.toFirst();(inc=ili2.current());++ili2) + { + m_output.openHash() + .addFieldQuotedString("name", inc->includeName); + if (inc->fileDef && !inc->fileDef->isReference()) + { + m_output.addFieldQuotedString("ref", inc->fileDef->getOutputFileBase()); + } + m_output.closeHash(); + } + } + m_output.closeList(); + + generatePerlModSection(fd,fd->getMemberList(MemberList::decDefineMembers),"defines"); + generatePerlModSection(fd,fd->getMemberList(MemberList::decProtoMembers),"prototypes"); + generatePerlModSection(fd,fd->getMemberList(MemberList::decTypedefMembers),"typedefs"); + generatePerlModSection(fd,fd->getMemberList(MemberList::decEnumMembers),"enums"); + generatePerlModSection(fd,fd->getMemberList(MemberList::decFuncMembers),"functions"); + generatePerlModSection(fd,fd->getMemberList(MemberList::decVarMembers),"variables"); + + addPerlModDocBlock(m_output,"brief",fd->getDefFileName(),fd->getDefLine(),0,0,fd->briefDescription()); + addPerlModDocBlock(m_output,"detailed",fd->getDefFileName(),fd->getDefLine(),0,0,fd->documentation()); + + m_output.closeHash(); +} + +void PerlModGenerator::generatePerlModForGroup(GroupDef *gd) +{ + // + members + // + member groups + // + files + // + classes + // + namespaces + // - packages + // + pages + // + child groups + // - examples + // + brief description + // + detailed description + + if (gd->isReference()) return; // skip external references + + m_output.openHash() + .addFieldQuotedString("name", gd->name()) + .addFieldQuotedString("title", gd->groupTitle()); + + FileList *fl = gd->getFiles(); + if (fl) + { + m_output.openList("files"); + QListIterator fli(*fl); + FileDef *fd = fl->first(); + for (fli.toFirst();(fd=fli.current());++fli) + m_output.openHash() + .addFieldQuotedString("name", fd->name()) + .closeHash(); + m_output.closeList(); + } + + ClassSDict *cl = gd->getClasses(); + if (cl) + { + m_output.openList("classes"); + ClassSDict::Iterator cli(*cl); + ClassDef *cd; + for (cli.toFirst();(cd=cli.current());++cli) + m_output.openHash() + .addFieldQuotedString("name", cd->name()) + .closeHash(); + m_output.closeList(); + } + + NamespaceSDict *nl = gd->getNamespaces(); + if (nl) + { + m_output.openList("namespaces"); + NamespaceSDict::Iterator nli(*nl); + NamespaceDef *nd; + for (nli.toFirst();(nd=nli.current());++nli) + m_output.openHash() + .addFieldQuotedString("name", nd->name()) + .closeHash(); + m_output.closeList(); + } + + PageSDict *pl = gd->getPages(); + if (pl) + { + m_output.openList("pages"); + PageSDict::Iterator pli(*pl); + PageDef *pd; + for (pli.toFirst();(pd=pli.current());++pli) + m_output.openHash() + .addFieldQuotedString("title", pd->title()) + .closeHash(); + m_output.closeList(); + } + + GroupList *gl = gd->getSubGroups(); + if (gl) + { + m_output.openList("groups"); + GroupListIterator gli(*gl); + GroupDef *sgd; + for (gli.toFirst();(sgd=gli.current());++gli) + m_output.openHash() + .addFieldQuotedString("title", sgd->groupTitle()) + .closeHash(); + m_output.closeList(); + } + + if (gd->getMemberGroupSDict()) + { + MemberGroupSDict::Iterator mgli(*gd->getMemberGroupSDict()); + MemberGroup *mg; + for (;(mg=mgli.current());++mgli) + generatePerlModSection(gd,mg->members(),"user-defined",mg->header()); + } + + generatePerlModSection(gd,gd->getMemberList(MemberList::decDefineMembers),"defines"); + generatePerlModSection(gd,gd->getMemberList(MemberList::decProtoMembers),"prototypes"); + generatePerlModSection(gd,gd->getMemberList(MemberList::decTypedefMembers),"typedefs"); + generatePerlModSection(gd,gd->getMemberList(MemberList::decEnumMembers),"enums"); + generatePerlModSection(gd,gd->getMemberList(MemberList::decFuncMembers),"functions"); + generatePerlModSection(gd,gd->getMemberList(MemberList::decVarMembers),"variables"); + + addPerlModDocBlock(m_output,"brief",gd->getDefFileName(),gd->getDefLine(),0,0,gd->briefDescription()); + addPerlModDocBlock(m_output,"detailed",gd->getDefFileName(),gd->getDefLine(),0,0,gd->documentation()); + + m_output.closeHash(); +} + +void PerlModGenerator::generatePerlModForPage(PageDef *pd) +{ + // + name + // + title + // + documentation + + if (pd->isReference()) return; + + m_output.openHash() + .addFieldQuotedString("name", pd->name()); + + SectionInfo *si = Doxygen::sectionDict.find(pd->name()); + if (si) + m_output.addFieldQuotedString("title", si->title); + + addPerlModDocBlock(m_output,"detailed",pd->docFile(),pd->docLine(),0,0,pd->documentation()); + m_output.closeHash(); +} + +bool PerlModGenerator::generatePerlModOutput() +{ + QFile outputFile; + if (!createOutputFile(outputFile, pathDoxyDocsPM)) + return false; + + QTextStream outputTextStream(&outputFile); + PerlModOutputStream outputStream(&outputTextStream); + m_output.setPerlModOutputStream(&outputStream); + m_output.add("$doxydocs=").openHash(); + + m_output.openList("classes"); + ClassSDict::Iterator cli(*Doxygen::classSDict); + ClassDef *cd; + for (cli.toFirst();(cd=cli.current());++cli) + generatePerlModForClass(cd); + m_output.closeList(); + + m_output.openList("namespaces"); + NamespaceSDict::Iterator nli(*Doxygen::namespaceSDict); + NamespaceDef *nd; + for (nli.toFirst();(nd=nli.current());++nli) + generatePerlModForNamespace(nd); + m_output.closeList(); + + m_output.openList("files"); + FileNameListIterator fnli(*Doxygen::inputNameList); + FileName *fn; + for (;(fn=fnli.current());++fnli) + { + FileNameIterator fni(*fn); + FileDef *fd; + for (;(fd=fni.current());++fni) + generatePerlModForFile(fd); + } + m_output.closeList(); + + m_output.openList("groups"); + GroupSDict::Iterator gli(*Doxygen::groupSDict); + GroupDef *gd; + for (;(gd=gli.current());++gli) + { + generatePerlModForGroup(gd); + } + m_output.closeList(); + + m_output.openList("pages"); + PageSDict::Iterator pdi(*Doxygen::pageSDict); + PageDef *pd=0; + for (pdi.toFirst();(pd=pdi.current());++pdi) + { + generatePerlModForPage(pd); + } + if (Doxygen::mainPage) + { + generatePerlModForPage(Doxygen::mainPage); + } + m_output.closeList(); + + m_output.closeHash().add(";\n1;\n"); + return true; +} + +bool PerlModGenerator::createOutputFile(QFile &f, const char *s) +{ + f.setName(s); + if (!f.open(IO_WriteOnly)) + { + err("Cannot open file %s for writing!\n", s); + return false; + } + return true; +} + +bool PerlModGenerator::createOutputDir(QDir &perlModDir) +{ + QCString outputDirectory = Config_getString("OUTPUT_DIRECTORY"); + if (outputDirectory.isEmpty()) + { + outputDirectory=QDir::currentDirPath(); + } + else + { + QDir dir(outputDirectory); + if (!dir.exists()) + { + dir.setPath(QDir::currentDirPath()); + if (!dir.mkdir(outputDirectory)) + { + err("Error: tag OUTPUT_DIRECTORY: Output directory `%s' does not " + "exist and cannot be created\n",outputDirectory.data()); + exit(1); + } + else if (!Config_getBool("QUIET")) + { + err("Notice: Output directory `%s' does not exist. " + "I have created it for you.\n", outputDirectory.data()); + } + dir.cd(outputDirectory); + } + outputDirectory=dir.absPath(); + } + + QDir dir(outputDirectory); + if (!dir.exists()) + { + dir.setPath(QDir::currentDirPath()); + if (!dir.mkdir(outputDirectory)) + { + err("Cannot create directory %s\n",outputDirectory.data()); + return false; + } + } + + perlModDir.setPath(outputDirectory+"/perlmod"); + if (!perlModDir.exists() && !perlModDir.mkdir(outputDirectory+"/perlmod")) + { + err("Could not create perlmod directory in %s\n",outputDirectory.data()); + return false; + } + return true; +} + +bool PerlModGenerator::generateDoxyStructurePM() +{ + QFile doxyModelPM; + if (!createOutputFile(doxyModelPM, pathDoxyStructurePM)) + return false; + + QTextStream doxyModelPMStream(&doxyModelPM); + doxyModelPMStream << + "sub memberlist($) {\n" + " my $prefix = $_[0];\n" + " return\n" + "\t[ \"hash\", $prefix . \"s\",\n" + "\t {\n" + "\t members =>\n" + "\t [ \"list\", $prefix . \"List\",\n" + "\t\t[ \"hash\", $prefix,\n" + "\t\t {\n" + "\t\t kind => [ \"string\", $prefix . \"Kind\" ],\n" + "\t\t name => [ \"string\", $prefix . \"Name\" ],\n" + "\t\t static => [ \"string\", $prefix . \"Static\" ],\n" + "\t\t virtualness => [ \"string\", $prefix . \"Virtualness\" ],\n" + "\t\t protection => [ \"string\", $prefix . \"Protection\" ],\n" + "\t\t type => [ \"string\", $prefix . \"Type\" ],\n" + "\t\t parameters =>\n" + "\t\t [ \"list\", $prefix . \"Params\",\n" + "\t\t\t[ \"hash\", $prefix . \"Param\",\n" + "\t\t\t {\n" + "\t\t\t declaration_name => [ \"string\", $prefix . \"ParamName\" ],\n" + "\t\t\t type => [ \"string\", $prefix . \"ParamType\" ],\n" + "\t\t\t },\n" + "\t\t\t],\n" + "\t\t ],\n" + "\t\t detailed =>\n" + "\t\t [ \"hash\", $prefix . \"Detailed\",\n" + "\t\t\t{\n" + "\t\t\t doc => [ \"doc\", $prefix . \"DetailedDoc\" ],\n" + "\t\t\t return => [ \"doc\", $prefix . \"Return\" ],\n" + "\t\t\t see => [ \"doc\", $prefix . \"See\" ],\n" + "\t\t\t params =>\n" + "\t\t\t [ \"list\", $prefix . \"PDBlocks\",\n" + "\t\t\t [ \"hash\", $prefix . \"PDBlock\",\n" + "\t\t\t\t{\n" + "\t\t\t\t parameters =>\n" + "\t\t\t\t [ \"list\", $prefix . \"PDParams\",\n" + "\t\t\t\t [ \"hash\", $prefix . \"PDParam\",\n" + "\t\t\t\t\t{\n" + "\t\t\t\t\t name => [ \"string\", $prefix . \"PDParamName\" ],\n" + "\t\t\t\t\t},\n" + "\t\t\t\t ],\n" + "\t\t\t\t ],\n" + "\t\t\t\t doc => [ \"doc\", $prefix . \"PDDoc\" ],\n" + "\t\t\t\t},\n" + "\t\t\t ],\n" + "\t\t\t ],\n" + "\t\t\t},\n" + "\t\t ],\n" + "\t\t },\n" + "\t\t],\n" + "\t ],\n" + "\t },\n" + "\t];\n" + "}\n" + "\n" + "$doxystructure =\n" + " [ \"hash\", \"Root\",\n" + " {\n" + "\tfiles =>\n" + "\t [ \"list\", \"Files\",\n" + "\t [ \"hash\", \"File\",\n" + "\t {\n" + "\t\tname => [ \"string\", \"FileName\" ],\n" + "\t\ttypedefs => memberlist(\"FileTypedef\"),\n" + "\t\tvariables => memberlist(\"FileVariable\"),\n" + "\t\tfunctions => memberlist(\"FileFunction\"),\n" + "\t\tdetailed =>\n" + "\t\t [ \"hash\", \"FileDetailed\",\n" + "\t\t {\n" + "\t\t doc => [ \"doc\", \"FileDetailedDoc\" ],\n" + "\t\t },\n" + "\t\t ],\n" + "\t },\n" + "\t ],\n" + "\t ],\n" + "\tpages =>\n" + "\t [ \"list\", \"Pages\",\n" + "\t [ \"hash\", \"Page\",\n" + "\t {\n" + "\t\tname => [ \"string\", \"PageName\" ],\n" + "\t\tdetailed =>\n" + "\t\t [ \"hash\", \"PageDetailed\",\n" + "\t\t {\n" + "\t\t doc => [ \"doc\", \"PageDetailedDoc\" ],\n" + "\t\t },\n" + "\t\t ],\n" + "\t },\n" + "\t ],\n" + "\t ],\n" + "\tclasses =>\n" + "\t [ \"list\", \"Classes\",\n" + "\t [ \"hash\", \"Class\",\n" + "\t {\n" + "\t\tname => [ \"string\", \"ClassName\" ],\n" + "\t\tpublic_typedefs => memberlist(\"ClassPublicTypedef\"),\n" + "\t\tpublic_methods => memberlist(\"ClassPublicMethod\"),\n" + "\t\tpublic_members => memberlist(\"ClassPublicMember\"),\n" + "\t\tprotected_typedefs => memberlist(\"ClassProtectedTypedef\"),\n" + "\t\tprotected_methods => memberlist(\"ClassProtectedMethod\"),\n" + "\t\tprotected_members => memberlist(\"ClassProtectedMember\"),\n" + "\t\tprivate_typedefs => memberlist(\"ClassPrivateTypedef\"),\n" + "\t\tprivate_methods => memberlist(\"ClassPrivateMethod\"),\n" + "\t\tprivate_members => memberlist(\"ClassPrivateMember\"),\n" + "\t\tdetailed =>\n" + "\t\t [ \"hash\", \"ClassDetailed\",\n" + "\t\t {\n" + "\t\t doc => [ \"doc\", \"ClassDetailedDoc\" ],\n" + "\t\t },\n" + "\t\t ],\n" + "\t },\n" + "\t ],\n" + "\t ],\n" + "\tgroups =>\n" + "\t [ \"list\", \"Groups\",\n" + "\t [ \"hash\", \"Group\",\n" + "\t {\n" + "\t\tname => [ \"string\", \"GroupName\" ],\n" + "\t\ttitle => [ \"string\", \"GroupTitle\" ],\n" + "\t\tfiles =>\n" + "\t\t [ \"list\", \"Files\",\n" + "\t\t [ \"hash\", \"File\",\n" + "\t\t {\n" + "\t\t name => [ \"string\", \"Filename\" ]\n" + "\t\t }\n" + "\t\t ],\n" + "\t\t ],\n" + "\t\tclasses =>\n" + "\t\t [ \"list\", \"Classes\",\n" + "\t\t [ \"hash\", \"Class\",\n" + "\t\t {\n" + "\t\t name => [ \"string\", \"Classname\" ]\n" + "\t\t }\n" + "\t\t ],\n" + "\t\t ],\n" + "\t\tnamespaces =>\n" + "\t\t [ \"list\", \"Namespaces\",\n" + "\t\t [ \"hash\", \"Namespace\",\n" + "\t\t {\n" + "\t\t name => [ \"string\", \"NamespaceName\" ]\n" + "\t\t }\n" + "\t\t ],\n" + "\t\t ],\n" + "\t\tpages =>\n" + "\t\t [ \"list\", \"Pages\",\n" + "\t\t [ \"hash\", \"Page\"," + "\t\t {\n" + "\t\t title => [ \"string\", \"PageName\" ]\n" + "\t\t }\n" + "\t\t ],\n" + "\t\t ],\n" + "\t\tgroups =>\n" + "\t\t [ \"list\", \"Groups\",\n" + "\t\t [ \"hash\", \"Group\",\n" + "\t\t {\n" + "\t\t title => [ \"string\", \"GroupName\" ]\n" + "\t\t }\n" + "\t\t ],\n" + "\t\t ],\n" + "\t\tfunctions => memberlist(\"GroupFunction\"),\n" + "\t\tdetailed =>\n" + "\t\t [ \"hash\", \"GroupDetailed\",\n" + "\t\t {\n" + "\t\t doc => [ \"doc\", \"GroupDetailedDoc\" ],\n" + "\t\t },\n" + "\t\t ],\n" + "\t }\n" + "\t ],\n" + "\t ],\n" + " },\n" + " ];\n" + "\n" + "1;\n"; + + return true; +} + +bool PerlModGenerator::generateDoxyRules() +{ + QFile doxyRules; + if (!createOutputFile(doxyRules, pathDoxyRules)) + return false; + + bool perlmodLatex = Config_getBool("PERLMOD_LATEX"); + QString prefix = Config_getString("PERLMOD_MAKEVAR_PREFIX"); + + QTextStream doxyRulesStream(&doxyRules); + doxyRulesStream << + prefix << "DOXY_EXEC_PATH = " << pathDoxyExec << "\n" << + prefix << "DOXYFILE = " << pathDoxyfile << "\n" << + prefix << "DOXYDOCS_PM = " << pathDoxyDocsPM << "\n" << + prefix << "DOXYSTRUCTURE_PM = " << pathDoxyStructurePM << "\n" << + prefix << "DOXYRULES = " << pathDoxyRules << "\n"; + if (perlmodLatex) + doxyRulesStream << + prefix << "DOXYLATEX_PL = " << pathDoxyLatexPL << "\n" << + prefix << "DOXYLATEXSTRUCTURE_PL = " << pathDoxyLatexStructurePL << "\n" << + prefix << "DOXYSTRUCTURE_TEX = " << pathDoxyStructureTex << "\n" << + prefix << "DOXYDOCS_TEX = " << pathDoxyDocsTex << "\n" << + prefix << "DOXYFORMAT_TEX = " << pathDoxyFormatTex << "\n" << + prefix << "DOXYLATEX_TEX = " << pathDoxyLatexTex << "\n" << + prefix << "DOXYLATEX_DVI = " << pathDoxyLatexDVI << "\n" << + prefix << "DOXYLATEX_PDF = " << pathDoxyLatexPDF << "\n"; + + doxyRulesStream << + "\n" + ".PHONY: clean-perlmod\n" + "clean-perlmod::\n" + "\trm -f $(" << prefix << "DOXYSTRUCTURE_PM) \\\n" + "\t$(" << prefix << "DOXYDOCS_PM)"; + if (perlmodLatex) + doxyRulesStream << + " \\\n" + "\t$(" << prefix << "DOXYLATEX_PL) \\\n" + "\t$(" << prefix << "DOXYLATEXSTRUCTURE_PL) \\\n" + "\t$(" << prefix << "DOXYDOCS_TEX) \\\n" + "\t$(" << prefix << "DOXYSTRUCTURE_TEX) \\\n" + "\t$(" << prefix << "DOXYFORMAT_TEX) \\\n" + "\t$(" << prefix << "DOXYLATEX_TEX) \\\n" + "\t$(" << prefix << "DOXYLATEX_PDF) \\\n" + "\t$(" << prefix << "DOXYLATEX_DVI) \\\n" + "\t$(addprefix $(" << prefix << "DOXYLATEX_TEX:tex=),out aux log)"; + doxyRulesStream << "\n\n"; + + doxyRulesStream << + "$(" << prefix << "DOXYRULES) \\\n" + "$(" << prefix << "DOXYMAKEFILE) \\\n" + "$(" << prefix << "DOXYSTRUCTURE_PM) \\\n" + "$(" << prefix << "DOXYDOCS_PM)"; + if (perlmodLatex) { + doxyRulesStream << + " \\\n" + "$(" << prefix << "DOXYLATEX_PL) \\\n" + "$(" << prefix << "DOXYLATEXSTRUCTURE_PL) \\\n" + "$(" << prefix << "DOXYFORMAT_TEX) \\\n" + "$(" << prefix << "DOXYLATEX_TEX)"; + } + doxyRulesStream << + ": \\\n" + "\t$(" << prefix << "DOXYFILE)\n" + "\tcd $(" << prefix << "DOXY_EXEC_PATH) ; doxygen \"$<\"\n"; + + if (perlmodLatex) { + doxyRulesStream << + "\n" + "$(" << prefix << "DOXYDOCS_TEX): \\\n" + "$(" << prefix << "DOXYLATEX_PL) \\\n" + "$(" << prefix << "DOXYDOCS_PM)\n" + "\tperl -I\"$(\"$@\"\n" + "\n" + "$(" << prefix << "DOXYSTRUCTURE_TEX): \\\n" + "$(" << prefix << "DOXYLATEXSTRUCTURE_PL) \\\n" + "$(" << prefix << "DOXYSTRUCTURE_PM)\n" + "\tperl -I\"$(\"$@\"\n" + "\n" + "$(" << prefix << "DOXYLATEX_PDF) \\\n" + "$(" << prefix << "DOXYLATEX_DVI): \\\n" + "$(" << prefix << "DOXYLATEX_TEX) \\\n" + "$(" << prefix << "DOXYFORMAT_TEX) \\\n" + "$(" << prefix << "DOXYSTRUCTURE_TEX) \\\n" + "$(" << prefix << "DOXYDOCS_TEX)\n" + "\n" + "$(" << prefix << "DOXYLATEX_PDF): \\\n" + "$(" << prefix << "DOXYLATEX_TEX)\n" + "\tpdflatex -interaction=nonstopmode \"$<\"\n" + "\n" + "$(" << prefix << "DOXYLATEX_DVI): \\\n" + "$(" << prefix << "DOXYLATEX_TEX)\n" + "\tlatex -interaction=nonstopmode \"$<\"\n"; + } + + return true; +} + +bool PerlModGenerator::generateMakefile() +{ + QFile makefile; + if (!createOutputFile(makefile, pathMakefile)) + return false; + + bool perlmodLatex = Config_getBool("PERLMOD_LATEX"); + QString prefix = Config_getString("PERLMOD_MAKEVAR_PREFIX"); + + QTextStream makefileStream(&makefile); + makefileStream << + ".PHONY: default clean" << (perlmodLatex ? " pdf" : "") << "\n" + "default: " << (perlmodLatex ? "pdf" : "clean") << "\n" + "\n" + "include " << pathDoxyRules << "\n" + "\n" + "clean: clean-perlmod\n"; + + if (perlmodLatex) { + makefileStream << + "pdf: $(" << prefix << "DOXYLATEX_PDF)\n" + "dvi: $(" << prefix << "DOXYLATEX_DVI)\n"; + } + + return true; +} + +bool PerlModGenerator::generateDoxyLatexStructurePL() +{ + QFile doxyLatexStructurePL; + if (!createOutputFile(doxyLatexStructurePL, pathDoxyLatexStructurePL)) + return false; + + QTextStream doxyLatexStructurePLStream(&doxyLatexStructurePL); + doxyLatexStructurePLStream << + "use DoxyStructure;\n" + "\n" + "sub process($) {\n" + "\tmy $node = $_[0];\n" + "\tmy ($type, $name) = @$node[0, 1];\n" + "\tmy $command;\n" + "\tif ($type eq \"string\") { $command = \"String\" }\n" + "\telsif ($type eq \"doc\") { $command = \"Doc\" }\n" + "\telsif ($type eq \"hash\") {\n" + "\t\t$command = \"Hash\";\n" + "\t\tfor my $subnode (values %{$$node[2]}) {\n" + "\t\t\tprocess($subnode);\n" + "\t\t}\n" + "\t}\n" + "\telsif ($type eq \"list\") {\n" + "\t\t$command = \"List\";\n" + "\t\tprocess($$node[2]);\n" + "\t}\n" + "\tprint \"\\\\\" . $command . \"Node{\" . $name . \"}%\\n\";\n" + "}\n" + "\n" + "process($doxystructure);\n"; + + return true; +} + +bool PerlModGenerator::generateDoxyLatexPL() +{ + QFile doxyLatexPL; + if (!createOutputFile(doxyLatexPL, pathDoxyLatexPL)) + return false; + + QTextStream doxyLatexPLStream(&doxyLatexPL); + doxyLatexPLStream << + "use DoxyStructure;\n" + "use DoxyDocs;\n" + "\n" + "sub latex_quote($) {\n" + "\tmy $text = $_[0];\n" + "\t$text =~ s/\\\\/\\\\textbackslash /g;\n" + "\t$text =~ s/\\|/\\\\textbar /g;\n" + "\t$text =~ s//\\\\textgreater /g;\n" + "\t$text =~ s/~/\\\\textasciitilde /g;\n" + "\t$text =~ s/\\^/\\\\textasciicircum /g;\n" + "\t$text =~ s/[\\$&%#_{}]/\\\\$&/g;\n" + "\tprint $text;\n" + "}\n" + "\n" + "sub generate_doc($) {\n" + "\tmy $doc = $_[0];\n" + "\tfor my $item (@$doc) {\n" + "\t\tmy $type = $$item{type};\n" + "\t\tif ($type eq \"text\") {\n" + "\t\t\tlatex_quote($$item{content});\n" + "\t\t} elsif ($type eq \"parbreak\") {\n" + "\t\t\tprint \"\\n\\n\";\n" + "\t\t} elsif ($type eq \"style\") {\n" + "\t\t\tmy $style = $$item{style};\n" + "\t\t\tif ($$item{enable} eq \"yes\") {\n" + "\t\t\t\tif ($style eq \"bold\") { print '\\bfseries'; }\n" + "\t\t\t\tif ($style eq \"italic\") { print '\\itshape'; }\n" + "\t\t\t\tif ($style eq \"code\") { print '\\ttfamily'; }\n" + "\t\t\t} else {\n" + "\t\t\t\tif ($style eq \"bold\") { print '\\mdseries'; }\n" + "\t\t\t\tif ($style eq \"italic\") { print '\\upshape'; }\n" + "\t\t\t\tif ($style eq \"code\") { print '\\rmfamily'; }\n" + "\t\t\t}\n" + "\t\t\tprint '{}';\n" + "\t\t} elsif ($type eq \"symbol\") {\n" + "\t\t\tmy $symbol = $$item{symbol};\n" + "\t\t\tif ($symbol eq \"copyright\") { print '\\copyright'; }\n" + "\t\t\telsif ($symbol eq \"szlig\") { print '\\ss'; }\n" + "\t\t\tprint '{}';\n" + "\t\t} elsif ($type eq \"accent\") {\n" + "\t\t\tmy ($accent) = $$item{accent};\n" + "\t\t\tif ($accent eq \"umlaut\") { print '\\\"'; }\n" + "\t\t\telsif ($accent eq \"acute\") { print '\\\\\\''; }\n" + "\t\t\telsif ($accent eq \"grave\") { print '\\`'; }\n" + "\t\t\telsif ($accent eq \"circ\") { print '\\^'; }\n" + "\t\t\telsif ($accent eq \"tilde\") { print '\\~'; }\n" + "\t\t\telsif ($accent eq \"cedilla\") { print '\\c'; }\n" + "\t\t\telsif ($accent eq \"ring\") { print '\\r'; }\n" + "\t\t\tprint \"{\" . $$item{letter} . \"}\"; \n" + "\t\t} elsif ($type eq \"list\") {\n" + "\t\t\tmy $env = ($$item{style} eq \"ordered\") ? \"enumerate\" : \"itemize\";\n" + "\t\t\tprint \"\\n\\\\begin{\" . $env .\"}\";\n" + "\t\t \tfor my $subitem (@{$$item{content}}) {\n" + "\t\t\t\tprint \"\\n\\\\item \";\n" + "\t\t\t\tgenerate_doc($subitem);\n" + "\t\t \t}\n" + "\t\t\tprint \"\\n\\\\end{\" . $env .\"}\";\n" + "\t\t} elsif ($type eq \"url\") {\n" + "\t\t\tlatex_quote($$item{content});\n" + "\t\t}\n" + "\t}\n" + "}\n" + "\n" + "sub generate($$) {\n" + "\tmy ($item, $node) = @_;\n" + "\tmy ($type, $name) = @$node[0, 1];\n" + "\tif ($type eq \"string\") {\n" + "\t\tprint \"\\\\\" . $name . \"{\";\n" + "\t\tlatex_quote($item);\n" + "\t\tprint \"}\";\n" + "\t} elsif ($type eq \"doc\") {\n" + "\t\tif (@$item) {\n" + "\t\t\tprint \"\\\\\" . $name . \"{\";\n" + "\t\t\tgenerate_doc($item);\n" + "\t\t\tprint \"}%\\n\";\n" + "\t\t} else {\n" + "#\t\t\tprint \"\\\\\" . $name . \"Empty%\\n\";\n" + "\t\t}\n" + "\t} elsif ($type eq \"hash\") {\n" + "\t\tmy ($key, $value);\n" + "\t\twhile (($key, $subnode) = each %{$$node[2]}) {\n" + "\t\t\tmy $subname = $$subnode[1];\n" + "\t\t\tprint \"\\\\Defcs{field\" . $subname . \"}{\";\n" + "\t\t\tif ($$item{$key}) {\n" + "\t\t\t\tgenerate($$item{$key}, $subnode);\n" + "\t\t\t} else {\n" + "#\t\t\t\t\tprint \"\\\\\" . $subname . \"Empty%\\n\";\n" + "\t\t\t}\n" + "\t\t\tprint \"}%\\n\";\n" + "\t\t}\n" + "\t\tprint \"\\\\\" . $name . \"%\\n\";\n" + "\t} elsif ($type eq \"list\") {\n" + "\t\tmy $index = 0;\n" + "\t\tif (@$item) {\n" + "\t\t\tprint \"\\\\\" . $name . \"{%\\n\";\n" + "\t\t\tfor my $subitem (@$item) {\n" + "\t\t\t\tif ($index) {\n" + "\t\t\t\t\tprint \"\\\\\" . $name . \"Sep%\\n\";\n" + "\t\t\t\t}\n" + "\t\t\t\tgenerate($subitem, $$node[2]);\n" + "\t\t\t\t$index++;\n" + "\t\t\t}\n" + "\t\t\tprint \"}%\\n\";\n" + "\t\t} else {\n" + "#\t\t\tprint \"\\\\\" . $name . \"Empty%\\n\";\n" + "\t\t}\n" + "\t}\n" + "}\n" + "\n" + "generate($doxydocs, $doxystructure);\n"; + + return true; +} + +bool PerlModGenerator::generateDoxyFormatTex() +{ + QFile doxyFormatTex; + if (!createOutputFile(doxyFormatTex, pathDoxyFormatTex)) + return false; + + QTextStream doxyFormatTexStream(&doxyFormatTex); + doxyFormatTexStream << + "\\def\\Defcs#1{\\long\\expandafter\\def\\csname#1\\endcsname}\n" + "\\Defcs{Empty}{}\n" + "\\def\\IfEmpty#1{\\expandafter\\ifx\\csname#1\\endcsname\\Empty}\n" + "\n" + "\\def\\StringNode#1{\\Defcs{#1}##1{##1}}\n" + "\\def\\DocNode#1{\\Defcs{#1}##1{##1}}\n" + "\\def\\ListNode#1{\\Defcs{#1}##1{##1}\\Defcs{#1Sep}{}}\n" + "\\def\\HashNode#1{\\Defcs{#1}{}}\n" + "\n" + "\\input{" << pathDoxyStructureTex << "}\n" + "\n" + "\\newbox\\BoxA\n" + "\\dimendef\\DimenA=151\\relax\n" + "\\dimendef\\DimenB=152\\relax\n" + "\\countdef\\ZoneDepth=151\\relax\n" + "\n" + "\\def\\Cs#1{\\csname#1\\endcsname}\n" + "\\def\\Letcs#1{\\expandafter\\let\\csname#1\\endcsname}\n" + "\\def\\Heading#1{\\vskip 4mm\\relax\\textbf{#1}}\n" + "\\def\\See#1{\\begin{flushleft}\\Heading{See also: }#1\\end{flushleft}}\n" + "\n" + "\\def\\Frame#1{\\vskip 3mm\\relax\\fbox{ \\vbox{\\hsize0.95\\hsize\\vskip 1mm\\relax\n" + "\\raggedright#1\\vskip 0.5mm\\relax} }}\n" + "\n" + "\\def\\Zone#1#2#3{%\n" + "\\Defcs{Test#1}{#2}%\n" + "\\Defcs{Emit#1}{#3}%\n" + "\\Defcs{#1}{%\n" + "\\advance\\ZoneDepth1\\relax\n" + "\\Letcs{Mode\\number\\ZoneDepth}0\\relax\n" + "\\Letcs{Present\\number\\ZoneDepth}0\\relax\n" + "\\Cs{Test#1}\n" + "\\expandafter\\if\\Cs{Present\\number\\ZoneDepth}1%\n" + "\\advance\\ZoneDepth-1\\relax\n" + "\\Letcs{Present\\number\\ZoneDepth}1\\relax\n" + "\\expandafter\\if\\Cs{Mode\\number\\ZoneDepth}1%\n" + "\\advance\\ZoneDepth1\\relax\n" + "\\Letcs{Mode\\number\\ZoneDepth}1\\relax\n" + "\\Cs{Emit#1}\n" + "\\advance\\ZoneDepth-1\\relax\\fi\n" + "\\advance\\ZoneDepth1\\relax\\fi\n" + "\\advance\\ZoneDepth-1\\relax}}\n" + "\n" + "\\def\\Member#1#2{%\n" + "\\Defcs{Test#1}{\\Cs{field#1Detailed}\n" + "\\IfEmpty{field#1DetailedDoc}\\else\\Letcs{Present#1}1\\fi}\n" + "\\Defcs{#1}{\\Letcs{Present#1}0\\relax\n" + "\\Cs{Test#1}\\if1\\Cs{Present#1}\\Letcs{Present\\number\\ZoneDepth}1\\relax\n" + "\\if1\\Cs{Mode\\number\\ZoneDepth}#2\\fi\\fi}}\n" + "\n" + "\\def\\TypedefMemberList#1#2{%\n" + "\\Defcs{#1DetailedDoc}##1{\\vskip 5.5mm\\relax##1}%\n" + "\\Defcs{#1Name}##1{\\textbf{##1}}%\n" + "\\Defcs{#1See}##1{\\See{##1}}%\n" + "%\n" + "\\Zone{#1s}{\\Cs{field#1List}}{\\subsubsection{#2}\\Cs{field#1List}}%\n" + "\\Member{#1}{\\Frame{typedef \\Cs{field#1Type} \\Cs{field#1Name}}%\n" + "\\Cs{field#1DetailedDoc}\\Cs{field#1See}\\vskip 5mm\\relax}}%\n" + "\n" + "\\def\\VariableMemberList#1#2{%\n" + "\\Defcs{#1DetailedDoc}##1{\\vskip 5.5mm\\relax##1}%\n" + "\\Defcs{#1Name}##1{\\textbf{##1}}%\n" + "\\Defcs{#1See}##1{\\See{##1}}%\n" + "%\n" + "\\Zone{#1s}{\\Cs{field#1List}}{\\subsubsection{#2}\\Cs{field#1List}}%\n" + "\\Member{#1}{\\Frame{\\Cs{field#1Type}{} \\Cs{field#1Name}}%\n" + "\\Cs{field#1DetailedDoc}\\Cs{field#1See}\\vskip 5mm\\relax}}%\n" + "\n" + "\\def\\FunctionMemberList#1#2{%\n" + "\\Defcs{#1PDParamName}##1{\\textit{##1}}%\n" + "\\Defcs{#1PDParam}{\\Cs{field#1PDParamName}}%\n" + "\\Defcs{#1PDParamsSep}{, }%\n" + "\\Defcs{#1PDBlocksSep}{\\vskip 2mm\\relax}%\n" + "%\n" + "\\Defcs{#1PDBlocks}##1{%\n" + "\\Heading{Parameters:}\\vskip 1.5mm\\relax\n" + "\\DimenA0pt\\relax\n" + "\\Defcs{#1PDBlock}{\\setbox\\BoxA\\hbox{\\Cs{field#1PDParams}}%\n" + "\\ifdim\\DimenA<\\wd\\BoxA\\DimenA\\wd\\BoxA\\fi}%\n" + "##1%\n" + "\\advance\\DimenA3mm\\relax\n" + "\\DimenB\\hsize\\advance\\DimenB-\\DimenA\\relax\n" + "\\Defcs{#1PDBlock}{\\hbox to\\hsize{\\vtop{\\hsize\\DimenA\\relax\n" + "\\Cs{field#1PDParams}}\\hfill\n" + "\\vtop{\\hsize\\DimenB\\relax\\Cs{field#1PDDoc}}}}%\n" + "##1}\n" + "\n" + "\\Defcs{#1ParamName}##1{\\textit{##1}}\n" + "\\Defcs{#1Param}{\\Cs{field#1ParamType}{} \\Cs{field#1ParamName}}\n" + "\\Defcs{#1ParamsSep}{, }\n" + "\n" + "\\Defcs{#1Name}##1{\\textbf{##1}}\n" + "\\Defcs{#1See}##1{\\See{##1}}\n" + "\\Defcs{#1Return}##1{\\Heading{Returns: }##1}\n" + "\\Defcs{field#1Title}{\\Frame{\\Cs{field#1Type}{} \\Cs{field#1Name}(\\Cs{field#1Params})}}%\n" + "%\n" + "\\Zone{#1s}{\\Cs{field#1List}}{\\subsubsection{#2}\\Cs{field#1List}}%\n" + "\\Member{#1}{%\n" + "\\Cs{field#1Title}\\vskip 6mm\\relax\\Cs{field#1DetailedDoc}\n" + "\\Cs{field#1Return}\\Cs{field#1PDBlocks}\\Cs{field#1See}\\vskip 5mm\\relax}}\n" + "\n" + "\\def\\FileDetailed{\\fieldFileDetailedDoc\\par}\n" + "\\def\\ClassDetailed{\\fieldClassDetailedDoc\\par}\n" + "\n" + "\\def\\FileSubzones{\\fieldFileTypedefs\\fieldFileVariables\\fieldFileFunctions}\n" + "\n" + "\\def\\ClassSubzones{%\n" + "\\fieldClassPublicTypedefs\\fieldClassPublicMembers\\fieldClassPublicMethods\n" + "\\fieldClassProtectedTypedefs\\fieldClassProtectedMembers\\fieldClassProtectedMethods\n" + "\\fieldClassPrivateTypedefs\\fieldClassPrivateMembers\\fieldClassPrivateMethods}\n" + "\n" + "\\Member{Page}{\\subsection{\\fieldPageName}\\fieldPageDetailedDoc}\n" + "\n" + "\\TypedefMemberList{FileTypedef}{Typedefs}\n" + "\\VariableMemberList{FileVariable}{Variables}\n" + "\\FunctionMemberList{FileFunction}{Functions}\n" + "\\Zone{File}{\\FileSubzones}{\\subsection{\\fieldFileName}\\fieldFileDetailed\\FileSubzones}\n" + "\n" + "\\TypedefMemberList{ClassPublicTypedef}{Public Typedefs}\n" + "\\TypedefMemberList{ClassProtectedTypedef}{Protected Typedefs}\n" + "\\TypedefMemberList{ClassPrivateTypedef}{Private Typedefs}\n" + "\\VariableMemberList{ClassPublicMember}{Public Members}\n" + "\\VariableMemberList{ClassProtectedMember}{Protected Members}\n" + "\\VariableMemberList{ClassPrivateMember}{Private Members}\n" + "\\FunctionMemberList{ClassPublicMethod}{Public Methods}\n" + "\\FunctionMemberList{ClassProtectedMethod}{Protected Methods}\n" + "\\FunctionMemberList{ClassPrivateMethod}{Private Methods}\n" + "\\Zone{Class}{\\ClassSubzones}{\\subsection{\\fieldClassName}\\fieldClassDetailed\\ClassSubzones}\n" + "\n" + "\\Zone{AllPages}{\\fieldPages}{\\section{Pages}\\fieldPages}\n" + "\\Zone{AllFiles}{\\fieldFiles}{\\section{Files}\\fieldFiles}\n" + "\\Zone{AllClasses}{\\fieldClasses}{\\section{Classes}\\fieldClasses}\n" + "\n" + "\\newlength{\\oldparskip}\n" + "\\newlength{\\oldparindent}\n" + "\\newlength{\\oldfboxrule}\n" + "\n" + "\\ZoneDepth0\\relax\n" + "\\Letcs{Mode0}1\\relax\n" + "\n" + "\\def\\EmitDoxyDocs{%\n" + "\\setlength{\\oldparskip}{\\parskip}\n" + "\\setlength{\\oldparindent}{\\parindent}\n" + "\\setlength{\\oldfboxrule}{\\fboxrule}\n" + "\\setlength{\\parskip}{0cm}\n" + "\\setlength{\\parindent}{0cm}\n" + "\\setlength{\\fboxrule}{1pt}\n" + "\\AllPages\\AllFiles\\AllClasses\n" + "\\setlength{\\parskip}{\\oldparskip}\n" + "\\setlength{\\parindent}{\\oldparindent}\n" + "\\setlength{\\fboxrule}{\\oldfboxrule}}\n"; + + return true; +} + +bool PerlModGenerator::generateDoxyLatexTex() +{ + QFile doxyLatexTex; + if (!createOutputFile(doxyLatexTex, pathDoxyLatexTex)) + return false; + + QTextStream doxyLatexTexStream(&doxyLatexTex); + doxyLatexTexStream << + "\\documentclass[a4paper,12pt]{article}\n" + "\\usepackage[latin1]{inputenc}\n" + "\\usepackage[none]{hyphenat}\n" + "\\usepackage[T1]{fontenc}\n" + "\\usepackage{hyperref}\n" + "\\usepackage{times}\n" + "\n" + "\\input{doxyformat}\n" + "\n" + "\\begin{document}\n" + "\\input{" << pathDoxyDocsTex << "}\n" + "\\sloppy\n" + "\\EmitDoxyDocs\n" + "\\end{document}\n"; + + return true; +} + +void PerlModGenerator::generate() +{ + // + classes + // + namespaces + // + files + // - packages + // + groups + // + related pages + // - examples + + QDir perlModDir; + if (!createOutputDir(perlModDir)) + return; + + bool perlmodLatex = Config_getBool("PERLMOD_LATEX"); + + pathDoxyDocsPM = perlModDir.absPath() + "/DoxyDocs.pm"; + pathDoxyStructurePM = perlModDir.absPath() + "/DoxyStructure.pm"; + pathMakefile = perlModDir.absPath() + "/Makefile"; + pathDoxyRules = perlModDir.absPath() + "/doxyrules.make"; + + if (perlmodLatex) { + pathDoxyStructureTex = perlModDir.absPath() + "/doxystructure.tex"; + pathDoxyFormatTex = perlModDir.absPath() + "/doxyformat.tex"; + pathDoxyLatexTex = perlModDir.absPath() + "/doxylatex.tex"; + pathDoxyLatexDVI = perlModDir.absPath() + "/doxylatex.dvi"; + pathDoxyLatexPDF = perlModDir.absPath() + "/doxylatex.pdf"; + pathDoxyDocsTex = perlModDir.absPath() + "/doxydocs.tex"; + pathDoxyLatexPL = perlModDir.absPath() + "/doxylatex.pl"; + pathDoxyLatexStructurePL = perlModDir.absPath() + "/doxylatex-structure.pl"; + } + + if (!(generatePerlModOutput() + && generateDoxyStructurePM() + && generateMakefile() + && generateDoxyRules())) + return; + + if (perlmodLatex) { + if (!(generateDoxyLatexStructurePL() + && generateDoxyLatexPL() + && generateDoxyLatexTex() + && generateDoxyFormatTex())) + return; + } +} + +void generatePerlMod() +{ + PerlModGenerator pmg(Config_getBool("PERLMOD_PRETTY")); + pmg.generate(); +} + +// Local Variables: +// c-basic-offset: 2 +// End: + +/* This elisp function for XEmacs makes Control-Z transform + the text in the region into a valid C string. + + (global-set-key '(control z) (lambda () (interactive) + (save-excursion + (if (< (mark) (point)) (exchange-point-and-mark)) + (let ((start (point)) (replacers + '(("\\\\" "\\\\\\\\") + ("\"" "\\\\\"") + ("\t" "\\\\t") + ("^.*$" "\"\\&\\\\n\"")))) + (while replacers + (while (re-search-forward (caar replacers) (mark) t) + (replace-match (cadar replacers) t)) + (goto-char start) + (setq replacers (cdr replacers))))))) +*/