Doxygen
Loading...
Searching...
No Matches
dirdef.cpp
Go to the documentation of this file.
1/******************************************************************************
2 *
3 * Copyright (C) 1997-2020 by Dimitri van Heesch.
4 *
5 * Permission to use, copy, modify, and distribute this software and its
6 * documentation under the terms of the GNU General Public License is hereby
7 * granted. No representations are made about the suitability of this software
8 * for any purpose. It is provided "as is" without express or implied warranty.
9 * See the GNU General Public License for more details.
10 *
11 * Documents produced by Doxygen are derivative works derived from the
12 * input used in their production; they are not affected by this license.
13 *
14 */
15
16#include <algorithm>
17
18#include "dirdef.h"
19#include "md5.h"
20#include "filename.h"
21#include "doxygen.h"
22#include "util.h"
23#include "outputlist.h"
24#include "language.h"
25#include "message.h"
26#include "dot.h"
27#include "dotdirdeps.h"
28#include "layout.h"
29#include "config.h"
30#include "docparser.h"
31#include "definitionimpl.h"
32#include "filedef.h"
33#include "trace.h"
34
35//----------------------------------------------------------------------
36
37class DirDefImpl : public DefinitionMixin<DirDef>
38{
39 public:
40 DirDefImpl(const QCString &path);
41 ~DirDefImpl() override;
43
44 DefType definitionType() const override { return TypeDir; }
46 QCString getOutputFileBase() const override;
47 QCString anchor() const override { return QCString(); }
48 bool isLinkableInProject() const override;
49 bool isLinkable() const override;
50 QCString displayName(bool=TRUE) const override { return m_dispName; }
51 const QCString shortName() const override { return m_shortName; }
52 void addSubDir(DirDef *subdir) override;
53 const FileList &getFiles() const override { return m_fileList; }
54 void addFile(FileDef *fd) override;
55 const DirList &subDirs() const override { return m_subdirs; }
56 bool hasSubdirs() const override { return !m_subdirs.empty(); }
57 int level() const override { return m_level; }
58 DirDef *parent() const override { return m_parent; }
59 int dirIndex() const override { return m_dirIndex; }
60 const UsedDirLinkedMap &usedDirs() const override { return m_usedDirs; }
61 bool isParentOf(const DirDef *dir) const override;
62 bool depGraphIsTrivial() const override;
63 QCString shortTitle() const override;
64 bool hasDetailedDescription() const override;
65 void writeDocumentation(OutputList &ol) override;
66 void writePageNavigation(OutputList &ol) const override;
67 void writeTagFile(TextStream &t) override;
68 void setDiskName(const QCString &name) override { m_diskName = name; }
69 void sort() override;
70 void setParent(DirDef *parent) override;
71 void setDirIndex(int index) override;
72 void setLevel() override;
73 void addUsesDependency(const DirDef *usedDir,const FileDef *srcFd,
74 const FileDef *dstFd,bool srcDirect, bool dstDirect) override;
75 void computeDependencies() override;
76 void findSectionsInDocumentation() override;
77
78 bool hasDirectoryGraph() const override;
79 void overrideDirectoryGraph(bool e) override;
80
81 public:
82 static DirDef *mergeDirectoryInTree(const QCString &path);
83
84 private:
85
86 void writeDetailedDescription(OutputList &ol,const QCString &title);
90 void writeFileList(OutputList &ol);
93
94 static DirDef *createNewDir(const QCString &path);
95 static bool matchPath(const QCString &path,const StringVector &l);
96
101 FileList m_fileList; // list of files in the group
102 int m_dirIndex = -1;
107};
108
110{
111 return new DirDefImpl(path);
112}
113
114
115//----------------------------------------------------------------------
116// method implementation
117
119{
120 bool fullPathNames = Config_getBool(FULL_PATH_NAMES);
121 // get display name (stripping the paths mentioned in STRIP_FROM_PATH)
122 // get short name (last part of path)
123 m_shortName = path;
124 m_diskName = path;
125 if (m_shortName.at(m_shortName.length()-1)=='/')
126 { // strip trailing /
127 m_shortName = m_shortName.left(m_shortName.length()-1);
128 }
129 int pi=m_shortName.findRev('/');
130 if (pi!=-1)
131 { // remove everything till the last /
132 m_shortName = m_shortName.mid(pi+1);
133 }
135 m_dispName = fullPathNames ? stripFromPath(path) : m_shortName;
136 if (m_dispName.length()>0 && m_dispName.at(m_dispName.length()-1)=='/')
137 { // strip trailing /
138 m_dispName = m_dispName.left(m_dispName.length()-1);
139 }
140
141 m_level=-1;
142 m_parent=nullptr;
143 m_hasDirectoryGraph=Config_getBool(DIRECTORY_GRAPH);
144
145}
146
150
152{
153 return !isReference() && hasDocumentation();
154}
155
157{
158 return isReference() || isLinkableInProject();
159}
160
162{
163 m_subdirs.push_back(subdir);
164 subdir->setOuterScope(this);
165 subdir->setParent(this);
166}
167
169{
170 m_parent=p;
171}
172
174{
175 m_dirIndex=index;
176}
177
179{
180 m_fileList.push_back(fd);
181 fd->setDirDef(this);
182}
183
185{
186 std::stable_sort(m_subdirs.begin(), m_subdirs.end(), compareDirDefs);
187 std::stable_sort(m_fileList.begin(), m_fileList.end(), compareFileDefs);
188}
189
190static QCString encodeDirName(const QCString &anchor)
191{
192 AUTO_TRACE();
193 // convert to md5 hash
194 uint8_t md5_sig[16];
195 char sigStr[33];
196 MD5Buffer(anchor.data(),static_cast<unsigned int>(anchor.length()),md5_sig);
197 MD5SigToString(md5_sig,sigStr);
198 AUTO_TRACE_EXIT("result={}",sigStr);
199 return sigStr;
200
201 // old algorithm
202// QCString result;
203
204// int l = anchor.length(),i;
205// for (i=0;i<l;i++)
206// {
207// char c = anchor.at(i);
208// if ((c>='a' && c<='z') || (c>='A' && c<='Z') || (c>='0' && c<='9'))
209// {
210// result+=c;
211// }
212// else
213// {
214// static char hexStr[]="0123456789ABCDEF";
215// char escChar[]={ '_', 0, 0, 0 };
216// escChar[1]=hexStr[c>>4];
217// escChar[2]=hexStr[c&0xf];
218// result+=escChar;
219// }
220// }
221// return result;
222}
223
225{
226 QCString dir = "dir_"+encodeDirName(m_diskName);
227 AUTO_TRACE("diskName={} result={}",m_diskName,dir);
228 return dir;
229}
230
232{
233 AUTO_TRACE();
234 if ((!briefDescription().isEmpty() && Config_getBool(REPEAT_BRIEF)) ||
235 !documentation().isEmpty())
236 {
239 ol.writeRuler();
243 ol.writeAnchor(QCString(),"details");
245 ol.startGroupHeader("details");
246 ol.parseText(title);
247 ol.endGroupHeader();
248
249 // repeat brief description
250 if (!briefDescription().isEmpty() && Config_getBool(REPEAT_BRIEF))
251 {
254 }
255 // separator between brief and details
256 if (!briefDescription().isEmpty() && Config_getBool(REPEAT_BRIEF) &&
257 !documentation().isEmpty())
258 {
262 ol.enableAll();
265 ol.writeString("\n\n");
267 }
268
269 // write documentation
270 if (!documentation().isEmpty())
271 {
272 ol.generateDoc(docFile(),docLine(),this,nullptr,documentation()+"\n",TRUE,FALSE,
274 }
275 }
276}
277
279{
280 AUTO_TRACE();
282 {
283 auto parser { createDocParser() };
284 auto ast { validatingParseDoc(
285 *parser.get(), briefFile(),briefLine(),this,nullptr,briefDescription(),TRUE,FALSE,
286 QCString(),FALSE,FALSE) };
287 if (!ast->isEmpty())
288 {
289 ol.startParagraph();
292 ol.writeString(" - ");
294 ol.writeDoc(ast.get(),this,nullptr);
297 ol.writeString(" \n");
299
300 if (Config_getBool(REPEAT_BRIEF) ||
301 !documentation().isEmpty()
302 )
303 {
305 ol.startTextLink(QCString(),"details");
306 ol.parseText(theTranslator->trMore());
307 ol.endTextLink();
308 }
310
311 ol.endParagraph();
312 }
313 }
314 ol.writeSynopsis();
315}
316
321
323{
324 // write graph dependency graph
325 if (Config_getBool(HAVE_DOT) && m_hasDirectoryGraph /*&& Config_getBool(DIRECTORY_GRAPH)*/)
326 {
327 DotDirDeps dirDep(this);
328 if (!dirDep.isTrivial())
329 {
330 msg("Generating dependency graph for directory {}\n",displayName());
332 //ol.startParagraph();
333 ol.startDirDepGraph();
334 ol.parseText(theTranslator->trDirDepGraph(shortName()));
335 ol.endDirDepGraph(dirDep);
336 //ol.endParagraph();
337 ol.enableAll();
338 }
339 }
340}
341
343{
344 AUTO_TRACE();
345 int numSubdirs = 0;
346 for(const auto dd : m_subdirs)
347 {
348 if (dd->hasDocumentation() || !dd->getFiles().empty())
349 {
350 numSubdirs++;
351 }
352 }
353
354 AUTO_TRACE_ADD("numSubdirs={}",numSubdirs);
355 // write subdir list
356 if (numSubdirs>0)
357 {
358 ol.startMemberHeader("subdirs");
359 ol.parseText(theTranslator->trDir(TRUE,FALSE));
360 ol.endMemberHeader();
361 ol.startMemberList();
362 for(const auto dd : m_subdirs)
363 {
364 if (dd->hasDocumentation() || !dd->getFiles().empty())
365 {
367 QCString anc=dd->anchor();
368 if (anc.isEmpty()) anc=dd->shortName(); else anc.prepend(dd->shortName()+"_");
370 {
373 ol.writeString("<span class=\"iconfolder\"><div class=\"folder-icon\"></div></span>");
374 ol.enableAll();
376 ol.parseText(theTranslator->trDir(FALSE,TRUE)+" ");
378 }
380 ol.writeObjectLink(dd->getReference(),dd->getOutputFileBase(),QCString(),dd->shortName());
382 if (!dd->briefDescription().isEmpty() && Config_getBool(BRIEF_MEMBER_DESC))
383 {
384 ol.startMemberDescription(dd->getOutputFileBase());
385 ol.generateDoc(briefFile(),briefLine(),dd,nullptr,dd->briefDescription(),
386 FALSE, // indexWords
387 FALSE, // isExample
388 QCString(), // exampleName
389 TRUE, // single line
390 TRUE // link from index
391 );
393 }
394 ol.endMemberDeclaration(dd->anchor(),QCString());
395 }
396 }
397
398 ol.endMemberList();
399 }
400}
401
403{
404 AUTO_TRACE();
405 int numFiles = 0;
406 for (const auto &fd : m_fileList)
407 {
408 bool genSourceFile=false;
409 if (fileVisibleInIndex(fd,genSourceFile))
410 {
411 numFiles++;
412 }
413 else if (genSourceFile)
414 {
415 numFiles++;
416 }
417 }
418
419 AUTO_TRACE_ADD("numFiles={}",numFiles);
420 // write file list
421 if (numFiles>0)
422 {
423 ol.startMemberHeader("files");
424 ol.parseText(theTranslator->trFile(TRUE,FALSE));
425 ol.endMemberHeader();
426 ol.startMemberList();
427 for (const auto &fd : m_fileList)
428 {
429 bool src = false;
430 bool doc = fileVisibleInIndex(fd,src);
431 if (doc || src)
432 {
434 QCString anc = fd->anchor();
435 if (anc.isEmpty()) anc=fd->displayName(); else anc.prepend(fd->displayName()+"_");
437 {
440 bool genSrc = fd->generateSourceFile();
441 if (genSrc)
442 {
443 ol.startTextLink(fd->includeName(),QCString());
444 }
445 ol.writeString("<span class=\"icondoc\"><div class=\"doc-icon\"></div></span>");
446 if (genSrc)
447 {
448 ol.endTextLink();
449 }
450 ol.enableAll();
452 ol.docify(theTranslator->trFile(FALSE,TRUE)+" ");
454 }
456 if (fd->isLinkable())
457 {
458 ol.writeObjectLink(fd->getReference(),fd->getOutputFileBase(),QCString(),fd->displayName());
459 }
460 else
461 {
462 ol.startBold();
463 ol.docify(fd->displayName());
464 ol.endBold();
465 }
467 if (!fd->briefDescription().isEmpty() && Config_getBool(BRIEF_MEMBER_DESC))
468 {
469 ol.startMemberDescription(fd->getOutputFileBase());
470 ol.generateDoc(briefFile(),briefLine(),fd,nullptr,fd->briefDescription(),
471 FALSE, // indexWords
472 FALSE, // isExample
473 QCString(), // exampleName
474 TRUE, // single line
475 TRUE // link from index
476 );
478 }
479 ol.endMemberDeclaration(fd->anchor(),QCString());
480 }
481 }
482 ol.endMemberList();
483 }
484}
485
490
495
497{
498 if (Config_getBool(HIDE_COMPOUND_REFERENCE))
499 {
500 return m_shortName;
501 }
502 else
503 {
504 return theTranslator->trDirReference(m_shortName);
505 }
506}
507
509{
510 bool repeatBrief = Config_getBool(REPEAT_BRIEF);
511 return (!briefDescription().isEmpty() && repeatBrief) || !documentation().isEmpty();
512}
513
515{
516 tagFile << " <compound kind=\"dir\">\n";
517 tagFile << " <name>" << convertToXML(displayName()) << "</name>\n";
518 tagFile << " <path>" << convertToXML(stripFromPath(name())) << "</path>\n";
521 tagFile << " <filename>" << fn << "</filename>\n";
522 for (const auto &lde : LayoutDocManager::instance().docEntries(LayoutDocManager::Directory))
523 {
524 switch (lde->kind())
525 {
526 case LayoutDocEntry::DirSubDirs:
527 {
528 if (m_subdirs.size()>0)
529 {
530 for(const auto dd : m_subdirs)
531 {
532 tagFile << " <dir>" << convertToXML(dd->displayName()) << "</dir>\n";
533 }
534 }
535 }
536 break;
537 case LayoutDocEntry::DirFiles:
538 {
539 for (const auto &fd : m_fileList)
540 {
541 tagFile << " <file>" << convertToXML(fd->name()) << "</file>\n";
542 }
543 }
544 break;
545 default:
546 break;
547 }
548 }
550 tagFile << " </compound>\n";
551}
552
554{
555 bool generateTreeView = Config_getBool(GENERATE_TREEVIEW);
557
558 QCString title;
559 if (Config_getBool(HIDE_COMPOUND_REFERENCE))
560 {
561 title=m_dispName;
562 }
563 else
564 {
565 title=theTranslator->trDirReference(m_dispName);
566 }
567 AUTO_TRACE("title={}",title);
568 startFile(ol,getOutputFileBase(),name(),title,HighlightedItem::Files,!generateTreeView);
569
570 if (!generateTreeView)
571 {
572 // write navigation path
574 ol.endQuickIndices();
575 }
576
580 ol.parseText(shortTitle());
581 ol.enableAll();
583 ol.parseText(title);
585 endTitle(ol,getOutputFileBase(),title);
586 ol.startContents();
587
588 //---------------------------------------- start flexible part -------------------------------
589
590 SrcLangExt lang = getLanguage();
591 for (const auto &lde : LayoutDocManager::instance().docEntries(LayoutDocManager::Directory))
592 {
593 switch (lde->kind())
594 {
595 case LayoutDocEntry::BriefDesc:
597 break;
598 case LayoutDocEntry::DirGraph:
600 break;
601 case LayoutDocEntry::MemberDeclStart:
603 break;
604 case LayoutDocEntry::DirSubDirs:
605 writeSubDirList(ol);
606 break;
607 case LayoutDocEntry::DirFiles:
608 writeFileList(ol);
609 break;
610 case LayoutDocEntry::MemberDeclEnd:
612 break;
613 case LayoutDocEntry::DetailedDesc:
614 {
615 const LayoutDocEntrySection *ls = dynamic_cast<const LayoutDocEntrySection*>(lde.get());
616 if (ls)
617 {
618 writeDetailedDescription(ol,ls->title(lang));
619 }
620 }
621 break;
622 case LayoutDocEntry::ClassIncludes:
623 case LayoutDocEntry::ClassInlineClasses:
624 case LayoutDocEntry::ClassInheritanceGraph:
625 case LayoutDocEntry::ClassNestedClasses:
626 case LayoutDocEntry::ClassCollaborationGraph:
627 case LayoutDocEntry::ClassAllMembersLink:
628 case LayoutDocEntry::ClassUsedFiles:
629 case LayoutDocEntry::NamespaceNestedNamespaces:
630 case LayoutDocEntry::NamespaceNestedConstantGroups:
631 case LayoutDocEntry::NamespaceClasses:
632 case LayoutDocEntry::NamespaceConcepts:
633 case LayoutDocEntry::NamespaceInterfaces:
634 case LayoutDocEntry::NamespaceStructs:
635 case LayoutDocEntry::NamespaceExceptions:
636 case LayoutDocEntry::NamespaceInlineClasses:
637 case LayoutDocEntry::ConceptDefinition:
638 case LayoutDocEntry::FileClasses:
639 case LayoutDocEntry::FileConcepts:
640 case LayoutDocEntry::FileInterfaces:
641 case LayoutDocEntry::FileStructs:
642 case LayoutDocEntry::FileExceptions:
643 case LayoutDocEntry::FileNamespaces:
644 case LayoutDocEntry::FileConstantGroups:
645 case LayoutDocEntry::FileIncludes:
646 case LayoutDocEntry::FileIncludeGraph:
647 case LayoutDocEntry::FileIncludedByGraph:
648 case LayoutDocEntry::FileSourceLink:
649 case LayoutDocEntry::FileInlineClasses:
650 case LayoutDocEntry::GroupClasses:
651 case LayoutDocEntry::GroupConcepts:
652 case LayoutDocEntry::GroupModules:
653 case LayoutDocEntry::GroupInlineClasses:
654 case LayoutDocEntry::GroupNamespaces:
655 case LayoutDocEntry::GroupDirs:
656 case LayoutDocEntry::GroupNestedGroups:
657 case LayoutDocEntry::GroupFiles:
658 case LayoutDocEntry::GroupGraph:
659 case LayoutDocEntry::GroupPageDocs:
660 case LayoutDocEntry::ModuleExports:
661 case LayoutDocEntry::ModuleClasses:
662 case LayoutDocEntry::ModuleConcepts:
663 case LayoutDocEntry::ModuleUsedFiles:
664 case LayoutDocEntry::AuthorSection:
665 case LayoutDocEntry::MemberGroups:
666 case LayoutDocEntry::MemberDecl:
667 case LayoutDocEntry::MemberDef:
668 case LayoutDocEntry::MemberDefStart:
669 case LayoutDocEntry::MemberDefEnd:
670 err("Internal inconsistency: member '{}' should not be part of "
671 "LayoutDocManager::Directory entry list\n",qPrint(lde->entryToString()));
672 break;
673 }
674 }
675
676 //---------------------------------------- end flexible part -------------------------------
677
678 ol.endContents();
679
680 endFileWithNavPath(ol,this);
681
683}
684
686{
687 if (m_level==-1) // level not set before
688 {
689 DirDef *p = parent();
690 if (p)
691 {
692 p->setLevel();
693 m_level = p->level()+1;
694 }
695 else
696 {
697 m_level = 0;
698 }
699 }
700}
701
702/** Add as "uses" dependency between \a this dir and \a dir,
703 * that was caused by a dependency on file \a fd.
704 * srcDirect and dstDirect indicate if it is a direct dependencies (true) or if
705 * the dependencies was indirect (e.g. a parent dir that has a child dir that has the dependencies)
706 */
707void DirDefImpl::addUsesDependency(const DirDef *dir,const FileDef *srcFd,
708 const FileDef *dstFd,bool srcDirect, bool dstDirect)
709{
710 if (this==dir) return; // do not add self-dependencies
711 AUTO_TRACE("add dependency {}->{} due to {}->{}",
712 qPrint(shortName()),
713 qPrint(dir->shortName()),
714 qPrint(srcFd->name()),
715 qPrint(dstFd->name()));
716
717 // levels match => add direct dependency
718 bool added=FALSE;
719 UsedDir *usedDir = m_usedDirs.find(dir->getOutputFileBase());
720 if (usedDir) // dir dependency already present
721 {
722 const FilePair *usedPair = usedDir->findFilePair(FilePair::key(srcFd,dstFd));
723 if (usedPair==nullptr) // new file dependency
724 {
725 AUTO_TRACE_ADD("{} => {} new file dependency",srcFd->name(),dstFd->name());
726 usedDir->addFileDep(srcFd,dstFd, srcDirect, dstDirect);
727 added=TRUE;
728 }
729 else
730 {
731 // dir & file dependency already added
732 }
733 }
734 else // new directory dependency
735 {
736 AUTO_TRACE_ADD("{} => {} new file dependency",srcFd->name(),dstFd->name());
737 auto newUsedDir = std::make_unique<UsedDir>(dir);
738 newUsedDir->addFileDep(srcFd,dstFd, srcDirect, dstDirect);
739 m_usedDirs.add(dir->getOutputFileBase(),std::move(newUsedDir));
740 added=TRUE;
741 }
742 if (added)
743 {
744 if (dir->parent())
745 {
746 // add relation to parent of used dir
748 srcFd,
749 dstFd,
750 srcDirect,
751 false); // indirect dependency on dest dir
752 }
753 if (parent())
754 {
755 // add relation for the parent of this dir as well
757 srcFd,
758 dstFd,
759 false, // indirect dependency from source dir
760 dstDirect);
761 }
762 }
763}
764
770
771/** Computes the dependencies between directories
772 */
774{
775 AUTO_TRACE();
776 for (const auto &fd : m_fileList)
777 {
778 AUTO_TRACE_ADD("dir={} file={}",shortName(),fd->name());
779 for (const auto &ii : fd->includeFileList())
780 {
781 AUTO_TRACE_ADD("#include {}",ii.includeName);
782 if (ii.fileDef && ii.fileDef->isLinkable()) // linkable file
783 {
784 DirDef *usedDir = ii.fileDef->getDirDef();
785 if (usedDir)
786 {
787 // add dependency: thisDir->usedDir
788 AUTO_TRACE_ADD("add dependency {}->{}",name(),usedDir->name());
789 addUsesDependency(usedDir,fd,ii.fileDef,true,true);
790 }
791 }
792 }
793 }
794
795 std::stable_sort(m_usedDirs.begin(),m_usedDirs.end(),
796 [](const auto &u1,const auto &u2)
797 { return qstricmp_sort(u1->dir()->getOutputFileBase(),u2->dir()->getOutputFileBase())<0; });
798
799 for (const auto& usedDirectory : m_usedDirs)
800 {
801 usedDirectory->sort();
802 }
803}
804
805bool DirDefImpl::isParentOf(const DirDef *dir) const
806{
807 if (dir->parent()==this) // this is a parent of dir
808 return TRUE;
809 else if (dir->parent()) // repeat for the parent of dir
810 return isParentOf(dir->parent());
811 else
812 return FALSE;
813}
814
816{
817 return m_usedDirs.empty() && m_parent==nullptr;
818}
819
820//----------------------------------------------------------------------
821
823 m_dir(dir)
824{
825}
826
827void UsedDir::addFileDep(const FileDef *srcFd,const FileDef *dstFd, bool srcDirect, bool dstDirect)
828{
829 m_filePairs.add(FilePair::key(srcFd,dstFd),std::make_unique<FilePair>(srcFd,dstFd));
830 m_hasDirectDeps = m_hasDirectDeps || (srcDirect && dstDirect);
833}
834
836{
837 std::stable_sort(m_filePairs.begin(),
838 m_filePairs.end(),
839 [](const auto &left,const auto &right)
840 {
841 int orderHi = qstricmp_sort(left->source()->name(),right->source()->name());
842 if (orderHi!=0) return orderHi<0;
843 int orderLo = qstricmp_sort(left->destination()->name(),right->destination()->name());
844 return orderLo<0;
845 });
846}
847
849{
850 return m_filePairs.find(name);
851}
852
854{
855 AUTO_TRACE();
856 ASSERT(path!=nullptr);
857 DirDef *dir = Doxygen::dirLinkedMap->find(path);
858 if (dir==nullptr) // new dir
859 {
860 dir = Doxygen::dirLinkedMap->add(path,
861 std::unique_ptr<DirDef>(
862 createDirDef(path)));
863 AUTO_TRACE_ADD("Adding new dir {} shortName {}",path,dir->shortName());
864 }
865 return dir;
866}
867
869{
870 for (const auto &s : l)
871 {
872 std::string prefix = s.substr(0,path.length());
873 if (qstricmp_sort(prefix.c_str(),path)==0) // case insensitive compare
874 {
875 return TRUE;
876 }
877 }
878 return FALSE;
879}
880
881/*! strip part of \a path if it matches
882 * one of the paths in the Config_getList(STRIP_FROM_PATH) list
883 */
885{
886 AUTO_TRACE("path={}",path);
887 int p=0,i=0;
888 DirDef *dir=nullptr;
889 while ((i=path.find('/',p))!=-1)
890 {
891 QCString part=path.left(i+1);
892 if (!matchPath(part,Config_getList(STRIP_FROM_PATH)) && (part!="/" && part!="//" && part!="//?/"))
893 {
895 }
896 p=i+1;
897 }
898 return dir;
899}
900
905
907{
908 return m_hasDirectoryGraph;
909}
910
911//----------------------------------------------------------------------
912
913QCString FilePair::key(const FileDef *srcFd,const FileDef *dstFd)
914{
915 return srcFd->getOutputFileBase()+";"+dstFd->getOutputFileBase();
916}
917
918//----------------------------------------------------------------------
919
920static void writePartialDirPath(OutputList &ol,const DirDef *root,const DirDef *target)
921{
922 if (target->parent()!=root)
923 {
924 writePartialDirPath(ol,root,target->parent());
925 ol.writeString("&#160;/&#160;");
926 }
927 ol.writeObjectLink(target->getReference(),target->getOutputFileBase(),QCString(),target->shortName());
928}
929
930static void writePartialFilePath(OutputList &ol,const DirDef *root,const FileDef *fd)
931{
932 if (fd->getDirDef() && fd->getDirDef()!=root)
933 {
934 writePartialDirPath(ol,root,fd->getDirDef());
935 ol.writeString("&#160;/&#160;");
936 }
937 if (fd->isLinkable())
938 {
940 }
941 else
942 {
943 ol.startBold();
944 ol.docify(fd->name());
945 ol.endBold();
946 }
947}
948
950{
951 bool generateTreeView = Config_getBool(GENERATE_TREEVIEW);
954
955 QCString shortTitle=theTranslator->trDirRelation(
956 (m_src->shortName()+" &rarr; "+m_dst->dir()->shortName()));
957 QCString title=theTranslator->trDirRelation(
958 (m_src->displayName()+" -> "+m_dst->dir()->shortName()));
959 AUTO_TRACE("title={}",title);
961 title,HighlightedItem::None,!generateTreeView,m_src->getOutputFileBase());
962
963 if (!generateTreeView)
964 {
965 // write navigation path
966 m_src->writeNavigationPath(ol);
967 ol.endQuickIndices();
968 }
969 ol.startContents();
970
971 ol.writeString("<h3>"+shortTitle+"</h3>");
972 ol.writeString("<table class=\"dirtab\">");
973 ol.writeString("<tr class=\"dirtab\">");
974 ol.writeString("<th class=\"dirtab\">");
975 ol.parseText(theTranslator->trFileIn(m_src->pathFragment()));
976 ol.writeString("</th>");
977 ol.writeString("<th class=\"dirtab\">");
978 ol.parseText(theTranslator->trIncludesFileIn(m_dst->dir()->pathFragment()));
979 ol.writeString("</th>");
980 ol.writeString("</tr>");
981
982 for (const auto &fp : m_dst->filePairs())
983 {
984 ol.writeString("<tr class=\"dirtab\">");
985 ol.writeString("<td class=\"dirtab\">");
986 writePartialFilePath(ol,m_src,fp->source());
987 ol.writeString("</td>");
988 ol.writeString("<td class=\"dirtab\">");
989 writePartialFilePath(ol,m_dst->dir(),fp->destination());
990 ol.writeString("</td>");
991 ol.writeString("</tr>");
992 }
993 ol.writeString("</table>");
994
995 ol.endContents();
996
998
1000}
1001
1002//----------------------------------------------------------------------
1003// external functions
1004
1005/** In order to create stable, but unique directory names,
1006 * we compute the common part of the path shared by all directories.
1007 */
1009{
1010 AUTO_TRACE();
1011 QCString path;
1012 auto it = Doxygen::dirLinkedMap->begin();
1013 if (!Doxygen::dirLinkedMap->empty()) // we have at least one dir
1014 {
1015 // start will full path of first dir
1016 path=removeLongPathMarker((*it)->name());
1017 int i=path.findRev('/',static_cast<int>(path.length())-2);
1018 path=path.left(i+1);
1019 bool done=FALSE;
1020 if (i==-1)
1021 {
1022 path="";
1023 }
1024 else
1025 {
1026 while (!done)
1027 {
1028 int l = static_cast<int>(path.length());
1029 size_t count=0;
1030 for (const auto &dir : *Doxygen::dirLinkedMap)
1031 {
1032 QCString dirName = removeLongPathMarker(dir->name());
1033 //printf("dirName='%s' (l=%d) path='%s' (l=%d)\n",qPrint(dirName),dirName.length(),qPrint(path),path.length());
1034 if (dirName.length()>path.length())
1035 {
1036 if (dirName.left(l)!=path) // dirName does not start with path
1037 {
1038 i = l>=2 ? path.findRev('/',l-2) : -1;
1039 if (i==-1) // no unique prefix -> stop
1040 {
1041 path="";
1042 done=TRUE;
1043 }
1044 else // restart with shorter path
1045 {
1046 path=path.left(i+1);
1047 break;
1048 }
1049 }
1050 }
1051 else // dir is shorter than path -> take path of dir as new start
1052 {
1053 path=dir->name();
1054 l=static_cast<int>(path.length());
1055 i=path.findRev('/',l-2);
1056 if (i==-1) // no unique prefix -> stop
1057 {
1058 path="";
1059 done=TRUE;
1060 }
1061 else // restart with shorter path
1062 {
1063 path=path.left(i+1);
1064 }
1065 break;
1066 }
1067 count++;
1068 }
1069 if (count==Doxygen::dirLinkedMap->size())
1070 // path matches for all directories -> found the common prefix
1071 {
1072 done=TRUE;
1073 }
1074 }
1075 }
1076 }
1077 for (const auto &dir : *Doxygen::dirLinkedMap)
1078 {
1079 QCString diskName = dir->name().right(dir->name().length()-path.length());
1080 dir->setDiskName(diskName);
1081 AUTO_TRACE_ADD("set disk name: {} -> {}",dir->name(),diskName);
1082 }
1083}
1084
1086{
1087 AUTO_TRACE();
1088 // for each input file
1089 for (const auto &fn : *Doxygen::inputNameLinkedMap)
1090 {
1091 for (const auto &fd : *fn)
1092 {
1093 if (fd->getReference().isEmpty())
1094 {
1095 DirDef *dir=Doxygen::dirLinkedMap->find(fd->getPath());
1096 if (dir==nullptr) // new directory
1097 {
1098 dir = DirDefImpl::mergeDirectoryInTree(fd->getPath());
1099 }
1100 if (dir && !fd->isDocumentationFile()) dir->addFile(fd.get());
1101 }
1102 else
1103 {
1104 // do something for file imported via tag files.
1105 }
1106 }
1107 }
1108
1109 // compute relations between directories => introduce container dirs.
1110 for (const auto &dir : *Doxygen::dirLinkedMap)
1111 {
1112 QCString name = dir->name();
1113 int i=name.findRev('/',static_cast<int>(name.length())-2);
1114 if (i>0)
1115 {
1116 DirDef *parent = Doxygen::dirLinkedMap->find(name.left(i+1));
1117 //if (parent==0) parent=root;
1118 if (parent)
1119 {
1120 parent->addSubDir(dir.get());
1121 AUTO_TRACE_ADD("DirDefImpl::addSubdir(): Adding subdir {} to {}",
1122 dir->displayName(), parent->displayName());
1123 }
1124 }
1125 }
1126
1127 // sort the directory contents
1128 for (const auto &dir : *Doxygen::dirLinkedMap)
1129 {
1130 dir->sort();
1131 }
1132
1133 // short the directories themselves
1134 std::stable_sort(Doxygen::dirLinkedMap->begin(),
1136 [](const auto &d1,const auto &d2)
1137 {
1138 QCString s1 = d1->shortName(), s2 = d2->shortName();
1139 int i = qstricmp_sort(s1,s2);
1140 if (i==0) // if sort name are equal, sort on full path
1141 {
1142 QCString n1 = d1->name(), n2 = d2->name();
1143 int n = qstricmp_sort(n1,n2);
1144 return n < 0;
1145 }
1146 return i < 0;
1147 });
1148
1149 // set the directory index identifier
1150 int dirIndex=0;
1151 for (const auto &dir : *Doxygen::dirLinkedMap)
1152 {
1153 dir->setDirIndex(dirIndex++);
1154 }
1155
1157}
1158
1160{
1161 AUTO_TRACE();
1162 // compute nesting level for each directory
1163 for (const auto &dir : *Doxygen::dirLinkedMap)
1164 {
1165 dir->setLevel();
1166 }
1167
1168 // compute uses dependencies between directories
1169 for (const auto &dir : *Doxygen::dirLinkedMap)
1170 {
1171 AUTO_TRACE_ADD("computeDependencies for {}: #dirs={}",dir->name(),Doxygen::dirLinkedMap->size());
1172 dir->computeDependencies();
1173 }
1174}
1175
1177{
1178 AUTO_TRACE();
1179 for (const auto &dir : *Doxygen::dirLinkedMap)
1180 {
1181 ol.pushGeneratorState();
1182 if (!dir->hasDocumentation())
1183 {
1185 }
1186 dir->writeDocumentation(ol);
1187 ol.popGeneratorState();
1188 }
1189 //if (Config_getBool(DIRECTORY_GRAPH))
1190 {
1191 for (const auto &dr : Doxygen::dirRelations)
1192 {
1193 dr->writeDocumentation(ol);
1194 }
1195 }
1196}
1197
1198bool compareDirDefs(const DirDef *item1, const DirDef *item2)
1199{
1200 return qstricmp_sort(item1->shortName(),item2->shortName()) < 0;
1201}
1202
1203// --- Cast functions
1204
1206{
1207 if (d==nullptr) return nullptr;
1208 if (d && typeid(*d)==typeid(DirDefImpl))
1209 {
1210 return static_cast<DirDef*>(d);
1211 }
1212 else
1213 {
1214 return nullptr;
1215 }
1216}
1217
1219{
1220 if (d==nullptr) return nullptr;
1221 if (d && typeid(*d)==typeid(DirDefImpl))
1222 {
1223 return static_cast<const DirDef*>(d);
1224 }
1225 else
1226 {
1227 return nullptr;
1228 }
1229}
1230
constexpr auto prefix
Definition anchor.cpp:44
The common base class of all entity definitions found in the sources.
Definition definition.h:76
virtual bool isLinkable() const =0
virtual QCString getReference() const =0
virtual QCString getOutputFileBase() const =0
virtual const QCString & name() const =0
bool isReference() const override
const QCString & name() const override
void writeNavigationPath(OutputList &ol) const override
bool hasBriefDescription() const override
QCString docFile() const override
QCString briefFile() const override
QCString briefDescription(bool abbreviate=FALSE) const override
DefinitionMixin(const QCString &defFileName, int defLine, int defColumn, const QCString &name, const char *b=nullptr, const char *d=nullptr, bool isSymbol=TRUE)
void setLocalName(const QCString &name) override
int docLine() const override
int briefLine() const override
QCString documentation() const override
void writeDocAnchorsToTagFile(TextStream &fs) const override
bool hasDocumentation() const override
SrcLangExt getLanguage() const override
virtual void setOuterScope(Definition *d)=0
A model of a directory symbol.
Definition dirdef.h:110
virtual int level() const =0
virtual void setLevel()=0
virtual void addFile(FileDef *fd)=0
virtual DirDef * parent() const =0
virtual void addUsesDependency(const DirDef *usedDir, const FileDef *srcFd, const FileDef *dstFd, bool srcDirect, bool dstDirect)=0
virtual void setParent(DirDef *parent)=0
virtual const QCString shortName() const =0
void writeDocumentation(OutputList &ol) override
Definition dirdef.cpp:553
int m_dirIndex
Definition dirdef.cpp:102
void writeSubDirList(OutputList &ol)
Definition dirdef.cpp:342
void setLevel() override
Definition dirdef.cpp:685
static DirDef * createNewDir(const QCString &path)
Definition dirdef.cpp:853
DirDef * m_parent
Definition dirdef.cpp:104
static bool matchPath(const QCString &path, const StringVector &l)
Definition dirdef.cpp:868
QCString m_diskName
Definition dirdef.cpp:100
bool hasSubdirs() const override
Definition dirdef.cpp:56
QCString m_shortName
Definition dirdef.cpp:99
void writeTagFile(TextStream &t) override
Definition dirdef.cpp:514
const FileList & getFiles() const override
Definition dirdef.cpp:53
const QCString shortName() const override
Definition dirdef.cpp:51
bool isLinkable() const override
Definition dirdef.cpp:156
const UsedDirLinkedMap & usedDirs() const override
Definition dirdef.cpp:60
void addSubDir(DirDef *subdir) override
Definition dirdef.cpp:161
bool hasDetailedDescription() const override
Definition dirdef.cpp:508
DirDef * parent() const override
Definition dirdef.cpp:58
void startMemberDeclarations(OutputList &ol)
Definition dirdef.cpp:486
int level() const override
Definition dirdef.cpp:57
bool isLinkableInProject() const override
Definition dirdef.cpp:151
void writePageNavigation(OutputList &ol) const override
Definition dirdef.cpp:317
void writeDetailedDescription(OutputList &ol, const QCString &title)
Definition dirdef.cpp:231
DirList m_subdirs
Definition dirdef.cpp:97
DefType definitionType() const override
Definition dirdef.cpp:44
bool isParentOf(const DirDef *dir) const override
Definition dirdef.cpp:805
QCString getOutputFileBase() const override
Definition dirdef.cpp:224
int dirIndex() const override
Definition dirdef.cpp:59
QCString m_dispName
Definition dirdef.cpp:98
QCString anchor() const override
Definition dirdef.cpp:47
CodeSymbolType codeSymbolType() const override
Definition dirdef.cpp:45
void addFile(FileDef *fd) override
Definition dirdef.cpp:178
bool depGraphIsTrivial() const override
Definition dirdef.cpp:815
void findSectionsInDocumentation() override
Definition dirdef.cpp:765
void overrideDirectoryGraph(bool e) override
Definition dirdef.cpp:901
void addUsesDependency(const DirDef *usedDir, const FileDef *srcFd, const FileDef *dstFd, bool srcDirect, bool dstDirect) override
Add as "uses" dependency between this dir and dir, that was caused by a dependency on file fd.
Definition dirdef.cpp:707
void writeDirectoryGraph(OutputList &ol)
Definition dirdef.cpp:322
const DirList & subDirs() const override
Definition dirdef.cpp:55
UsedDirLinkedMap m_usedDirs
Definition dirdef.cpp:105
void setDiskName(const QCString &name) override
Definition dirdef.cpp:68
static DirDef * mergeDirectoryInTree(const QCString &path)
Definition dirdef.cpp:884
void writeFileList(OutputList &ol)
Definition dirdef.cpp:402
QCString shortTitle() const override
Definition dirdef.cpp:496
DirDefImpl(const QCString &path)
Definition dirdef.cpp:118
void setDirIndex(int index) override
Definition dirdef.cpp:173
void endMemberDeclarations(OutputList &ol)
Definition dirdef.cpp:491
int m_level
Definition dirdef.cpp:103
FileList m_fileList
Definition dirdef.cpp:101
void sort() override
Definition dirdef.cpp:184
~DirDefImpl() override
Definition dirdef.cpp:147
void writeBriefDescription(OutputList &ol)
Definition dirdef.cpp:278
void computeDependencies() override
Computes the dependencies between directories.
Definition dirdef.cpp:773
QCString displayName(bool=TRUE) const override
Definition dirdef.cpp:50
void setParent(DirDef *parent) override
Definition dirdef.cpp:168
bool m_hasDirectoryGraph
Definition dirdef.cpp:106
bool hasDirectoryGraph() const override
Definition dirdef.cpp:906
A list of directories.
Definition dirdef.h:178
const DirDef * m_src
Definition dirdef.h:167
UsedDir * m_dst
Definition dirdef.h:168
void writeDocumentation(OutputList &ol)
Definition dirdef.cpp:949
QCString getOutputFileBase() const
Definition dirdef.h:163
Representation of an directory dependency graph.
Definition dotdirdeps.h:26
bool isTrivial() const
static FileNameLinkedMap * inputNameLinkedMap
Definition doxygen.h:105
static DirLinkedMap * dirLinkedMap
Definition doxygen.h:129
static DirRelationLinkedMap dirRelations
Definition doxygen.h:130
A model of a file symbol.
Definition filedef.h:99
virtual void setDirDef(DirDef *dd)=0
virtual DirDef * getDirDef() const =0
Class representing a pair of FileDef objects.
Definition dirdef.h:42
static QCString key(const FileDef *srcFd, const FileDef *dstFd)
Definition dirdef.cpp:913
static LayoutDocManager & instance()
Returns a reference to this singleton.
Definition layout.cpp:1435
Class representing a list of output generators that are written to in parallel.
Definition outputlist.h:314
void writeString(const QCString &text)
Definition outputlist.h:413
void startMemberDeclaration()
Definition outputlist.h:571
void disable(OutputType o)
void endDirDepGraph(DotDirDeps &g)
Definition outputlist.h:664
void writeRuler()
Definition outputlist.h:523
void startGroupHeader(const QCString &id=QCString(), int extraLevels=0)
Definition outputlist.h:455
void enable(OutputType o)
void endContents()
Definition outputlist.h:622
void endMemberDescription()
Definition outputlist.h:569
void writeObjectLink(const QCString &ref, const QCString &file, const QCString &anchor, const QCString &name)
Definition outputlist.h:441
void writeDoc(const IDocNodeAST *ast, const Definition *ctx, const MemberDef *md)
Definition outputlist.h:385
void startMemberDescription(const QCString &anchor, const QCString &inheritId=QCString(), bool typ=false)
Definition outputlist.h:567
void docify(const QCString &s)
Definition outputlist.h:439
void startParagraph(const QCString &classDef=QCString())
Definition outputlist.h:409
void endParagraph()
Definition outputlist.h:411
void startDirDepGraph()
Definition outputlist.h:662
void startMemberSections()
Definition outputlist.h:463
void startMemberList()
Definition outputlist.h:483
void endTextLink()
Definition outputlist.h:446
void startBold()
Definition outputlist.h:563
void endMemberItem(OutputGenerator::MemberItemType type)
Definition outputlist.h:497
void endMemberList()
Definition outputlist.h:485
void writeSynopsis()
Definition outputlist.h:594
void pushGeneratorState()
void insertMemberAlign(bool templ=FALSE)
Definition outputlist.h:519
void disableAllBut(OutputType o)
void popGeneratorState()
void writeAnchor(const QCString &fileName, const QCString &name)
Definition outputlist.h:525
void endBold()
Definition outputlist.h:565
void endGroupHeader(int extraLevels=0)
Definition outputlist.h:457
void endQuickIndices()
Definition outputlist.h:606
void writePageOutline()
Definition outputlist.h:618
void generateDoc(const QCString &fileName, int startLine, const Definition *ctx, const MemberDef *md, const QCString &docStr, bool indexWords, bool isExample, const QCString &exampleName, bool singleLine, bool linkFromIndex, bool markdownSupport=Config_getBool(MARKDOWN_SUPPORT), bool autolinkSupport=Config_getBool(AUTOLINK_SUPPORT))
void startContents()
Definition outputlist.h:620
void endMemberDeclaration(const QCString &anchor, const QCString &inheritId)
Definition outputlist.h:573
void enableAll()
void endMemberHeader()
Definition outputlist.h:473
void startMemberItem(const QCString &anchor, OutputGenerator::MemberItemType type, const QCString &id=QCString())
Definition outputlist.h:495
void parseText(const QCString &textStr)
void startTextLink(const QCString &file, const QCString &anchor)
Definition outputlist.h:444
void startMemberHeader(const QCString &anchor, int typ=2)
Definition outputlist.h:471
void endMemberSections()
Definition outputlist.h:465
This is an alternative implementation of QCString.
Definition qcstring.h:101
int find(char c, int index=0, bool cs=TRUE) const
Definition qcstring.cpp:43
QCString & prepend(const char *s)
Definition qcstring.h:407
size_t length() const
Returns the length of the string, not counting the 0-terminator.
Definition qcstring.h:153
bool isEmpty() const
Returns TRUE iff the string is empty.
Definition qcstring.h:150
QCString right(size_t len) const
Definition qcstring.h:219
int findRev(char c, int index=-1, bool cs=TRUE) const
Definition qcstring.cpp:91
const char * data() const
Returns a pointer to the contents of the string in the form of a 0-terminated C string.
Definition qcstring.h:159
QCString left(size_t len) const
Definition qcstring.h:214
Text streaming class that buffers data.
Definition textstream.h:36
Usage information of a directory.
Definition dirdef.h:64
bool m_hasDirectDstDeps
Definition dirdef.h:103
void sort()
Definition dirdef.cpp:835
FilePairLinkedMap m_filePairs
Definition dirdef.h:99
bool m_hasDirectDeps
Definition dirdef.h:101
void addFileDep(const FileDef *srcFd, const FileDef *dstFd, bool srcDirect, bool dstDirect)
Take up dependency between files.
Definition dirdef.cpp:827
const DirDef * dir() const
Definition dirdef.h:78
UsedDir(const DirDef *dir)
Definition dirdef.cpp:822
FilePair * findFilePair(const QCString &name)
Definition dirdef.cpp:848
bool m_hasDirectSrcDeps
Definition dirdef.h:102
const DirDef * m_dir
Definition dirdef.h:98
#define Config_getList(name)
Definition config.h:38
#define Config_getBool(name)
Definition config.h:33
#define NON_COPYABLE(cls)
Macro to help implementing the rule of 5 for a non-copyable & movable class.
Definition construct.h:37
std::vector< std::string > StringVector
Definition containers.h:33
DirIterator begin(DirIterator it) noexcept
Definition dir.cpp:170
DirIterator end(const DirIterator &) noexcept
Definition dir.cpp:175
DirDef * createDirDef(const QCString &path)
Definition dirdef.cpp:109
static void writePartialFilePath(OutputList &ol, const DirDef *root, const FileDef *fd)
Definition dirdef.cpp:930
bool compareDirDefs(const DirDef *item1, const DirDef *item2)
Definition dirdef.cpp:1198
static void writePartialDirPath(OutputList &ol, const DirDef *root, const DirDef *target)
Definition dirdef.cpp:920
static QCString encodeDirName(const QCString &anchor)
Definition dirdef.cpp:190
void buildDirectories()
Definition dirdef.cpp:1085
void computeDirDependencies()
Definition dirdef.cpp:1159
DirDef * toDirDef(Definition *d)
Definition dirdef.cpp:1205
static void computeCommonDirPrefix()
In order to create stable, but unique directory names, we compute the common part of the path shared ...
Definition dirdef.cpp:1008
void generateDirDocs(OutputList &ol)
Definition dirdef.cpp:1176
#define AUTO_TRACE_ADD(...)
Definition docnode.cpp:47
#define AUTO_TRACE(...)
Definition docnode.cpp:46
#define AUTO_TRACE_EXIT(...)
Definition docnode.cpp:48
constexpr DocNodeVariant * parent(DocNodeVariant *n)
returns the parent node of a given node n or nullptr if the node has no parent.
Definition docnode.h:1330
void docFindSections(const QCString &input, const Definition *d, const QCString &fileName)
IDocNodeASTPtr validatingParseDoc(IDocParser &parserIntf, const QCString &fileName, int startLine, const Definition *ctx, const MemberDef *md, const QCString &input, bool indexWords, bool isExample, const QCString &exampleName, bool singleLine, bool linkFromIndex, bool markdownSupport, bool autolinkSupport)
IDocParserPtr createDocParser()
factory function to create a parser
Definition docparser.cpp:55
static void writeTagFile()
bool compareFileDefs(const FileDef *fd1, const FileDef *fd2)
Definition filedef.cpp:1925
void startTitle(OutputList &ol, const QCString &fileName, const DefinitionMutable *def)
Definition index.cpp:384
void endTitle(OutputList &ol, const QCString &fileName, const QCString &name)
Definition index.cpp:394
void startFile(OutputList &ol, const QCString &name, const QCString &manName, const QCString &title, HighlightedItem hli, bool additionalIndices, const QCString &altSidebarName, int hierarchyLevel, const QCString &allMembersFile)
Definition index.cpp:401
void endFileWithNavPath(OutputList &ol, const DefinitionMutable *d, bool showPageNavigation)
Definition index.cpp:448
Translator * theTranslator
Definition language.cpp:71
#define msg(fmt,...)
Definition message.h:94
#define err(fmt,...)
Definition message.h:127
int qstricmp_sort(const char *str1, const char *str2)
Definition qcstring.h:86
const char * qPrint(const char *s)
Definition qcstring.h:672
#define TRUE
Definition qcstring.h:37
#define FALSE
Definition qcstring.h:34
#define ASSERT(x)
Definition qcstring.h:39
Definition layout.h:102
QCString title(SrcLangExt lang) const
Definition layout.cpp:1779
CodeSymbolType
Definition types.h:481
SrcLangExt
Definition types.h:207
bool fileVisibleInIndex(const FileDef *fd, bool &genSourceFile)
Definition util.cpp:6600
static QCString stripFromPath(const QCString &p, const StringVector &l)
Definition util.cpp:310
QCString convertToXML(const QCString &s, bool keepEntities)
Definition util.cpp:4425
QCString removeLongPathMarker(QCString path)
Definition util.cpp:299
void addHtmlExtensionIfMissing(QCString &fName)
Definition util.cpp:5412
A bunch of utility functions.