BLT/src/blt/std/format.cpp

387 lines
13 KiB
C++
Raw Normal View History

/*
* Created by Brett on 26/01/23.
* Licensed under GNU General Public License V3.0
* See LICENSE file for license detail
*/
#include <blt/std/format.h>
2023-11-30 19:56:45 -05:00
#include <blt/std/string.h>
#include <cmath>
#include "blt/std/logging.h"
2023-11-30 17:17:53 -05:00
#include "blt/std/assert.h"
#include <stack>
#include <queue>
2023-11-30 17:17:53 -05:00
#include <algorithm>
std::vector<std::string> blt::string::TableFormatter::createTable(bool top, bool bottom)
{
std::vector<std::string> table;
const auto& tableHeader = generateColumnHeader();
const auto& topSeparator = generateTopSeparator(tableHeader.size());
2023-03-03 10:41:38 -05:00
const auto& lineSeparator = generateSeparator(tableHeader.size() - 1);
if (top)
table.push_back(topSeparator);
table.push_back(tableHeader);
table.push_back(lineSeparator);
for (const auto& row : rows)
{
std::string rowString = "|";
for (unsigned int i = 0; i < row.rowValues.size(); i++)
{
const auto& rowValue = row.rowValues[i];
const auto& column = columns[i];
const int spaceLeft = int(column.maxColumnLength) - int(rowValue.size());
// we want to prefer putting the space on the right size, flooring left and ceiling right ensures this.
rowString += createPadding((int) std::floor(spaceLeft / 2.0) + m_columnPadding);
rowString += rowValue;
rowString += createPadding((int) std::ceil(spaceLeft / 2.0) + m_columnPadding);
rowString += "|";
}
table.push_back(rowString);
}
if (bottom)
table.push_back(lineSeparator);
return table;
}
std::string blt::string::TableFormatter::generateColumnHeader()
{
updateMaxColumnLengths();
std::string header = "|";
for (unsigned int i = 0; i < columns.size(); i++)
{
const auto& column = columns[i];
auto columnPaddingLength = (int(column.maxColumnLength) - int(column.columnName.size())) / 2.0;
header += createPadding(int(m_columnPadding + (int) std::floor(columnPaddingLength)));
header += column.columnName;
header += createPadding(int(m_columnPadding + (int) std::ceil(columnPaddingLength)));
if (i < columns.size() - 1)
header += "|";
}
header += "|";
return header;
}
std::string blt::string::TableFormatter::generateTopSeparator(size_t size)
{
2023-01-29 17:10:36 -05:00
auto sizeOfName = m_tableName.empty() ? 0 : m_tableName.size() + 4;
auto sizeNameRemoved = size - sizeOfName;
std::string halfWidthLeftSeparator;
std::string halfWidthRightSeparator;
auto sizeNameFloor = (size_t) std::floor((double) sizeNameRemoved / 2.0);
auto sizeNameCeil = (size_t) std::ceil((double) sizeNameRemoved / 2.0);
2023-01-29 17:10:36 -05:00
halfWidthLeftSeparator.reserve(sizeNameCeil);
halfWidthRightSeparator.reserve(sizeNameFloor);
halfWidthLeftSeparator += "+";
2023-04-05 17:21:19 -04:00
for (unsigned int i = 0; i < sizeNameFloor - 1; i++)
2023-01-29 17:10:36 -05:00
halfWidthLeftSeparator += "-";
2023-04-05 17:21:19 -04:00
for (unsigned int i = 0; i < sizeNameCeil - 1; i++)
2023-01-29 17:10:36 -05:00
halfWidthRightSeparator += "-";
halfWidthRightSeparator += "+";
2023-01-29 17:10:36 -05:00
std::string separator;
separator += halfWidthLeftSeparator;
if (sizeOfName != 0)
{
2023-01-29 17:10:36 -05:00
separator += "{ ";
separator += m_tableName;
separator += " }";
}
separator += halfWidthRightSeparator;
return separator;
}
std::string blt::string::TableFormatter::generateSeparator(size_t size)
{
size_t nextIndex = 0;
size_t currentColumnIndex = 0;
std::string wholeWidthSeparator;
for (unsigned int i = 0; i < size; i++)
{
if (i == nextIndex)
{
auto currentColumnSize = columns[currentColumnIndex++].maxColumnLength + m_columnPadding * 2;
2023-03-03 10:41:38 -05:00
nextIndex += currentColumnSize + 1;
wholeWidthSeparator += "+";
} else
wholeWidthSeparator += "-";
}
2023-03-03 10:42:38 -05:00
wholeWidthSeparator += "+";
return wholeWidthSeparator;
}
void blt::string::TableFormatter::updateMaxColumnLengths()
{
for (unsigned int i = 0; i < columns.size(); i++)
{
auto& column = columns[i];
column.maxColumnLength = column.columnName.size();
for (const auto& row : rows)
{
column.maxColumnLength = std::max(column.maxColumnLength, row.rowValues[i].size());
}
}
}
2023-01-29 23:36:42 -05:00
/*
* -----------------------
* Tree Formatter
* -----------------------
*/
struct node_data
{
blt::string::TreeFormatter::Node* node;
2023-11-30 17:17:53 -05:00
std::vector<std::string> box;
size_t level;
node_data(blt::string::TreeFormatter::Node* node, size_t level): node(node), level(level)
{}
};
2023-11-30 17:17:53 -05:00
struct level_data
{
std::vector<node_data> level;
size_t count = 0;
size_t depth = 0;
size_t max_horizontal_length = 0;
};
std::vector<std::string> blt::string::TreeFormatter::construct()
{
2023-11-30 17:17:53 -05:00
std::stack<level_data> levels;
std::queue<node_data> bfs;
bfs.emplace(root, 0);
2023-11-30 17:17:53 -05:00
// construct a stack of the highest node -> the lowest node.
level_data currentLevel;
currentLevel.depth = 0;
2023-11-30 19:56:45 -05:00
size_t maxLineLength = 0;
while (!bfs.empty())
{
auto n = bfs.front();
2023-11-30 17:17:53 -05:00
if (currentLevel.depth != n.level)
{
levels.push(currentLevel);
currentLevel = {};
}
currentLevel.count++;
if (n.node != nullptr)
{
auto box = generateBox(n.node);
currentLevel.max_horizontal_length = std::max(currentLevel.max_horizontal_length, box[0].size());
2023-11-30 19:56:45 -05:00
std::string replacement = "-";
// UGLY TODO: fix this
if (n.node->left != nullptr)
replacement = "$";
else if (n.node->right != nullptr)
replacement = "#";
if (replacement[0] == '$' && n.node->right != nullptr)
replacement = "@";
blt::string::replaceAll(box.front(), "%", replacement);
2023-11-30 17:17:53 -05:00
n.box = std::move(box);
}
currentLevel.level.push_back(n);
currentLevel.depth = n.level;
bfs.pop();
//std::cout << "Node at level " << n.level << " " << n.node << "\n";
if (n.node == nullptr)
continue;
if (n.node->left != nullptr)
bfs.emplace(n.node->left, n.level + 1);
2023-11-30 17:17:53 -05:00
else
bfs.emplace(nullptr, n.level + 1);
if (n.node->right != nullptr)
bfs.emplace(n.node->right, n.level + 1);
2023-11-30 17:17:53 -05:00
else
bfs.emplace(nullptr, n.level + 1);
}
std::vector<std::string> lines;
size_t lineLength = 0;
const size_t lineHeight = format.verticalPadding * 2 + 3;
//std::cout << levels.size() << "\n";
2023-11-30 19:56:45 -05:00
const size_t verticalSpacing = format.verticalSpacing % 2 == 0 ? format.verticalSpacing + 1 : format.verticalSpacing;
2023-11-30 17:17:53 -05:00
while (!levels.empty())
{
std::vector<std::string> currentLines;
const auto& n = levels.top();
for (const auto& b : n.level)
{
std::vector<std::string> box = b.box;
if (b.node == nullptr || box.empty())
{
for (size_t i = 0; i < lineHeight; i++)
box.push_back(createPadding(n.max_horizontal_length));
}
if (currentLines.empty())
{
for (const auto& v : box)
currentLines.push_back(v);
} else
{
BLT_ASSERT(currentLines.size() == box.size() && "Box lines should match current lines!");
for (size_t i = 0; i < currentLines.size(); i++)
{
currentLines[i] += createPadding(format.horizontalSpacing);
currentLines[i] += box[i];
}
}
}
std::int64_t padLength = (static_cast<std::int64_t>(lineLength) - static_cast<std::int64_t>(currentLines[0].length())) / 2;
if (padLength < 0)
padLength = 0;
for (const auto& v : currentLines)
{
lineLength = std::max(lineLength, v.length());
lines.push_back(createPadding(padLength) + v);
2023-11-30 19:56:45 -05:00
maxLineLength = std::max(maxLineLength, lines.back().length());
2023-11-30 17:17:53 -05:00
}
levels.pop();
2023-11-30 19:56:45 -05:00
if (!levels.empty())
for (size_t i = 0; i < verticalSpacing; i++)
lines.emplace_back(" ");
2023-11-30 17:17:53 -05:00
}
2023-11-30 19:56:45 -05:00
for (auto& line : lines)
if (line.length() < maxLineLength)
line += createPadding(maxLineLength - line.length());
std::reverse(lines.begin(), lines.end());
2023-11-30 17:17:53 -05:00
size_t index = 1;
2023-11-30 19:56:45 -05:00
while (index < lines.size())
2023-11-30 17:17:53 -05:00
{
2023-11-30 19:56:45 -05:00
if (auto poses = blt::string::containsAll(lines[index], std::unordered_set{'$', '#', '@'}))
2023-11-30 17:17:53 -05:00
{
2023-11-30 19:56:45 -05:00
const auto& nextLine = lines[index + verticalSpacing + 1];
auto poses2 = blt::string::containsAll(nextLine, std::unordered_set{'%'}).value();
size_t consume = 0;
for (auto p : poses.value())
2023-11-30 17:17:53 -05:00
{
2023-11-30 19:56:45 -05:00
char type = lines[index][p];
for (size_t n = 0; n < verticalSpacing / 2; n++)
lines[index + n + 1][p] = '|';
auto start = index + (verticalSpacing / 2) + 1;
switch (type)
2023-11-30 17:17:53 -05:00
{
2023-11-30 19:56:45 -05:00
case '@':
for (size_t i = poses2[consume] + 1; i < poses2[consume + 1]; i++)
lines[start][i] = '-';
lines[start][poses2[consume]] = '+';
lines[start][poses2[consume + 1]] = '+';
for (size_t n = 0; n < verticalSpacing / 2; n++)
{
lines[start + n + 1][poses2[consume]] = '|';
lines[start + n + 1][poses2[consume + 1]] = '|';
}
consume += 2;
break;
case '$':
for (size_t i = poses2[consume] + 1; i < p; i++)
lines[start][i] = '-';
lines[start][poses2[consume]] = '+';
for (size_t n = 0; n < verticalSpacing / 2; n++)
lines[start + n + 1][poses2[consume]] = '|';
consume++;
break;
case '#':
for (size_t i = p; i < poses2[consume]; i++)
lines[start][i] = '-';
lines[start][poses2[consume]] = '+';
for (size_t n = 0; n < verticalSpacing / 2; n++)
lines[start + n + 1][poses2[consume]] = '|';
consume++;
break;
default:
break;
2023-11-30 17:17:53 -05:00
}
2023-11-30 20:14:18 -05:00
lines[start][p] = '+';
2023-11-30 17:17:53 -05:00
}
}
2023-11-30 19:56:45 -05:00
blt::string::replaceAll(lines[index], "%", "+");
blt::string::replaceAll(lines[index], "#", "+");
blt::string::replaceAll(lines[index], "@", "+");
blt::string::replaceAll(lines[index], "$", "+");
index++;
}
2023-11-30 19:56:45 -05:00
blt::string::replaceAll(lines.front(), "%", "-");
return lines;
}
std::vector<std::string> blt::string::TreeFormatter::generateBox(blt::string::TreeFormatter::Node* node) const
{
if (node == nullptr)
return {};
std::vector<std::string> lines;
auto& data = node->data;
auto dataLength = data.length();
auto paddingLeft = format.horizontalPadding;
auto paddingRight = format.horizontalPadding;
auto totalLength = paddingLeft + paddingRight + dataLength;
if (totalLength % 2 != 0)
{
paddingRight += 1;
totalLength += 1;
}
// create horizontal line based on the calculated length
std::string hline = createLine(totalLength, '+', '-');
2023-11-30 17:17:53 -05:00
hline[(hline.size() - 1) / 2] = '%';
std::string paddedLine = createLine(totalLength, '|', ' ');
std::string dataStr;
dataStr.reserve(totalLength);
dataStr += '|';
dataStr += createPadding(paddingLeft - 1);
dataStr += data;
dataStr += createPadding(paddingRight - 1);
dataStr += '|';
lines.push_back(hline);
for (int i = 0; i < format.verticalPadding; i++)
lines.push_back(paddedLine);
lines.push_back(std::move(dataStr));
for (int i = 0; i < format.verticalPadding; i++)
lines.push_back(paddedLine);
lines.push_back(std::move(hline));
return lines;
}
std::string blt::string::createLine(size_t totalLength, char endingChar, char spacingChar)
{
std::string line;
line.reserve(totalLength);
line += endingChar;
line += createPadding(totalLength - 2, spacingChar);
line += endingChar;
return line;
}
std::string blt::string::createPadding(size_t length, char spacing)
{
std::string padding;
padding.reserve(length);
for (size_t i = 0; i < length; i++)
padding += spacing;
return padding;
}