/****************************************************************************** * * * * Copyright (C) 1997-2002 by Dimitri van Heesch. * * 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. * */ /* http://www.cubic.org/source/archive/fileform/txt/man/ has some nice introductions to groff and man pages. */ #include #include "qtbc.h" #include #include "message.h" #include "mangen.h" #include "config.h" #include "util.h" #include "doxygen.h" #include static QCString getExtension() { QCString ext = Config_getString("MAN_EXTENSION"); if( ext.length() >= 2 && ext.data()[0] == '.' && isdigit( ext.data()[1] ) ) { ext = ext.mid(1, 1); } else { ext = "3"; } return ext; } ManGenerator::ManGenerator() : OutputGenerator() { dir=Config_getString("MAN_OUTPUT")+"/man" + getExtension(); firstCol=TRUE; paragraph=FALSE; col=0; upperCase=FALSE; insideTabbing=FALSE; inHeader=FALSE; } ManGenerator::~ManGenerator() { } void ManGenerator::append(const OutputGenerator *g) { QCString r=g->getContents(); if (upperCase) t << r.upper(); else t << r; if (!r.isEmpty()) firstCol = r.at(r.length()-1)=='\n'; else firstCol = ((ManGenerator *)g)->firstCol; col+=((ManGenerator *)g)->col; inHeader=((ManGenerator *)g)->inHeader; paragraph=FALSE; } void ManGenerator::init() { QCString ext = getExtension(); QCString &manOutput = Config_getString("MAN_OUTPUT"); QDir d(manOutput); if (!d.exists() && !d.mkdir(manOutput)) { err("Could not create output directory %s\n",manOutput.data()); exit(1); } d.setPath(manOutput+"/man"+ext); if (!d.exists() && !d.mkdir(manOutput+"/man"+ext)) { err("Could not create output directory %s/man%s\n",manOutput.data(),ext.data()); exit(1); } } static QCString buildFileName(const char *name) { QCString fileName; const char *p=name; char c; while ((c=*p++)) { switch (c) { case ':': fileName+="_"; if (*p==':') p++; break; case '<': case '>': case '&': case '*': case '!': case '^': case '~': case '%': case '+': fileName+="_"; break; default: fileName+=c; } } QCString &manExtension = Config_getString("MAN_EXTENSION"); if (convertToQCString(fileName.right(2))!=manExtension) { fileName+=manExtension; } return fileName; } void ManGenerator::startFile(const char *,const char *manName, const char *,bool) { startPlainFile( buildFileName( manName ) ); firstCol=TRUE; } void ManGenerator::endFile() { t << endl; endPlainFile(); } void ManGenerator::endTitleHead(const char *,const char *name) { t << ".TH \"" << name << "\" " << getExtension() << " \"" << dateToString(FALSE) << "\" \""; if (Config_getString("PROJECT_NAME").isEmpty()) t << "Doxygen"; else t << Config_getString("PROJECT_NAME"); t << "\" \\\" -*- nroff -*-" << endl; t << ".ad l" << endl; t << ".nh" << endl; t << ".SH NAME" << endl; t << name << " \\- "; firstCol=FALSE; inHeader=TRUE; } void ManGenerator::newParagraph() { if (!paragraph) { if (!firstCol) t << endl; t << ".PP" << endl; firstCol=TRUE; } paragraph=TRUE; } void ManGenerator::writeString(const char *text) { docify(text); } void ManGenerator::writeIndexItem(const char *,const char *, const char *) { } void ManGenerator::writeStartAnnoItem(const char *,const char *, const char *,const char *) { } void ManGenerator::writeObjectLink(const char *,const char *, const char *, const char *name) { startBold(); docify(name); endBold(); } void ManGenerator::writeCodeLink(const char *,const char *, const char *, const char *name) { docify(name); } void ManGenerator::startHtmlLink(const char *) { } void ManGenerator::endHtmlLink() { } void ManGenerator::writeMailLink(const char *url) { docify(url); } void ManGenerator::startGroupHeader() { if (!firstCol) t << endl; t << ".SH \""; upperCase=TRUE; firstCol=FALSE; } void ManGenerator::endGroupHeader() { t << "\"\n.PP " << endl; firstCol=TRUE; paragraph=TRUE; upperCase=FALSE; } void ManGenerator::startMemberHeader() { if (!firstCol) t << endl; t << ".SS \""; } void ManGenerator::endMemberHeader() { t << "\"\n"; firstCol=TRUE; paragraph=FALSE; } void ManGenerator::docify(const char *str) { if (str) { const char *p=str; char c=0; while ((c=*p++)) { switch(c) { case '\\': t << "\\\\"; col++; break; case '\n': t << "\n"; col=0; break; case '\"': c = '\''; // no break! default: t << c; col++; break; } } firstCol=(c=='\n'); //printf("%s",str);fflush(stdout); } paragraph=FALSE; } void ManGenerator::codify(const char *str) { //static char spaces[]=" "; if (str) { const char *p=str; char c; int spacesToNextTabStop; while (*p) { c=*p++; switch(c) { case '\t': spacesToNextTabStop = Config_getInt("TAB_SIZE") - (col%Config_getInt("TAB_SIZE")); t << spaces.left(spacesToNextTabStop); col+=spacesToNextTabStop; break; case '\n': t << "\n"; firstCol=TRUE; col=0; break; case '\\': t << "\\"; col++; break; case '\"': c = '\''; // no break! default: t << c; firstCol=FALSE; col++; break; } } //printf("%s",str);fflush(stdout); } paragraph=FALSE; } void ManGenerator::writeChar(char c) { firstCol=(c=='\n'); if (firstCol) col=0; else col++; switch (c) { case '\\': t << "\\\\"; break; case '\"': c = '\''; // no break! default: t << c; break; } //printf("%c",c);fflush(stdout); paragraph=FALSE; } void ManGenerator::startDescList(SectionTypes) { if (!firstCol) { t << endl << ".PP" << endl; firstCol=TRUE; paragraph=TRUE; col=0; } paragraph=FALSE; startBold(); } void ManGenerator::startParamList(ParamListTypes) { if (!firstCol) { t << endl << ".PP" << endl; firstCol=TRUE; paragraph=TRUE; col=0; } paragraph=FALSE; } void ManGenerator::startTitle() { if (!firstCol) t << endl; t << ".SH \""; firstCol=FALSE; paragraph=FALSE; } void ManGenerator::endTitle() { t << "\""; } void ManGenerator::writeListItem() { if (!firstCol) t << endl; t << ".TP" << endl; firstCol=TRUE; paragraph=FALSE; col=0; } void ManGenerator::startCodeFragment() { newParagraph(); t << ".nf" << endl; firstCol=TRUE; paragraph=FALSE; } void ManGenerator::endCodeFragment() { if (!firstCol) t << endl; t << ".fi" << endl; firstCol=TRUE; paragraph=FALSE; col=0; } void ManGenerator::startMemberDoc(const char *,const char *,const char *,const char *) { if (!firstCol) t << endl; t << ".SS \""; firstCol=FALSE; paragraph=FALSE; } void ManGenerator::startDoxyAnchor(const char *,const char *manName, const char *, const char *name) { // something to be done? if( !Config_getBool("MAN_LINKS") ) { return; // no } // the name of the link file is derived from the name of the anchor: // - truncate after an (optional) :: QCString baseName = name; int i=baseName.findRev(':'); if (i!=-1) baseName=baseName.right(baseName.length()-i-1); // - remove dangerous characters and append suffix, then add dir prefix QCString fileName=dir+"/"+buildFileName( baseName ); QFile linkfile( fileName ); // - only create file if it doesn't exist already if ( !linkfile.open( IO_ReadOnly ) ) { if ( linkfile.open( IO_WriteOnly ) ) { QTextStream linkstream; linkstream.setDevice(&linkfile); #if QT_VERSION >= 200 linkstream.setEncoding(QTextStream::Latin1); #endif linkstream << ".so man" << getExtension() << "/" << buildFileName( manName ) << endl; } } linkfile.close(); } void ManGenerator::endMemberDoc() { t << "\""; } void ManGenerator::startSubsection() { if (!firstCol) t << endl; t << ".SS \""; firstCol=FALSE; paragraph=FALSE; } void ManGenerator::endSubsection() { t << "\""; } void ManGenerator::startSubsubsection() { if (!firstCol) t << endl; t << "\n.SS \""; firstCol=FALSE; paragraph=FALSE; } void ManGenerator::endSubsubsection() { t << "\""; } void ManGenerator::writeSynopsis() { if (!firstCol) t << endl; t << ".SH SYNOPSIS\n.br\n.PP\n"; firstCol=TRUE; paragraph=FALSE; } void ManGenerator::startDescItem() { if (!firstCol) t << endl; t << ".IP \""; firstCol=FALSE; } void ManGenerator::endDescTitle() { endBold(); paragraph=TRUE; } void ManGenerator::writeDescItem() { if (!firstCol) t << endl; if (!paragraph) t << ".in -1c" << endl; t << ".in +1c" << endl; firstCol=TRUE; paragraph=FALSE; col=0; } void ManGenerator::endDescItem() { t << "\" 1c" << endl; firstCol=TRUE; } void ManGenerator::startMemberItem(int annType) { if (firstCol && !insideTabbing) t << ".in +1c\n"; t << "\n.ti -1c\n.RI \""; firstCol=FALSE; if (annType!=0) insideTabbing=TRUE; } void ManGenerator::endMemberItem(bool endItem) { if (endItem) { insideTabbing=FALSE; t << "\"\n.br\n.RI \""; } t << "\"\n.br"; } void ManGenerator::startMemberList() { if (!insideTabbing) { t << "\n.in +1c"; firstCol=FALSE; } } void ManGenerator::endMemberList() { if (!insideTabbing) { t << "\n.in -1c"; firstCol=FALSE; } } void ManGenerator::startMemberGroupHeader(bool) { t << "\n.PP\n.RI \"\\fB"; } void ManGenerator::endMemberGroupHeader() { t << "\\fP\"\n.br\n"; firstCol=TRUE; } void ManGenerator::startMemberGroupDocs() { } void ManGenerator::endMemberGroupDocs() { t << "\n.PP"; } void ManGenerator::startMemberGroup() { t << "\n.in +1c"; } void ManGenerator::endMemberGroup(bool) { t << "\n.in -1c"; firstCol=FALSE; } void ManGenerator::startSection(const char *,const char *,bool sub) { if( !inHeader ) { if( sub ) { startMemberHeader(); } else { startGroupHeader(); } } } void ManGenerator::endSection(const char *,bool sub) { if( !inHeader ) { if( sub ) { endMemberHeader(); } else { endGroupHeader(); } } else { t << "\n"; firstCol=TRUE; paragraph=FALSE; inHeader=FALSE; } }