#pragma once
/*
* Copyright (C) 2024 Brett Terpstra
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
#ifndef BLT_LOGGING_LOGGING_H
#define BLT_LOGGING_LOGGING_H
#include
#include
#include
#include
#include
namespace blt::logging
{
struct logger_t
{
explicit logger_t() = default;
template
std::string log(std::string fmt, Args&&... args)
{
compile(std::move(fmt));
auto sequence = std::make_integer_sequence{};
while (auto pair = consume_until_fmt())
{
auto [begin, end] = *pair;
if (end - begin > 0)
{
auto format_data = handle_fmt(m_fmt.substr(begin + 1, begin - end - 1));
auto [arg_pos, fmt_type] = format_data;
if (arg_pos == -1)
arg_pos = static_cast(m_arg_pos++);
if (fmt_type)
{
if (fmt_type == fmt_type_t::GENERAL)
{
apply_func([this](auto&& value)
{
if (static_cast(value) > 0xFFFFFFFFFul)
exponential();
else
fixed();
m_stream << std::forward(value);
}, arg_pos, sequence, std::forward(args)...);
} else if (fmt_type == fmt_type_t::CHAR)
{
} else if (fmt_type == fmt_type_t::BINARY)
{
}
}
else
{
apply_func([this](auto&& value)
{
m_stream << std::forward(value);
}, arg_pos, sequence, std::forward(args)...);
}
}
else
apply_func([this](auto&& value)
{
m_stream << std::forward(value);
}, m_arg_pos++, sequence, std::forward(args)...);
}
finish();
return to_string();
}
std::string to_string();
private:
template
void apply_func(const Func& func, const size_t arg, std::integer_sequence, Args&&... args)
{
((handle_func(func, arg, std::forward(args))), ...);
}
template
void handle_func(const Func& func, const size_t arg, T&& t)
{
if (index == arg)
func(std::forward(t));
}
[[nodiscard]] std::pair> handle_fmt(std::string_view fmt);
void exponential();
void fixed();
void compile(std::string fmt);
std::optional> consume_until_fmt();
void finish();
std::string m_fmt;
std::stringstream m_stream;
fmt_parser_t m_parser;
size_t m_last_fmt_pos = 0;
size_t m_arg_pos = 0;
};
void print(const std::string& fmt);
void newline();
logger_t& get_global_logger();
template
void print(std::string fmt, Args&&... args)
{
auto& logger = get_global_logger();
print(logger.log(std::move(fmt), std::forward(args)...));
}
template
void println(std::string fmt, Args&&... args)
{
print(std::move(fmt), std::forward(args)...);
newline();
}
}
#endif // BLT_LOGGING_LOGGING_H