Files
MuditaOS/module-gui/gui/widgets/TextLine.hpp
Przemyslaw Brudny bc76f7360e [EGD-5684] Add RichText empty line support
Added RichText option to add empty line. Tests added. Fixed
scroll up navigation line first char calculation.
2021-02-10 13:22:05 +01:00

157 lines
5.0 KiB
C++

// Copyright (c) 2017-2020, Mudita Sp. z.o.o. All rights reserved.
// For licensing, see https://github.com/mudita/MuditaOS/LICENSE.md
#pragma once
#include <numeric>
#include <list>
#include "Common.hpp"
#include "Label.hpp"
#include "TextDocument.hpp"
#include "BoxLayout.hpp"
#include "TextLineCursor.hpp"
namespace gui
{
enum class UnderlineDrawMode
{
WholeLine,
Concurrent
};
/// interface element for TextDocument->getLine() <-- Text
class TextLine
{
unsigned int shownLetterCount = 0;
Length widthUsed = 0;
Length heightUsed = 0;
Length maxWidth = 0;
std::list<Label *> lineContent;
Rect *underline = nullptr;
bool drawUnderline = false;
UnderlineDrawMode drawUnderlineMode = UnderlineDrawMode::Concurrent;
Position underlinePadding = 0;
Position underlineHeight = 0;
TextBlock::End end = TextBlock::End::None;
Position storedYOffset = 0;
bool lineEnd = false;
bool lineVisible = true;
unsigned int lineStartBlockNumber = text::npos;
unsigned int lineStartBlockPosition = text::npos;
void createUnderline(unsigned int max_w, unsigned int max_height);
void updateUnderline(const short &x, const short &y);
void setLineStartConditions(unsigned int startBlockNumber, unsigned int startBlockPosition);
public:
/// creates TextLine with data from text based on TextCursor position filling max_width
TextLine(BlockCursor &, unsigned int max_width);
TextLine(TextLine &) = delete;
TextLine(TextLine &&) noexcept;
TextLine(BlockCursor &cursor,
unsigned int max_width,
unsigned int init_height,
bool drawUnderline,
UnderlineDrawMode drawUnderlineMode,
Position underlinePadding)
: TextLine(cursor, max_width)
{
this->drawUnderline = drawUnderline;
this->drawUnderlineMode = drawUnderlineMode;
this->underlinePadding = underlinePadding;
this->underlineHeight = init_height;
}
~TextLine();
/// number of letters in Whole TextLines
[[nodiscard]] unsigned int length() const noexcept
{
return shownLetterCount;
}
/// count of elements in whole TextLine
[[nodiscard]] unsigned int count() const noexcept
{
return std::accumulate(lineContent.begin(), lineContent.end(), 0U, [](const auto sum, const auto &content) {
return sum + content->getTextLength();
});
}
[[nodiscard]] bool empty() const noexcept
{
return ((end == TextBlock::End::Newline && count() == 1) || (end == TextBlock::End::None && count() == 0));
}
[[nodiscard]] Length width() const noexcept
{
return widthUsed;
}
[[nodiscard]] Length height() const noexcept
{
return heightUsed;
}
[[nodiscard]] TextBlock::End getEnd() const noexcept
{
return end;
}
[[nodiscard]] bool getLineEnd() const noexcept
{
return lineEnd;
}
[[nodiscard]] auto isVisible() const noexcept
{
return lineVisible;
}
void setVisible(bool val) noexcept
{
lineVisible = val;
}
[[nodiscard]] auto getLineStartBlockNumber() const noexcept
{
return lineStartBlockNumber;
}
[[nodiscard]] auto getLineStartBlockPosition() const noexcept
{
return lineStartBlockPosition;
}
[[nodiscard]] const Item *getElement(unsigned int pos) const noexcept
{
unsigned int local_pos = 0;
for (auto &el : lineContent) {
local_pos += el->getTextLength();
if (local_pos >= pos) {
return el;
}
}
return nullptr;
}
[[nodiscard]] int32_t getX() const noexcept
{
return lineContent.front()->area().pos(Axis::X);
}
void setPosition(const short &x, const short &y);
void setParent(Item *parent);
[[nodiscard]] Length getWidth() const;
[[nodiscard]] Length getWidthTo(unsigned int pos) const;
void erase();
/// align TextLine due to alignment axis in parent_width
///
/// moves Text parts in Text. To not call n times callbacks on resize, call prior to setting parent
void alignH(Alignment align, Length parent_length) const;
void alignV(Alignment align, Length parent_length, Length lines_height);
[[nodiscard]] auto getText(unsigned int pos) const -> UTF8;
};
} // namespace gui