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