FTXUI  6.0.2
C++ functional terminal UI.
All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
dbox.cpp
Go to the documentation of this file.
1// Copyright 2020 Arthur Sonzogni. All rights reserved.
2// Use of this source code is governed by the MIT license that can be found in
3// the LICENSE file.
4#include <algorithm> // for max
5#include <cstddef> // for size_t
6#include <memory> // for __shared_ptr_access, shared_ptr, make_shared
7#include <utility> // for move
8#include <vector>
9
10#include "ftxui/dom/elements.hpp" // for Element, Elements, dbox
11#include "ftxui/dom/node.hpp" // for Node, Elements
12#include "ftxui/dom/requirement.hpp" // for Requirement
13#include "ftxui/screen/box.hpp" // for Box
14#include "ftxui/screen/pixel.hpp" // for Pixel
15
16namespace ftxui {
17
18namespace {
19class DBox : public Node {
20 public:
21 explicit DBox(Elements children) : Node(std::move(children)) {}
22
23 void ComputeRequirement() override {
24 requirement_ = Requirement{};
25 for (auto& child : children_) {
26 child->ComputeRequirement();
27
28 // Propagate the focused requirement.
29 if (requirement_.focused.Prefer(child->requirement().focused)) {
30 requirement_.focused = child->requirement().focused;
31 }
32
33 // Extend the min_x and min_y to contain all the children
34 requirement_.min_x =
35 std::max(requirement_.min_x, child->requirement().min_x);
36 requirement_.min_y =
37 std::max(requirement_.min_y, child->requirement().min_y);
38 }
39 }
40
41 void SetBox(Box box) override {
42 Node::SetBox(box);
43
44 for (auto& child : children_) {
45 child->SetBox(box);
46 }
47 }
48
49 void Render(Screen& screen) override {
50 if (children_.size() <= 1) {
52 return;
53 }
54
55 const int width = box_.x_max - box_.x_min + 1;
56 const int height = box_.y_max - box_.y_min + 1;
57 std::vector<Pixel> pixels(std::size_t(width * height));
58
59 for (auto& child : children_) {
60 child->Render(screen);
61
62 // Accumulate the pixels
63 Pixel* acc = pixels.data();
64 for (int x = 0; x < width; ++x) {
65 for (int y = 0; y < height; ++y) {
66 auto& pixel = screen.PixelAt(x + box_.x_min, y + box_.y_min);
67 acc->background_color =
68 Color::Blend(acc->background_color, pixel.background_color);
69 acc->automerge = pixel.automerge || acc->automerge;
70 if (pixel.character.empty()) {
71 acc->foreground_color =
72 Color::Blend(acc->foreground_color, pixel.background_color);
73 } else {
74 acc->blink = pixel.blink;
75 acc->bold = pixel.bold;
76 acc->dim = pixel.dim;
77 acc->inverted = pixel.inverted;
78 acc->italic = pixel.italic;
79 acc->underlined = pixel.underlined;
80 acc->underlined_double = pixel.underlined_double;
81 acc->strikethrough = pixel.strikethrough;
82 acc->hyperlink = pixel.hyperlink;
83 acc->character = pixel.character;
84 acc->foreground_color = pixel.foreground_color;
85 }
86 ++acc; // NOLINT
87
88 pixel = Pixel();
89 }
90 }
91 }
92
93 // Render the accumulated pixels:
94 Pixel* acc = pixels.data();
95 for (int x = 0; x < width; ++x) {
96 for (int y = 0; y < height; ++y) {
97 screen.PixelAt(x + box_.x_min, y + box_.y_min) = *acc++; // NOLINT
98 }
99 }
100 }
101};
102} // namespace
103
104/// @brief Stack several element on top of each other.
105/// @param children_ The input element.
106/// @return The right aligned element.
107/// @ingroup dom
109 return std::make_shared<DBox>(std::move(children_));
110}
111
112} // namespace ftxui
static Color Blend(const Color &lhs, const Color &rhs)
Blend two colors together using the alpha channel.
Definition color.cpp:281
virtual void SetBox(Box box)
Assign a position and a dimension to an element for drawing.
Definition node.cpp:43
friend void Render(Screen &screen, Node *node, Selection &selection)
Definition node.cpp:100
std::shared_ptr< Node > Element
Definition elements.hpp:22
std::shared_ptr< T > Make(Args &&... args)
Definition component.hpp:26
std::vector< Element > Elements
Definition elements.hpp:23
Element dbox(Elements)
Stack several element on top of each other.
Definition dbox.cpp:108
void Render(Screen &screen, const Element &element)
Display an element on a ftxui::Screen.
Definition node.cpp:88