1 /******************************************************************************
5 * Copyright (C) 1997-2014 by Dimitri van Heesch.
7 * Permission to use, copy, modify, and distribute this software and its
8 * documentation under the terms of the GNU General Public License is hereby
9 * granted. No representations are made about the suitability of this software
10 * for any purpose. It is provided "as is" without express or implied warranty.
11 * See the GNU General Public License for more details.
13 * Documents produced by Doxygen are derivative works derived from the
14 * input used in their production; they are not affected by this license.
23 #include "classlist.h"
24 #include "outputlist.h"
25 #include "namespacedef.h"
28 #include "memberlist.h"
30 #include "membergroup.h"
33 #include "docparser.h"
34 #include "searchindex.h"
36 #include "vhdldocgen.h"
38 #include "arguments.h"
40 #include "membername.h"
44 //---------------------------------------------------------------------------
46 GroupDef::GroupDef(const char *df,int dl,const char *na,const char *t,
47 const char *refFileName) : Definition(df,dl,1,na)
49 fileList = new FileList;
50 classSDict = new ClassSDict(17);
51 groupList = new GroupList;
52 namespaceSDict = new NamespaceSDict(17);
53 pageDict = new PageSDict(17);
54 exampleDict = new PageSDict(17);
55 dirList = new DirList;
56 allMemberNameInfoSDict = new MemberNameInfoSDict(17);
57 allMemberNameInfoSDict->setAutoDelete(TRUE);
60 fileName=stripExtension(refFileName);
64 fileName = (QCString)"group_"+na;
67 memberGroupSDict = new MemberGroupSDict;
68 memberGroupSDict->setAutoDelete(TRUE);
70 allMemberList = new MemberList(MemberListType_allMembersList);
74 m_subGrouping=Config_getBool("SUBGROUPING");
82 delete namespaceSDict;
86 delete allMemberNameInfoSDict;
87 delete memberGroupSDict;
91 void GroupDef::setGroupTitle( const char *t )
93 if ( t && qstrlen(t) )
101 title.at(0)=toupper(title.at(0));
107 void GroupDef::distributeMemberGroupDocumentation()
109 MemberGroupSDict::Iterator mgli(*memberGroupSDict);
111 for (;(mg=mgli.current());++mgli)
113 mg->distributeMemberGroupDocumentation();
117 void GroupDef::findSectionsInDocumentation()
119 docFindSections(documentation(),this,0,docFile());
120 MemberGroupSDict::Iterator mgli(*memberGroupSDict);
122 for (;(mg=mgli.current());++mgli)
124 mg->findSectionsInDocumentation();
127 QListIterator<MemberList> mli(m_memberLists);
129 for (mli.toFirst();(ml=mli.current());++mli)
131 if (ml->listType()&MemberListType_declarationLists)
133 ml->findSectionsInDocumentation();
138 void GroupDef::addFile(const FileDef *def)
140 static bool sortBriefDocs = Config_getBool("SORT_BRIEF_DOCS");
141 if (def->isHidden()) return;
144 fileList->inSort(def);
146 fileList->append(def);
149 bool GroupDef::addClass(const ClassDef *cd)
151 static bool sortBriefDocs = Config_getBool("SORT_BRIEF_DOCS");
152 if (cd->isHidden()) return FALSE;
154 if (classSDict->find(cd->qualifiedName())==0)
156 QCString qn = cd->qualifiedName();
157 //printf("--- addClass %s sort=%d\n",qn.data(),sortBriefDocs);
160 classSDict->inSort(cd->qualifiedName(),cd);
164 int i=qn.findRev("::");
165 if (i==-1) i=qn.find('.');
167 //printf("i=%d\n",i);
170 // add nested classes (e.g. A::B, A::C) after their parent (A) in
171 // order of insertion
172 QCString scope = qn.left(i);
173 int j=classSDict->findAt(scope);
176 while (j<(int)classSDict->count() &&
177 classSDict->at(j)->qualifiedName().left(i)==scope)
179 //printf("skipping over %s\n",classSDict->at(j)->qualifiedName().data());
182 //printf("Found scope at index %d\n",j);
183 classSDict->insertAt(j,cd->qualifiedName(),cd);
187 if (!found) // no insertion point found -> just append
189 classSDict->append(cd->qualifiedName(),cd);
197 bool GroupDef::addNamespace(const NamespaceDef *def)
199 static bool sortBriefDocs = Config_getBool("SORT_BRIEF_DOCS");
200 if (def->isHidden()) return FALSE;
202 if (namespaceSDict->find(def->name())==0)
205 namespaceSDict->inSort(def->name(),def);
207 namespaceSDict->append(def->name(),def);
213 void GroupDef::addDir(const DirDef *def)
215 if (def->isHidden()) return;
216 if (Config_getBool("SORT_BRIEF_DOCS"))
217 dirList->inSort(def);
219 dirList->append(def);
222 void GroupDef::addPage(PageDef *def)
224 if (def->isHidden()) return;
225 //printf("Making page %s part of a group\n",def->name.data());
226 pageDict->append(def->name(),def);
227 def->makePartOfGroup(this);
230 void GroupDef::addExample(const PageDef *def)
232 if (def->isHidden()) return;
233 exampleDict->append(def->name(),def);
237 void GroupDef::addMembersToMemberGroup()
239 QListIterator<MemberList> mli(m_memberLists);
241 for (mli.toFirst();(ml=mli.current());++mli)
243 if (ml->listType()&MemberListType_declarationLists)
245 ::addMembersToMemberGroup(ml,&memberGroupSDict,this);
249 //printf("GroupDef::addMembersToMemberGroup() memberGroupList=%d\n",memberGroupList->count());
250 MemberGroupSDict::Iterator mgli(*memberGroupSDict);
252 for (;(mg=mgli.current());++mgli)
254 mg->setInGroup(TRUE);
259 bool GroupDef::insertMember(MemberDef *md,bool docOnly)
261 if (md->isHidden()) return FALSE;
263 //printf("GroupDef(%s)::insertMember(%s)\n", title.data(), md->name().data());
264 MemberNameInfo *mni=0;
265 if ((mni=(*allMemberNameInfoSDict)[md->name()]))
266 { // member with this name already found
267 MemberNameInfoIterator srcMnii(*mni);
269 for ( ; (srcMi=srcMnii.current()) ; ++srcMnii )
271 MemberDef *srcMd = srcMi->memberDef;
272 if (srcMd==md) return FALSE; // already added before!
274 bool sameScope = srcMd->getOuterScope()==md->getOuterScope() || // same class or namespace
275 // both inside a file => definition and declaration do not have to be in the same file
276 (srcMd->getOuterScope()->definitionType()==Definition::TypeFile &&
277 md->getOuterScope()->definitionType()==Definition::TypeFile);
279 ArgumentList *srcMdAl = srcMd->argumentList();
280 ArgumentList *mdAl = md->argumentList();
281 ArgumentList *tSrcMdAl = srcMd->templateArguments();
282 ArgumentList *tMdAl = md->templateArguments();
284 if (srcMd->isFunction() && md->isFunction() && // both are a function
285 ((tSrcMdAl==0 && tMdAl==0) ||
286 (tSrcMdAl!=0 && tMdAl!=0 && tSrcMdAl->count()==tMdAl->count())
287 ) && // same number of template arguments
288 matchArguments2(srcMd->getOuterScope(),srcMd->getFileDef(),srcMdAl,
289 md->getOuterScope(),md->getFileDef(),mdAl,
291 ) && // matching parameters
292 sameScope // both are found in the same scope
295 if (srcMd->getGroupAlias()==0)
297 md->setGroupAlias(srcMd);
299 else if (md!=srcMd->getGroupAlias())
301 md->setGroupAlias(srcMd->getGroupAlias());
303 return FALSE; // member is the same as one that is already added
306 mni->append(new MemberInfo(md,md->protection(),md->virtualness(),FALSE));
310 mni = new MemberNameInfo(md->name());
311 mni->append(new MemberInfo(md,md->protection(),md->virtualness(),FALSE));
312 allMemberNameInfoSDict->append(mni->memberName(),mni);
314 //printf("Added member!\n");
315 allMemberList->append(md);
316 switch(md->memberType())
318 case MemberType_Variable:
321 addMemberToList(MemberListType_decVarMembers,md);
323 addMemberToList(MemberListType_docVarMembers,md);
325 case MemberType_Function:
328 addMemberToList(MemberListType_decFuncMembers,md);
330 addMemberToList(MemberListType_docFuncMembers,md);
332 case MemberType_Typedef:
335 addMemberToList(MemberListType_decTypedefMembers,md);
337 addMemberToList(MemberListType_docTypedefMembers,md);
339 case MemberType_Enumeration:
342 addMemberToList(MemberListType_decEnumMembers,md);
344 addMemberToList(MemberListType_docEnumMembers,md);
346 case MemberType_EnumValue:
349 addMemberToList(MemberListType_decEnumValMembers,md);
351 addMemberToList(MemberListType_docEnumValMembers,md);
353 case MemberType_Define:
356 addMemberToList(MemberListType_decDefineMembers,md);
358 addMemberToList(MemberListType_docDefineMembers,md);
360 case MemberType_Signal:
363 addMemberToList(MemberListType_decSignalMembers,md);
365 addMemberToList(MemberListType_docSignalMembers,md);
367 case MemberType_Slot:
368 if (md->protection()==Public)
372 addMemberToList(MemberListType_decPubSlotMembers,md);
374 addMemberToList(MemberListType_docPubSlotMembers,md);
376 else if (md->protection()==Protected)
380 addMemberToList(MemberListType_decProSlotMembers,md);
382 addMemberToList(MemberListType_docProSlotMembers,md);
388 addMemberToList(MemberListType_decPriSlotMembers,md);
390 addMemberToList(MemberListType_docPriSlotMembers,md);
393 case MemberType_Event:
396 addMemberToList(MemberListType_decEventMembers,md);
398 addMemberToList(MemberListType_docEventMembers,md);
400 case MemberType_Property:
403 addMemberToList(MemberListType_decPropMembers,md);
405 addMemberToList(MemberListType_docPropMembers,md);
407 case MemberType_Friend:
410 addMemberToList(MemberListType_decFriendMembers,md);
412 addMemberToList(MemberListType_docFriendMembers,md);
415 err("GroupDef::insertMembers(): "
416 "member `%s' (typeid=%d) with scope `%s' inserted in group scope `%s'!\n",
417 md->name().data(),md->memberType(),
418 md->getClassDef() ? md->getClassDef()->name().data() : "",
424 void GroupDef::removeMember(MemberDef *md)
426 // fprintf(stderr, "GroupDef(%s)::removeMember( %s )\n", title.data(), md->name().data());
427 MemberNameInfo *mni = allMemberNameInfoSDict->find(md->name());
430 MemberNameInfoIterator mnii(*mni);
431 while( mnii.current() )
433 if( mnii.current()->memberDef == md )
435 mni->remove(mnii.current());
442 allMemberNameInfoSDict->remove(md->name());
445 removeMemberFromList(MemberListType_allMembersList,md);
446 switch(md->memberType())
448 case MemberType_Variable:
449 removeMemberFromList(MemberListType_decVarMembers,md);
450 removeMemberFromList(MemberListType_docVarMembers,md);
452 case MemberType_Function:
453 removeMemberFromList(MemberListType_decFuncMembers,md);
454 removeMemberFromList(MemberListType_docFuncMembers,md);
456 case MemberType_Typedef:
457 removeMemberFromList(MemberListType_decTypedefMembers,md);
458 removeMemberFromList(MemberListType_docTypedefMembers,md);
460 case MemberType_Enumeration:
461 removeMemberFromList(MemberListType_decEnumMembers,md);
462 removeMemberFromList(MemberListType_docEnumMembers,md);
464 case MemberType_EnumValue:
465 removeMemberFromList(MemberListType_decEnumValMembers,md);
466 removeMemberFromList(MemberListType_docEnumValMembers,md);
468 case MemberType_Define:
469 removeMemberFromList(MemberListType_decDefineMembers,md);
470 removeMemberFromList(MemberListType_docDefineMembers,md);
472 case MemberType_Signal:
473 removeMemberFromList(MemberListType_decSignalMembers,md);
474 removeMemberFromList(MemberListType_docSignalMembers,md);
476 case MemberType_Slot:
477 if (md->protection()==Public)
479 removeMemberFromList(MemberListType_decPubSlotMembers,md);
480 removeMemberFromList(MemberListType_docPubSlotMembers,md);
482 else if (md->protection()==Protected)
484 removeMemberFromList(MemberListType_decProSlotMembers,md);
485 removeMemberFromList(MemberListType_docProSlotMembers,md);
489 removeMemberFromList(MemberListType_decPriSlotMembers,md);
490 removeMemberFromList(MemberListType_docPriSlotMembers,md);
493 case MemberType_Event:
494 removeMemberFromList(MemberListType_decEventMembers,md);
495 removeMemberFromList(MemberListType_docEventMembers,md);
497 case MemberType_Property:
498 removeMemberFromList(MemberListType_decPropMembers,md);
499 removeMemberFromList(MemberListType_docPropMembers,md);
501 case MemberType_Friend:
502 removeMemberFromList(MemberListType_decFriendMembers,md);
503 removeMemberFromList(MemberListType_docFriendMembers,md);
506 err("GroupDef::removeMember(): unexpected member remove in file!\n");
511 bool GroupDef::containsGroup(const GroupDef *def)
513 return this==def || groupList->find(def) >= 0;
516 void GroupDef::addGroup(const GroupDef *def)
518 //printf("adding group `%s' to group `%s'\n",def->name().data(),name().data());
519 //if (Config_getBool("SORT_MEMBER_DOCS"))
520 // groupList->inSort(def);
522 groupList->append(def);
525 bool GroupDef::isASubGroup() const
527 GroupList *groups = partOfGroups();
528 return groups!=0 && groups->count()!=0;
531 int GroupDef::countMembers() const
533 return fileList->count()+
535 namespaceSDict->count()+
537 allMemberList->count()+
539 exampleDict->count();
542 /*! Compute the HTML anchor names for all members in the group */
543 void GroupDef::computeAnchors()
545 //printf("GroupDef::computeAnchors()\n");
546 setAnchors(allMemberList);
549 void GroupDef::writeDetailedDescription(OutputList &ol,const QCString &title)
551 if ((!briefDescription().isEmpty() && Config_getBool("REPEAT_BRIEF"))
552 || !documentation().isEmpty() || !inbodyDocumentation().isEmpty()
555 if (pageDict->count()!=countMembers()) // not only pages -> classical layout
557 ol.pushGeneratorState();
558 ol.disable(OutputGenerator::Html);
560 ol.popGeneratorState();
561 ol.pushGeneratorState();
562 ol.disableAllBut(OutputGenerator::Html);
563 ol.writeAnchor(0,"details");
564 ol.popGeneratorState();
565 ol.startGroupHeader();
570 // repeat brief description
571 if (!briefDescription().isEmpty() && Config_getBool("REPEAT_BRIEF"))
573 ol.generateDoc(briefFile(),briefLine(),this,0,briefDescription(),FALSE,FALSE);
575 // write separator between brief and details
576 if (!briefDescription().isEmpty() && Config_getBool("REPEAT_BRIEF") &&
577 !documentation().isEmpty())
579 ol.pushGeneratorState();
580 ol.disable(OutputGenerator::Man);
581 ol.disable(OutputGenerator::RTF);
582 // ol.newParagraph(); // FIXME:PARA
584 ol.disableAllBut(OutputGenerator::Man);
585 ol.enable(OutputGenerator::Latex);
586 ol.writeString("\n\n");
587 ol.popGeneratorState();
590 // write detailed documentation
591 if (!documentation().isEmpty())
593 ol.generateDoc(docFile(),docLine(),this,0,documentation()+"\n",TRUE,FALSE);
596 // write inbody documentation
597 if (!inbodyDocumentation().isEmpty())
599 ol.generateDoc(inbodyFile(),inbodyLine(),this,0,inbodyDocumentation()+"\n",TRUE,FALSE);
604 void GroupDef::writeBriefDescription(OutputList &ol)
606 if (!briefDescription().isEmpty() && Config_getBool("BRIEF_MEMBER_DESC"))
608 DocRoot *rootNode = validatingParseDoc(briefFile(),briefLine(),this,0,
609 briefDescription(),TRUE,FALSE,0,TRUE,FALSE);
610 if (rootNode && !rootNode->isEmpty())
613 ol.writeDoc(rootNode,this,0);
614 ol.pushGeneratorState();
615 ol.disable(OutputGenerator::RTF);
616 ol.writeString(" \n");
617 ol.enable(OutputGenerator::RTF);
619 if (Config_getBool("REPEAT_BRIEF") ||
620 !documentation().isEmpty()
623 ol.disableAllBut(OutputGenerator::Html);
624 ol.startTextLink(0,"details");
625 ol.parseText(theTranslator->trMore());
628 ol.popGeneratorState();
635 void GroupDef::writeGroupGraph(OutputList &ol)
637 if (Config_getBool("HAVE_DOT") /*&& Config_getBool("GROUP_GRAPHS")*/ )
639 DotGroupCollaboration graph(this);
640 if (!graph.isTrivial())
642 msg("Generating dependency graph for group %s\n",qualifiedName().data());
643 ol.pushGeneratorState();
644 ol.disable(OutputGenerator::Man);
645 //ol.startParagraph();
646 ol.startGroupCollaboration();
647 ol.parseText(theTranslator->trCollaborationDiagram(title));
648 ol.endGroupCollaboration(graph);
650 ol.popGeneratorState();
655 void GroupDef::writeFiles(OutputList &ol,const QCString &title)
657 // write list of files
658 if (fileList->count()>0)
660 ol.startMemberHeader("files");
662 ol.endMemberHeader();
663 ol.startMemberList();
664 QListIterator<FileDef> it(*fileList);
666 for (;(fd=it.current());++it)
668 ol.startMemberDeclaration();
669 ol.startMemberItem(fd->getOutputFileBase(),0);
670 ol.docify(theTranslator->trFile(FALSE,TRUE)+" ");
671 ol.insertMemberAlign();
672 ol.writeObjectLink(fd->getReference(),fd->getOutputFileBase(),0,fd->name());
673 if (!Config_getString("GENERATE_TAGFILE").isEmpty())
675 Doxygen::tagFile << " <file>" << convertToXML(fd->name()) << "</file>" << endl;
678 if (!fd->briefDescription().isEmpty() && Config_getBool("BRIEF_MEMBER_DESC"))
680 ol.startMemberDescription(fd->getOutputFileBase());
681 ol.generateDoc(briefFile(),briefLine(),fd,0,fd->briefDescription(),FALSE,FALSE,0,TRUE,FALSE);
682 ol.endMemberDescription();
684 ol.endMemberDeclaration(0,0);
690 void GroupDef::writeNamespaces(OutputList &ol,const QCString &title)
692 // write list of namespaces
693 namespaceSDict->writeDeclaration(ol,title);
696 void GroupDef::writeNestedGroups(OutputList &ol,const QCString &title)
698 // write list of groups
700 if (groupList->count()>0)
702 QListIterator<GroupDef> it(*groupList);
704 for (;(gd=it.current());++it)
706 if (gd->isVisible()) count++;
711 ol.startMemberHeader("groups");
713 ol.endMemberHeader();
714 ol.startMemberList();
715 if (Config_getBool("SORT_GROUP_NAMES"))
719 QListIterator<GroupDef> it(*groupList);
721 for (;(gd=it.current());++it)
725 ol.startMemberDeclaration();
726 ol.startMemberItem(gd->getOutputFileBase(),0);
727 //ol.docify(theTranslator->trGroup(FALSE,TRUE));
729 ol.insertMemberAlign();
730 ol.writeObjectLink(gd->getReference(),gd->getOutputFileBase(),0,gd->groupTitle());
731 if (!Config_getString("GENERATE_TAGFILE").isEmpty())
733 Doxygen::tagFile << " <subgroup>" << convertToXML(gd->name()) << "</subgroup>" << endl;
736 if (!gd->briefDescription().isEmpty() && Config_getBool("BRIEF_MEMBER_DESC"))
738 ol.startMemberDescription(gd->getOutputFileBase());
739 ol.generateDoc(briefFile(),briefLine(),gd,0,gd->briefDescription(),FALSE,FALSE,0,TRUE,FALSE);
740 ol.endMemberDescription();
742 ol.endMemberDeclaration(0,0);
749 void GroupDef::writeDirs(OutputList &ol,const QCString &title)
751 // write list of directories
752 if (dirList->count()>0)
754 ol.startMemberHeader("dirs");
756 ol.endMemberHeader();
757 ol.startMemberList();
758 QListIterator<DirDef> it(*dirList);
760 for (;(dd=it.current());++it)
762 ol.startMemberDeclaration();
763 ol.startMemberItem(dd->getOutputFileBase(),0);
764 ol.parseText(theTranslator->trDir(FALSE,TRUE));
765 ol.insertMemberAlign();
766 ol.writeObjectLink(dd->getReference(),dd->getOutputFileBase(),0,dd->shortName());
768 if (!Config_getString("GENERATE_TAGFILE").isEmpty())
770 Doxygen::tagFile << " <dir>" << convertToXML(dd->displayName()) << "</dir>" << endl;
772 if (!dd->briefDescription().isEmpty() && Config_getBool("BRIEF_MEMBER_DESC"))
774 ol.startMemberDescription(dd->getOutputFileBase());
775 ol.generateDoc(briefFile(),briefLine(),dd,0,dd->briefDescription(),FALSE,FALSE,0,TRUE,FALSE);
776 ol.endMemberDescription();
778 ol.endMemberDeclaration(0,0);
785 void GroupDef::writeClasses(OutputList &ol,const QCString &title)
787 // write list of classes
788 classSDict->writeDeclaration(ol,0,title,FALSE);
791 void GroupDef::writeInlineClasses(OutputList &ol)
793 classSDict->writeDocumentation(ol);
796 void GroupDef::writePageDocumentation(OutputList &ol)
799 PageSDict::Iterator pdi(*pageDict);
800 for (pdi.toFirst();(pd=pdi.current());++pdi)
802 if (!pd->isReference())
804 QCString pageName = pd->getOutputFileBase();
806 if (!Config_getString("GENERATE_TAGFILE").isEmpty())
808 Doxygen::tagFile << " <page>" << convertToXML(pageName) << "</page>" << endl;
812 if (!pd->title().isEmpty() && !pd->name().isEmpty() &&
813 (si=Doxygen::sectionDict->find(pd->name()))!=0)
815 ol.startSection(si->label,si->title,SectionInfo::Subsection);
816 ol.docify(si->title);
817 ol.endSection(si->label,SectionInfo::Subsection);
820 ol.generateDoc(pd->docFile(),pd->docLine(),pd,0,pd->documentation()+pd->inbodyDocumentation(),TRUE,FALSE,0,TRUE,FALSE);
826 void GroupDef::writeMemberGroups(OutputList &ol)
828 /* write user defined member groups */
829 if (memberGroupSDict)
831 memberGroupSDict->sort();
832 /* write user defined member groups */
833 MemberGroupSDict::Iterator mgli(*memberGroupSDict);
835 for (;(mg=mgli.current());++mgli)
837 mg->writeDeclarations(ol,0,0,0,this);
842 void GroupDef::startMemberDeclarations(OutputList &ol)
844 ol.startMemberSections();
847 void GroupDef::endMemberDeclarations(OutputList &ol)
849 ol.endMemberSections();
852 void GroupDef::startMemberDocumentation(OutputList &ol)
854 //printf("** GroupDef::startMemberDocumentation()\n");
855 if (Config_getBool("SEPARATE_MEMBER_PAGES"))
857 ol.pushGeneratorState();
858 ol.disable(OutputGenerator::Html);
859 Doxygen::suppressDocWarnings = TRUE;
863 void GroupDef::endMemberDocumentation(OutputList &ol)
865 //printf("** GroupDef::endMemberDocumentation()\n");
866 if (Config_getBool("SEPARATE_MEMBER_PAGES"))
868 ol.popGeneratorState();
869 Doxygen::suppressDocWarnings = FALSE;
873 void GroupDef::writeAuthorSection(OutputList &ol)
875 // write Author section (Man only)
876 ol.pushGeneratorState();
877 ol.disableAllBut(OutputGenerator::Man);
878 ol.startGroupHeader();
879 ol.parseText(theTranslator->trAuthor(TRUE,TRUE));
881 ol.parseText(theTranslator->trGeneratedAutomatically(Config_getString("PROJECT_NAME")));
882 ol.popGeneratorState();
885 void GroupDef::writeSummaryLinks(OutputList &ol)
887 ol.pushGeneratorState();
888 ol.disableAllBut(OutputGenerator::Html);
889 QListIterator<LayoutDocEntry> eli(
890 LayoutDocManager::instance().docEntries(LayoutDocManager::Group));
893 SrcLangExt lang = getLanguage();
894 for (eli.toFirst();(lde=eli.current());++eli)
896 if ((lde->kind()==LayoutDocEntry::GroupClasses && classSDict->declVisible()) ||
897 (lde->kind()==LayoutDocEntry::GroupNamespaces && namespaceSDict->declVisible()) ||
898 (lde->kind()==LayoutDocEntry::GroupFiles && fileList->count()>0) ||
899 (lde->kind()==LayoutDocEntry::GroupNestedGroups && groupList->count()>0) ||
900 (lde->kind()==LayoutDocEntry::GroupDirs && dirList->count()>0)
903 LayoutDocEntrySection *ls = (LayoutDocEntrySection*)lde;
904 QCString label = lde->kind()==LayoutDocEntry::GroupClasses ? "nested-classes" :
905 lde->kind()==LayoutDocEntry::GroupNamespaces ? "namespaces" :
906 lde->kind()==LayoutDocEntry::GroupFiles ? "files" :
907 lde->kind()==LayoutDocEntry::GroupNestedGroups ? "groups" :
909 ol.writeSummaryLink(0,label,ls->title(lang),first);
912 else if (lde->kind()==LayoutDocEntry::MemberDecl)
914 LayoutDocEntryMemberDecl *lmd = (LayoutDocEntryMemberDecl*)lde;
915 MemberList * ml = getMemberList(lmd->type);
916 if (ml && ml->declVisible())
918 ol.writeSummaryLink(0,MemberList::listTypeAsString(ml->listType()),lmd->title(lang),first);
925 ol.writeString(" </div>\n");
927 ol.popGeneratorState();
930 void GroupDef::writeDocumentation(OutputList &ol)
932 //static bool generateTreeView = Config_getBool("GENERATE_TREEVIEW");
933 ol.pushGeneratorState();
934 startFile(ol,getOutputFileBase(),name(),title,HLI_None);
936 ol.startHeaderSection();
937 writeSummaryLinks(ol);
938 ol.startTitleHead(getOutputFileBase());
939 ol.pushGeneratorState();
940 ol.disable(OutputGenerator::Man);
942 ol.popGeneratorState();
943 addGroupListToTitle(ol,this);
944 ol.endTitleHead(getOutputFileBase(),title);
945 ol.endHeaderSection();
948 if (Doxygen::searchIndex)
950 Doxygen::searchIndex->setCurrentDoc(this,anchor(),FALSE);
951 static QRegExp we("[a-zA-Z_][-a-zA-Z_0-9]*");
953 while ((i=we.match(title,p,&l))!=-1) // foreach word in the title
955 Doxygen::searchIndex->addWord(title.mid(i,l),TRUE);
960 Doxygen::indexList->addIndexItem(this,0,0,title);
962 if (!Config_getString("GENERATE_TAGFILE").isEmpty())
964 Doxygen::tagFile << " <compound kind=\"group\">" << endl;
965 Doxygen::tagFile << " <name>" << convertToXML(name()) << "</name>" << endl;
966 Doxygen::tagFile << " <title>" << convertToXML(title) << "</title>" << endl;
967 Doxygen::tagFile << " <filename>" << convertToXML(getOutputFileBase()) << Doxygen::htmlFileExtension << "</filename>" << endl;
971 //---------------------------------------- start flexible part -------------------------------
973 SrcLangExt lang=getLanguage();
974 QListIterator<LayoutDocEntry> eli(
975 LayoutDocManager::instance().docEntries(LayoutDocManager::Group));
977 for (eli.toFirst();(lde=eli.current());++eli)
981 case LayoutDocEntry::BriefDesc:
982 writeBriefDescription(ol);
984 case LayoutDocEntry::MemberDeclStart:
985 startMemberDeclarations(ol);
987 case LayoutDocEntry::GroupClasses:
989 LayoutDocEntrySection *ls = (LayoutDocEntrySection*)lde;
990 writeClasses(ol,ls->title(lang));
993 case LayoutDocEntry::GroupInlineClasses:
995 writeInlineClasses(ol);
998 case LayoutDocEntry::GroupNamespaces:
1000 LayoutDocEntrySection *ls = (LayoutDocEntrySection*)lde;
1001 writeNamespaces(ol,ls->title(lang));
1004 case LayoutDocEntry::MemberGroups:
1005 writeMemberGroups(ol);
1007 case LayoutDocEntry::MemberDecl:
1009 LayoutDocEntryMemberDecl *lmd = (LayoutDocEntryMemberDecl*)lde;
1010 writeMemberDeclarations(ol,lmd->type,lmd->title(lang));
1013 case LayoutDocEntry::MemberDeclEnd:
1014 endMemberDeclarations(ol);
1016 case LayoutDocEntry::DetailedDesc:
1018 LayoutDocEntrySection *ls = (LayoutDocEntrySection*)lde;
1019 writeDetailedDescription(ol,ls->title(lang));
1022 case LayoutDocEntry::MemberDefStart:
1023 startMemberDocumentation(ol);
1025 case LayoutDocEntry::MemberDef:
1027 LayoutDocEntryMemberDef *lmd = (LayoutDocEntryMemberDef*)lde;
1028 writeMemberDocumentation(ol,lmd->type,lmd->title(lang));
1031 case LayoutDocEntry::MemberDefEnd:
1032 endMemberDocumentation(ol);
1034 case LayoutDocEntry::GroupNestedGroups:
1036 LayoutDocEntrySection *ls = (LayoutDocEntrySection*)lde;
1037 writeNestedGroups(ol,ls->title(lang));
1040 case LayoutDocEntry::GroupPageDocs:
1041 writePageDocumentation(ol);
1043 case LayoutDocEntry::GroupDirs:
1045 LayoutDocEntrySection *ls = (LayoutDocEntrySection*)lde;
1046 writeDirs(ol,ls->title(lang));
1049 case LayoutDocEntry::GroupFiles:
1051 LayoutDocEntrySection *ls = (LayoutDocEntrySection*)lde;
1052 writeFiles(ol,ls->title(lang));
1055 case LayoutDocEntry::GroupGraph:
1056 writeGroupGraph(ol);
1058 case LayoutDocEntry::AuthorSection:
1059 writeAuthorSection(ol);
1061 case LayoutDocEntry::ClassIncludes:
1062 case LayoutDocEntry::ClassInheritanceGraph:
1063 case LayoutDocEntry::ClassNestedClasses:
1064 case LayoutDocEntry::ClassCollaborationGraph:
1065 case LayoutDocEntry::ClassAllMembersLink:
1066 case LayoutDocEntry::ClassUsedFiles:
1067 case LayoutDocEntry::ClassInlineClasses:
1068 case LayoutDocEntry::NamespaceNestedNamespaces:
1069 case LayoutDocEntry::NamespaceNestedConstantGroups:
1070 case LayoutDocEntry::NamespaceClasses:
1071 case LayoutDocEntry::NamespaceInlineClasses:
1072 case LayoutDocEntry::FileClasses:
1073 case LayoutDocEntry::FileNamespaces:
1074 case LayoutDocEntry::FileConstantGroups:
1075 case LayoutDocEntry::FileIncludes:
1076 case LayoutDocEntry::FileIncludeGraph:
1077 case LayoutDocEntry::FileIncludedByGraph:
1078 case LayoutDocEntry::FileSourceLink:
1079 case LayoutDocEntry::FileInlineClasses:
1080 case LayoutDocEntry::DirSubDirs:
1081 case LayoutDocEntry::DirFiles:
1082 case LayoutDocEntry::DirGraph:
1083 err("Internal inconsistency: member %d should not be part of "
1084 "LayoutDocManager::Group entry list\n",lde->kind());
1089 //---------------------------------------- end flexible part -------------------------------
1093 ol.popGeneratorState();
1095 if (!Config_getString("GENERATE_TAGFILE").isEmpty())
1097 writeDocAnchorsToTagFile();
1098 Doxygen::tagFile << " </compound>" << endl;
1101 if (Config_getBool("SEPARATE_MEMBER_PAGES"))
1103 allMemberList->sort();
1104 writeMemberPages(ol);
1109 void GroupDef::writeMemberPages(OutputList &ol)
1111 ol.pushGeneratorState();
1112 ol.disableAllBut(OutputGenerator::Html);
1114 QListIterator<MemberList> mli(m_memberLists);
1116 for (mli.toFirst();(ml=mli.current());++mli)
1118 if (ml->listType()&MemberListType_documentationLists)
1120 ml->writeDocumentationPage(ol,name(),this);
1124 ol.popGeneratorState();
1127 void GroupDef::writeQuickMemberLinks(OutputList &ol,MemberDef *currentMd) const
1129 static bool createSubDirs=Config_getBool("CREATE_SUBDIRS");
1131 ol.writeString(" <div class=\"navtab\">\n");
1132 ol.writeString(" <table>\n");
1134 MemberListIterator mli(*allMemberList);
1136 for (mli.toFirst();(md=mli.current());++mli)
1138 if (md->getGroupDef()==this && md->isLinkable() && !md->isEnumValue())
1140 ol.writeString(" <tr><td class=\"navtab\">");
1141 if (md->isLinkableInProject())
1143 if (md==currentMd) // selected item => highlight
1145 ol.writeString("<a class=\"qindexHL\" ");
1149 ol.writeString("<a class=\"qindex\" ");
1151 ol.writeString("href=\"");
1152 if (createSubDirs) ol.writeString("../../");
1153 ol.writeString(md->getOutputFileBase()+Doxygen::htmlFileExtension+"#"+md->anchor());
1154 ol.writeString("\">");
1155 ol.writeString(convertToHtml(md->localName()));
1156 ol.writeString("</a>");
1158 ol.writeString("</td></tr>\n");
1162 ol.writeString(" </table>\n");
1163 ol.writeString(" </div>\n");
1168 //---- helper functions ------------------------------------------------------
1170 void addClassToGroups(Entry *root,ClassDef *cd)
1172 QListIterator<Grouping> gli(*root->groups);
1174 for (;(g=gli.current());++gli)
1177 if (!g->groupname.isEmpty() && (gd=Doxygen::groupSDict->find(g->groupname)))
1179 if (gd->addClass(cd))
1181 cd->makePartOfGroup(gd);
1183 //printf("Compound %s: in group %s\n",cd->name().data(),gd->groupTitle());
1188 void addNamespaceToGroups(Entry *root,NamespaceDef *nd)
1190 //printf("root->groups->count()=%d\n",root->groups->count());
1191 QListIterator<Grouping> gli(*root->groups);
1193 for (;(g=gli.current());++gli)
1196 //printf("group `%s'\n",s->data());
1197 if (!g->groupname.isEmpty() && (gd=Doxygen::groupSDict->find(g->groupname)))
1199 if (gd->addNamespace(nd)) nd->makePartOfGroup(gd);
1200 //printf("Namespace %s: in group %s\n",nd->name().data(),s->data());
1205 void addDirToGroups(Entry *root,DirDef *dd)
1207 //printf("*** root->groups->count()=%d\n",root->groups->count());
1208 QListIterator<Grouping> gli(*root->groups);
1210 for (;(g=gli.current());++gli)
1213 //printf("group `%s'\n",g->groupname.data());
1214 if (!g->groupname.isEmpty() && (gd=Doxygen::groupSDict->find(g->groupname)))
1217 dd->makePartOfGroup(gd);
1218 //printf("Dir %s: in group %s\n",dd->name().data(),g->groupname.data());
1223 void addGroupToGroups(Entry *root,GroupDef *subGroup)
1225 //printf("addGroupToGroups for %s groups=%d\n",root->name.data(),
1226 // root->groups?root->groups->count():-1);
1227 QListIterator<Grouping> gli(*root->groups);
1229 for (;(g=gli.current());++gli)
1232 if (!g->groupname.isEmpty() && (gd=Doxygen::groupSDict->find(g->groupname)) &&
1233 !gd->containsGroup(subGroup) )
1235 gd->addGroup(subGroup);
1236 subGroup->makePartOfGroup(gd);
1238 else if (gd==subGroup)
1240 warn(root->fileName,root->startLine,"Trying to add group %s to itself!",
1246 /*! Add a member to the group with the highest priority */
1247 void addMemberToGroups(Entry *root,MemberDef *md)
1249 //printf("addMemberToGroups: Root %p = %s, md %p=%s groups=%d\n",
1250 // root, root->name.data(), md, md->name().data(), root->groups->count() );
1251 QListIterator<Grouping> gli(*root->groups);
1254 // Search entry's group list for group with highest pri.
1255 Grouping::GroupPri_t pri = Grouping::GROUPING_LOWEST;
1257 for (;(g=gli.current());++gli)
1260 if (!g->groupname.isEmpty() &&
1261 (gd=Doxygen::groupSDict->find(g->groupname)) &&
1264 if (fgd && gd!=fgd && g->pri==pri)
1266 warn(root->fileName.data(), root->startLine,
1267 "Member %s found in multiple %s groups! "
1268 "The member will be put in group %s, and not in group %s",
1269 md->name().data(), Grouping::getGroupPriName( pri ),
1270 gd->name().data(), fgd->name().data()
1278 //printf("fgd=%p\n",fgd);
1280 // put member into group defined by this entry?
1283 GroupDef *mgd = md->getGroupDef();
1284 //printf("mgd=%p\n",mgd);
1285 bool insertit = FALSE;
1292 bool moveit = FALSE;
1294 // move member from one group to another if
1295 // - the new one has a higher priority
1296 // - the new entry has the same priority, but with docs where the old one had no docs
1297 if (md->getGroupPri()<pri)
1303 if (md->getGroupPri()==pri)
1305 if (!root->doc.isEmpty() && !md->getGroupHasDocs())
1309 else if (!root->doc.isEmpty() && md->getGroupHasDocs())
1311 warn(md->getGroupFileName(),md->getGroupStartLine(),
1312 "Member documentation for %s found several times in %s groups!\n"
1313 "%s:%d: The member will remain in group %s, and won't be put into group %s",
1314 md->name().data(), Grouping::getGroupPriName( pri ),
1315 root->fileName.data(), root->startLine,
1325 //printf("removeMember\n");
1326 mgd->removeMember(md);
1333 //printf("insertMember found at %s line %d: %s: related %s\n",
1334 // md->getDefFileName().data(),md->getDefLine(),
1335 // md->name().data(),root->relates.data());
1336 bool success = fgd->insertMember(md);
1339 //printf("insertMember successful\n");
1340 md->setGroupDef(fgd,pri,root->fileName,root->startLine,
1341 !root->doc.isEmpty());
1342 ClassDef *cd = md->getClassDefOfAnonymousType();
1345 cd->setGroupDefForAllMembers(fgd,pri,root->fileName,root->startLine,root->doc.length() != 0);
1353 void addExampleToGroups(Entry *root,PageDef *eg)
1355 QListIterator<Grouping> gli(*root->groups);
1357 for (;(g=gli.current());++gli)
1360 if (!g->groupname.isEmpty() && (gd=Doxygen::groupSDict->find(g->groupname)))
1363 eg->makePartOfGroup(gd);
1364 //printf("Example %s: in group %s\n",eg->name().data(),s->data());
1369 QCString GroupDef::getOutputFileBase() const
1377 return convertNameToFile(fileName);
1381 void GroupDef::addListReferences()
1384 QList<ListItemInfo> *xrefItems = xrefListItems();
1385 addRefItem(xrefItems,
1386 getOutputFileBase(),
1387 theTranslator->trGroup(TRUE,TRUE),
1388 getOutputFileBase(),name(),
1392 MemberGroupSDict::Iterator mgli(*memberGroupSDict);
1394 for (;(mg=mgli.current());++mgli)
1396 mg->addListReferences(this);
1398 QListIterator<MemberList> mli(m_memberLists);
1400 for (mli.toFirst();(ml=mli.current());++mli)
1402 if (ml->listType()&MemberListType_documentationLists)
1404 ml->addListReferences(this);
1409 MemberList *GroupDef::createMemberList(MemberListType lt)
1411 m_memberLists.setAutoDelete(TRUE);
1412 QListIterator<MemberList> mli(m_memberLists);
1414 for (mli.toFirst();(ml=mli.current());++mli)
1416 if (ml->listType()==lt)
1421 // not found, create a new member list
1422 ml = new MemberList(lt);
1423 m_memberLists.append(ml);
1424 ml->setInGroup(TRUE);
1428 void GroupDef::addMemberToList(MemberListType lt,MemberDef *md)
1430 static bool sortBriefDocs = Config_getBool("SORT_BRIEF_DOCS");
1431 static bool sortMemberDocs = Config_getBool("SORT_MEMBER_DOCS");
1432 MemberList *ml = createMemberList(lt);
1433 ml->setNeedsSorting(
1434 ((ml->listType()&MemberListType_declarationLists) && sortBriefDocs) ||
1435 ((ml->listType()&MemberListType_documentationLists) && sortMemberDocs));
1439 void GroupDef::sortMemberLists()
1441 QListIterator<MemberList> mli(m_memberLists);
1443 for (;(ml=mli.current());++mli)
1445 if (ml->needsSorting()) { ml->sort(); ml->setNeedsSorting(FALSE); }
1449 MemberList *GroupDef::getMemberList(MemberListType lt) const
1451 QListIterator<MemberList> mli(m_memberLists);
1453 for (;(ml=mli.current());++mli)
1455 if (ml->listType()==lt)
1463 void GroupDef::writeMemberDeclarations(OutputList &ol,MemberListType lt,const QCString &title)
1465 static bool optimizeVhdl = Config_getBool("OPTIMIZE_OUTPUT_VHDL");
1467 MemberList * ml = getMemberList(lt);
1468 if (optimizeVhdl && ml)
1470 VhdlDocGen::writeVhdlDeclarations(ml,ol,this,0,0,0);
1475 ml->writeDeclarations(ol,0,0,0,this,title,0,definitionType());
1479 void GroupDef::writeMemberDocumentation(OutputList &ol,MemberListType lt,const QCString &title)
1481 MemberList * ml = getMemberList(lt);
1482 if (ml) ml->writeDocumentation(ol,name(),this,title);
1485 void GroupDef::removeMemberFromList(MemberListType lt,MemberDef *md)
1487 MemberList *ml = getMemberList(lt);
1488 if (ml) ml->remove(md);
1491 void GroupDef::sortSubGroups()
1496 bool GroupDef::isLinkableInProject() const
1498 return !isReference() && isLinkable();
1501 bool GroupDef::isLinkable() const
1503 return hasUserDocumentation();
1506 // let the "programming language" for a group depend on what is inserted into it.
1507 // First item that has an associated languages determines the language for the whole group.
1508 void GroupDef::updateLanguage(const Definition *d)
1510 if (getLanguage()==SrcLangExt_Unknown && d->getLanguage()!=SrcLangExt_Unknown)
1512 setLanguage(d->getLanguage());
1516 bool GroupDef::hasDetailedDescription() const
1518 static bool repeatBrief = Config_getBool("REPEAT_BRIEF");
1519 return ((!briefDescription().isEmpty() && repeatBrief) ||
1520 !documentation().isEmpty());