0.6.0
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
source_file.h
Go to the documentation of this file.
1/**
2 * @file src/common/model/source_file.h
3 *
4 * Copyright (c) 2021-2025 Bartek Kryza <bkryza@gmail.com>
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 */
18#pragma once
19
20#include "common/clang_utils.h"
23#include "common/model/path.h"
26#include "common/types.h"
27#include "util/util.h"
28
29#include <spdlog/spdlog.h>
30
31#include <set>
32#include <string>
33#include <vector>
34
36
37/**
38 * This enum represents different kinds of files in the diagram.
39 */
40enum class source_file_t {
41 kDirectory, /*!< Diagram element is a directory */
42 kHeader, /*!< Diagram element is a header */
43 kImplementation /*!< Diagram element is a source file (e.g. cpp) */
44};
45
46std::string to_string(source_file_t sf);
47
49#ifdef _WIN32
50 static constexpr std::string_view value = "\\";
51#else
52 static constexpr std::string_view value = "/";
53#endif
54};
55
57
58/**
59 * @brief Diagram element representing some file or directory.
60 *
61 * @embed{source_file_hierarchy_class.svg}
62 */
66 public common::model::nested_trait<common::model::source_file,
67 filesystem_path> {
68public:
69 source_file() = default;
70
71 explicit source_file(const std::filesystem::path &p)
72 {
73 auto preferred = p;
74 preferred.make_preferred();
75 set_path({preferred.parent_path().string(), path_type::kFilesystem});
76 set_name(preferred.filename().string());
77 is_absolute_ = preferred.is_absolute();
78 set_id(common::to_id(preferred));
79 }
80
81 source_file(const source_file &) = delete;
82 source_file(source_file &&) = default;
83 source_file &operator=(const source_file &) = delete;
85
86 bool operator==(const source_file &right) const
87 {
88 return (path_ == right.path_) && (name() == right.name()) &&
89 (type_ == right.type_);
90 }
91
92 /**
93 * Set the path to the element in the diagram.
94 *
95 * @param p Diagram path.
96 */
97 void set_path(const filesystem_path &p) { path_ = p; }
98
99 /**
100 * Is the elements path absolute?
101 *
102 * @return True if the elements path is absolute.
103 */
104 bool is_absolute() const { return is_absolute_; }
105
106 /**
107 * Set the type of the source file.
108 *
109 * @param type Type of the source file.
110 */
112
113 /**
114 * Get the source file elements type.
115 *
116 * @return Type of the source file.
117 */
118 source_file_t type() const { return type_; }
119
120 /**
121 * Set whether the file is a system header
122 *
123 * @param is_system Whether the file is a system header
124 */
125 void set_system_header(bool is_system) { is_system_header_ = is_system; }
126
127 /**
128 * Is the file a system header?
129 *
130 * @return True, if the source file is a system header
131 */
132 bool is_system_header() const { return is_system_header_; }
133
134 /**
135 * Get the source file's parent path.
136 *
137 * @return Source file parent path.
138 */
139 const filesystem_path &path() const { return path_; }
140
141 /**
142 * Return full path, i.e. parent path and elements name.
143 *
144 * @return Full source file path.
145 */
146 auto full_path() const { return path() | name(); }
147
148 /**
149 * Convert the source file path to std::filesystem::path, relative to `base`
150 *
151 * @param base Base path
152 * @return Filesystem path to the source file.
153 */
154 std::filesystem::path fs_path(const std::filesystem::path &base = {}) const
155 {
156 std::filesystem::path res;
157
158 for (const auto &path_element : path_) {
159 res /= path_element;
160 }
161
162 res /= name();
163
164 if (is_absolute_)
165 res = fs_path_sep::value / res;
166 else
167 res = base / res;
168
169 return res.lexically_normal();
170 }
171
172protected:
173 /**
174 * Return the full path string, i.e. parent path and elements name.
175 *
176 * @return Full source file path as string.
177 */
178 std::string full_name_impl(bool /*relative*/) const override
179 {
180 return (path_ | name()).to_string();
181 }
182
183private:
186 bool is_absolute_{false};
187 bool is_system_header_{false};
188};
189} // namespace clanguml::common::model
190
191namespace std {
192template <>
193struct hash<std::reference_wrapper<clanguml::common::model::source_file>> {
194 std::size_t operator()(
195 const std::reference_wrapper<clanguml::common::model::source_file> &key)
196 const
197 {
198 return key.get().id().value();
199 }
200};
201} // namespace std