0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
diagram.h
Go to the documentation of this file.
1/**
2 * @file src/class_diagram/model/diagram.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 "class.h"
25#include "common/types.h"
26#include "concept.h"
27#include "config/config.h"
28#include "enum.h"
29#include "objc_interface.h"
30
31#include <regex>
32#include <string>
33#include <unordered_set>
34#include <vector>
35
37
38using common::opt_ref;
39using common::model::diagram_element;
41using common::model::eid_t;
42using common::model::element_view;
43using common::model::element_views;
44using common::model::path;
46
50
51/**
52 * @brief Class representing a class diagram.
53 */
55 public element_views<class_, enum_, concept_, objc_interface>,
56 public nested_trait_ns {
57public:
59
60 diagram() = default;
61
62 diagram(const diagram &) = delete;
63 diagram(diagram &&) = default;
64 diagram &operator=(const diagram &) = delete;
65 diagram &operator=(diagram &&) = default;
66
67 /**
68 * @brief Get the diagram model type - in this case class.
69 *
70 * @return Type of class diagram.
71 */
72 diagram_t type() const override;
73
74 /**
75 * Inherit the should_include methods from the common diagram model.
76 */
78
79 /**
80 * @brief Whether a class_member should be included in the diagram.
81 *
82 * @param m Class member
83 * @return True, if class member should be included in the diagram.
84 */
85 bool should_include(const class_member &m) const;
86 bool should_include(const objc_member &m) const;
87
88 /**
89 * @brief Whether a class_method should be included in the diagram.
90 *
91 * @param m Class method
92 * @return True, if class method should be included in the diagram.
93 */
94 bool should_include(const class_method &m) const;
95 bool should_include(const objc_method &m) const;
96
97 /**
98 * @brief Search for element in the diagram by fully qualified name.
99 *
100 * @param full_name Fully qualified element name.
101 * @return Optional reference to a diagram element.
102 */
103 opt_ref<diagram_element> get(const std::string &full_name) const override;
104
105 /**
106 * @brief Search for element in the diagram by id.
107 *
108 * @param id Element id.
109 * @return Optional reference to a diagram element.
110 */
111 opt_ref<diagram_element> get(eid_t id) const override;
112
113 /**
114 * @brief Get list of references to classes in the diagram model.
115 *
116 * @return List of references to classes in the diagram model.
117 */
119
120 /**
121 * @brief Get list of references to enums in the diagram model.
122 *
123 * @return List of references to enums in the diagram model.
124 */
126
127 /**
128 * @brief Get list of references to concepts in the diagram model.
129 *
130 * @return List of references to concepts in the diagram model.
131 */
133
135
136 /**
137 * @brief Check, if diagram contains a specific element.
138 *
139 * @tparam ElementT Type of diagram element (e.g. class_)
140 * @param e Element to check
141 * @return True, if element already exists in the diagram
142 */
143 template <typename ElementT> bool contains(const ElementT &e);
144
145 /**
146 * @brief Find an element in the diagram by name.
147 *
148 * This method allows for typed search, where the type of searched for
149 * element is determined from template specialization.
150 *
151 * @tparam ElementT Type of element (e.g. class_)
152 * @param name Fully qualified name of the element
153 * @return Optional reference to a diagram element
154 */
155 template <typename ElementT>
156 opt_ref<ElementT> find(const std::string &name) const;
157
158 /**
159 * @brief Find elements in the diagram by regex pattern.
160 *
161 * This method allows for typed search, where the type of searched for
162 * element is determined from template specialization.
163 *
164 * @tparam ElementT Type of element (e.g. class_)
165 * @param name String or regex pattern
166 * @return List of optional references to matched elements.
167 */
168 template <typename ElementT>
169 std::vector<opt_ref<ElementT>> find(
170 const clanguml::common::string_or_regex &pattern) const;
171
172 /**
173 * @brief Find an element in the diagram by id.
174 *
175 * This method allows for typed search, where the type of searched for
176 * element is determined from template specialization.
177 *
178 * @tparam ElementT Type of element (e.g. class_)
179 * @param id Id of the element
180 * @return Optional reference to a diagram element
181 */
182 template <typename ElementT> opt_ref<ElementT> find(eid_t id) const;
183
184 /**
185 * @brief Get reference to vector of elements of specific type
186 *
187 * @tparam ElementT Type of elements view
188 * @return Reference to elements vector
189 */
190 template <typename ElementT>
192
193 /**
194 * @brief Add element to the diagram at a specified nested path.
195 *
196 * Adds an element to a diagram, at a specific package (if any exist).
197 * The package is specified by the `parent_path`, which can be either
198 * a namespace or a directory path.
199 *
200 * @tparam ElementT Type of diagram element.
201 * @param parent_path Path to the parent package of the new diagram element.
202 * @param e Diagram element to be added.
203 * @return True, if the element was added to the diagram.
204 */
205 template <typename ElementT>
206 bool add(const path &parent_path, std::unique_ptr<ElementT> &&e)
207 {
208 if (parent_path.type() == common::model::path_type::kNamespace) {
209 return add_with_namespace_path(std::move(e));
210 }
211
212 if (parent_path.type() == common::model::path_type::kModule) {
213 return add_with_module_path(parent_path, std::move(e));
214 }
215
216 return add_with_filesystem_path(parent_path, std::move(e));
217 }
218
219 template <typename ElementT> void move(eid_t id, const path &parent_path)
220 {
221 LOG_DBG("Moving element {} to package {}", id.value(),
222 parent_path.to_string());
223
224 auto e = nested_trait_ns::get_and_remove<ElementT>(id);
225 assert(e);
226
228 added_elements_.erase(id);
229
230 this->add<ElementT>(parent_path, std::move(e));
231 }
232
233 template <typename ElementT> void remove(eid_t id)
234 {
237 }
238
239 /**
240 * @brief Convert element id to PlantUML alias.
241 *
242 * @todo This method does not belong here - refactor to PlantUML specific
243 * code.
244 *
245 * @param id Id of the diagram element.
246 * @return PlantUML alias.
247 */
248 std::string to_alias(eid_t id) const;
249
250 /**
251 * @brief Given an initial set of classes, add all their parents to the
252 * argument.
253 * @param parents In and out parameter with the parent classes.
254 */
256
257 /**
258 * @brief Check if diagram contains element by id.
259 *
260 * @todo Remove in favour of 'contains'
261 *
262 * @param id Id of the element.
263 * @return True, if diagram contains an element with a specific id.
264 */
265 bool has_element(eid_t id) const override;
266
267 /**
268 * @brief Remove redundant dependency relationships
269 */
271
272 /**
273 * @brief Check whether the diagram is empty
274 *
275 * @return True, if diagram is empty
276 */
277 bool is_empty() const override;
278
279 void apply_filter() override;
280
281private:
282 std::set<eid_t> added_elements_;
283
284 template <typename ElementT>
285 bool add_with_namespace_path(std::unique_ptr<ElementT> &&e);
286
287 template <typename ElementT>
289 const common::model::path &parent_path, std::unique_ptr<ElementT> &&e);
290
291 template <typename ElementT>
293 const common::model::path &parent_path, std::unique_ptr<ElementT> &&e);
294};
295
296template <typename ElementT> bool diagram::contains(const ElementT &element)
297{
298 return std::any_of(element_view<ElementT>::view().cbegin(),
300 [&element](
301 const auto &element_opt) { return element_opt.get() == element; });
302}
303
304template <typename ElementT>
305bool diagram::add_with_namespace_path(std::unique_ptr<ElementT> &&e)
306{
307 if (added_elements_.count(e->id()) > 0)
308 return true;
309
310 added_elements_.emplace(e->id());
311
312 const auto base_name = e->name();
313 const auto full_name = e->full_name(false);
314 const auto element_type = e->type_name();
315
316 LOG_DBG("Adding {}: {}::{}, {}", element_type,
317 e->get_namespace().to_string(), base_name, full_name);
318
319 if (util::contains(base_name, "::"))
320 throw std::runtime_error("Name cannot contain namespace: " + base_name);
321
322 const auto ns = e->get_relative_namespace();
323
324 auto name = base_name;
325 auto name_and_ns = ns | name;
326 auto &e_ref = *e;
327 auto id = e_ref.id();
328
329 try {
330 if (!contains(e_ref)) {
331 if (add_element(ns, std::move(e)))
332 element_view<ElementT>::add(std::ref(e_ref));
333
334#if !defined(NDEBUG)
335 const auto maybe_el = get_element<ElementT>(name_and_ns);
336 const auto &el = maybe_el.value();
337
338 if ((el.name() != name) || !(el.get_relative_namespace() == ns))
339 throw std::runtime_error(
340 "Invalid element stored in the diagram tree");
341
342 LOG_DBG("Added {} {} ({} - [{}])", element_type, base_name,
343 full_name, id);
344#endif
345
346 return true;
347 }
348 }
349 catch (const std::runtime_error &e) {
350 LOG_WARN("Cannot add {} {} with id {} due to: {}", element_type, name,
351 id, e.what());
352 return false;
353 }
354
355 LOG_DBG("{} {} ({} - [{}]) already in the model", element_type, base_name,
356 full_name, id);
357
358 return false;
359}
360
361template <typename ElementT>
363 const common::model::path &parent_path, std::unique_ptr<ElementT> &&e)
364{
365 if (added_elements_.count(e->id()) > 0)
366 return true;
367
368 added_elements_.emplace(e->id());
369
370 const auto element_type = e->type_name();
371
372 // Make sure all parent modules are already packages in the
373 // model
374 for (auto it = parent_path.begin(); it != parent_path.end(); it++) {
375 auto pkg = std::make_unique<common::model::package>(
376 e->using_namespace(), parent_path.type());
377 pkg->set_name(*it);
378 auto ns =
379 common::model::path(parent_path.begin(), it, parent_path.type());
380 // ns.pop_back();
381 pkg->set_namespace(ns);
382 pkg->set_id(common::to_id(pkg->full_name(false)));
383
384 add(ns, std::move(pkg));
385 }
386
387 const auto base_name = e->name();
388 const auto full_name = e->full_name(false);
389 auto &e_ref = *e;
390
391 if (add_element(parent_path, std::move(e))) {
392 element_view<ElementT>::add(std::ref(e_ref));
393 return true;
394 }
395
396 return false;
397}
398
399template <typename ElementT>
401 const common::model::path &parent_path, std::unique_ptr<ElementT> &&e)
402{
403 if (added_elements_.count(e->id()) > 0)
404 return false;
405
406 LOG_DBG("Adding element {} at path {}", e->full_name(false),
407 parent_path.to_string());
408
409 const auto element_type = e->type_name();
410
411 // Make sure all parent modules are already packages in the
412 // model
413 for (auto it = parent_path.begin(); it != parent_path.end(); it++) {
414 auto pkg = std::make_unique<common::model::package>(
415 e->using_namespace(), parent_path.type());
416 pkg->set_name(*it);
417 auto package_path =
418 common::model::path(parent_path.begin(), it, parent_path.type());
419 pkg->set_namespace(package_path);
420 pkg->set_id(common::to_id("__directory__" + pkg->full_name(false)));
421
422 LOG_DBG("Adding filesystem package {} at path {}", pkg->name(),
423 package_path.to_string());
424
425 add(package_path, std::move(pkg));
426 }
427
428 const auto base_name = e->name();
429 const auto full_name = e->full_name(false);
430 auto &e_ref = *e;
431
432 if (add_element(parent_path, std::move(e))) {
433 added_elements_.emplace(e_ref.id());
434 element_view<ElementT>::add(std::ref(e_ref));
435 return true;
436 }
437
438 return false;
439}
440
441template <typename ElementT>
442opt_ref<ElementT> diagram::find(const std::string &name) const
443{
444 for (const auto &element : element_view<ElementT>::view()) {
445 const auto full_name = element.get().full_name(false);
446
447 auto full_name_escaped = full_name;
448 util::replace_all(full_name_escaped, "##", "::");
449
450 if (name == full_name || name == full_name_escaped) {
451 return {element};
452 }
453 }
454
455 return {};
456}
457
458template <typename ElementT>
459std::vector<opt_ref<ElementT>> diagram::find(
460 const common::string_or_regex &pattern) const
461{
462 std::vector<opt_ref<ElementT>> result;
463
464 for (const auto &element : element_view<ElementT>::view()) {
465 const auto full_name = element.get().full_name(false);
466 auto full_name_escaped = full_name;
467 util::replace_all(full_name_escaped, "##", "::");
468
469 if (pattern == full_name || pattern == full_name_escaped) {
470 result.emplace_back(element);
471 }
472 }
473
474 return result;
475}
476
477template <typename ElementT> opt_ref<ElementT> diagram::find(eid_t id) const
478{
479 for (const auto &element : element_view<ElementT>::view()) {
480 if (element.get().id() == id) {
481 return {element};
482 }
483 }
484
485 return {};
486}
487
488template <typename ElementT>
490{
492}
493
494//
495// Template method specialization pre-declarations...
496//
497template <>
498bool diagram::add_with_namespace_path<common::model::package>(
499 std::unique_ptr<common::model::package> &&p);
500
501template <>
502bool diagram::add_with_module_path<common::model::package>(
503 const common::model::path &parent_path,
504 std::unique_ptr<common::model::package> &&p);
505
506template <>
507bool diagram::add_with_filesystem_path<common::model::package>(
508 const common::model::path &parent_path,
509 std::unique_ptr<common::model::package> &&p);
510
511} // namespace clanguml::class_diagram::model
512
513namespace clanguml::common::model {
514template <>
515bool check_diagram_type<clanguml::class_diagram::model::diagram>(diagram_t t);
516}