256 lines
7.8 KiB
C++
256 lines
7.8 KiB
C++
#include <chrono>
|
|
#include <ftxui/dom/elements.hpp>
|
|
#include <ftxui/screen/color.hpp>
|
|
#include <ftxui/screen/screen.hpp>
|
|
#include <future>
|
|
#include <string>
|
|
#include <utility>
|
|
#include <variant>
|
|
|
|
#include "config/config.h"
|
|
#include "core/system_data.h"
|
|
#include "os/os.h"
|
|
|
|
namespace ui {
|
|
using ftxui::Color;
|
|
|
|
static constexpr inline bool SHOW_ICONS = true;
|
|
static constexpr i32 MAX_PARAGRAPH_LENGTH = 30;
|
|
|
|
// Color themes
|
|
struct Theme {
|
|
Color::Palette16 icon;
|
|
Color::Palette16 label;
|
|
Color::Palette16 value;
|
|
Color::Palette16 border;
|
|
};
|
|
|
|
static constexpr Theme DEFAULT_THEME = {
|
|
.icon = Color::Cyan,
|
|
.label = Color::Yellow,
|
|
.value = Color::White,
|
|
.border = Color::GrayLight,
|
|
};
|
|
|
|
struct Icons {
|
|
StringView user;
|
|
StringView palette;
|
|
StringView calendar;
|
|
StringView host;
|
|
StringView kernel;
|
|
StringView os;
|
|
StringView memory;
|
|
StringView weather;
|
|
StringView music;
|
|
StringView disk;
|
|
StringView shell;
|
|
StringView desktop;
|
|
StringView window_manager;
|
|
};
|
|
|
|
static constexpr Icons EMPTY_ICONS = {
|
|
.user = "",
|
|
.palette = "",
|
|
.calendar = "",
|
|
.host = "",
|
|
.kernel = "",
|
|
.os = "",
|
|
.memory = "",
|
|
.weather = "",
|
|
.music = "",
|
|
.disk = "",
|
|
.shell = "",
|
|
.desktop = "",
|
|
.window_manager = "",
|
|
};
|
|
|
|
static constexpr Icons NERD_ICONS = {
|
|
.user = " ",
|
|
.palette = " ",
|
|
.calendar = " ",
|
|
.host = " ",
|
|
.kernel = " ",
|
|
.os = " ",
|
|
.memory = " ",
|
|
.weather = " ",
|
|
.music = " ",
|
|
.disk = " ",
|
|
.shell = " ",
|
|
.desktop = " ",
|
|
.window_manager = " ",
|
|
};
|
|
}
|
|
|
|
namespace {
|
|
using namespace ftxui;
|
|
|
|
fn CreateColorCircles() -> Element {
|
|
return hbox(
|
|
std::views::iota(0, 16) | std::views::transform([](i32 colorIndex) {
|
|
return hbox({ text("◯") | bold | color(static_cast<Color::Palette256>(colorIndex)), text(" ") });
|
|
}) |
|
|
std::ranges::to<Elements>()
|
|
);
|
|
}
|
|
|
|
fn SystemInfoBox(const Config& config, const SystemData& data) -> Element {
|
|
const String& name = config.general.name;
|
|
const Weather weather = config.weather;
|
|
const bool nowPlayingEnabled = config.now_playing.enabled;
|
|
|
|
const auto& [userIcon, paletteIcon, calendarIcon, hostIcon, kernelIcon, osIcon, memoryIcon, weatherIcon, musicIcon, diskIcon, shellIcon, deIcon, wmIcon] =
|
|
ui::SHOW_ICONS ? ui::NERD_ICONS : ui::EMPTY_ICONS;
|
|
|
|
Elements content;
|
|
|
|
content.push_back(text(String(userIcon) + "Hello " + name + "! ") | bold | color(Color::Cyan));
|
|
content.push_back(separator() | color(ui::DEFAULT_THEME.border));
|
|
content.push_back(hbox(
|
|
{
|
|
text(String(paletteIcon)) | color(ui::DEFAULT_THEME.icon),
|
|
CreateColorCircles(),
|
|
}
|
|
));
|
|
content.push_back(separator() | color(ui::DEFAULT_THEME.border));
|
|
|
|
// Helper function for aligned rows
|
|
fn createRow = [&](const auto& icon, const auto& label, const auto& value) {
|
|
return hbox(
|
|
{
|
|
text(String(icon)) | color(ui::DEFAULT_THEME.icon),
|
|
text(String(static_cast<CStr>(label))) | color(ui::DEFAULT_THEME.label),
|
|
filler(),
|
|
text(String(value)) | color(ui::DEFAULT_THEME.value),
|
|
text(" "),
|
|
}
|
|
);
|
|
};
|
|
|
|
// System info rows
|
|
content.push_back(createRow(calendarIcon, "Date", data.date));
|
|
|
|
// Weather row
|
|
if (weather.enabled && data.weather_info.has_value()) {
|
|
const WeatherOutput& weatherInfo = data.weather_info.value();
|
|
|
|
if (weather.show_town_name)
|
|
content.push_back(hbox(
|
|
{
|
|
text(String(weatherIcon)) | color(ui::DEFAULT_THEME.icon),
|
|
text("Weather") | color(ui::DEFAULT_THEME.label),
|
|
filler(),
|
|
|
|
hbox(
|
|
{
|
|
text(std::format("{}°F ", std::lround(weatherInfo.main.temp))),
|
|
text("in "),
|
|
text(weatherInfo.name),
|
|
text(" "),
|
|
}
|
|
) |
|
|
color(ui::DEFAULT_THEME.value),
|
|
}
|
|
));
|
|
else
|
|
content.push_back(hbox(
|
|
{
|
|
text(String(weatherIcon)) | color(ui::DEFAULT_THEME.icon),
|
|
text("Weather") | color(ui::DEFAULT_THEME.label),
|
|
filler(),
|
|
|
|
hbox(
|
|
{
|
|
text(std::format("{}°F, {}", std::lround(weatherInfo.main.temp), weatherInfo.weather[0].description)),
|
|
text(" "),
|
|
}
|
|
) |
|
|
color(ui::DEFAULT_THEME.value),
|
|
}
|
|
));
|
|
}
|
|
|
|
content.push_back(separator() | color(ui::DEFAULT_THEME.border));
|
|
|
|
if (!data.host.empty())
|
|
content.push_back(createRow(hostIcon, "Host", data.host));
|
|
|
|
if (!data.kernel_version.empty())
|
|
content.push_back(createRow(kernelIcon, "Kernel", data.kernel_version));
|
|
|
|
if (data.os_version)
|
|
content.push_back(createRow(String(osIcon), "OS", *data.os_version));
|
|
else
|
|
ERROR_LOG("Failed to get OS version: {}", data.os_version.error());
|
|
|
|
if (data.mem_info)
|
|
content.push_back(createRow(memoryIcon, "RAM", std::format("{}", BytesToGiB { *data.mem_info })));
|
|
else
|
|
ERROR_LOG("Failed to get memory info: {}", data.mem_info.error());
|
|
|
|
// Add Disk usage row
|
|
content.push_back(
|
|
createRow(diskIcon, "Disk", std::format("{}/{}", BytesToGiB { data.disk_used }, BytesToGiB { data.disk_total }))
|
|
);
|
|
|
|
content.push_back(createRow(shellIcon, "Shell", data.shell));
|
|
|
|
content.push_back(separator() | color(ui::DEFAULT_THEME.border));
|
|
|
|
if (data.desktop_environment && *data.desktop_environment != data.window_manager)
|
|
content.push_back(createRow(deIcon, "DE", *data.desktop_environment));
|
|
|
|
if (data.window_manager)
|
|
content.push_back(createRow(wmIcon, "WM", *data.window_manager));
|
|
|
|
// Now Playing row
|
|
if (nowPlayingEnabled && data.now_playing) {
|
|
if (const Result<String, NowPlayingError>& nowPlayingResult = *data.now_playing; nowPlayingResult.has_value()) {
|
|
const String& npText = *nowPlayingResult;
|
|
|
|
content.push_back(separator() | color(ui::DEFAULT_THEME.border));
|
|
content.push_back(hbox(
|
|
{
|
|
text(String(musicIcon)) | color(ui::DEFAULT_THEME.icon),
|
|
text("Playing") | color(ui::DEFAULT_THEME.label),
|
|
text(" "),
|
|
filler(),
|
|
paragraph(npText) | color(Color::Magenta) | size(WIDTH, LESS_THAN, ui::MAX_PARAGRAPH_LENGTH),
|
|
text(" "),
|
|
}
|
|
));
|
|
} else {
|
|
const NowPlayingError& error = nowPlayingResult.error();
|
|
|
|
if (std::holds_alternative<NowPlayingCode>(error))
|
|
switch (std::get<NowPlayingCode>(error)) {
|
|
case NowPlayingCode::NoPlayers: DEBUG_LOG("No players found"); break;
|
|
case NowPlayingCode::NoActivePlayer: DEBUG_LOG("No active player found"); break;
|
|
default: std::unreachable();
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
if (std::holds_alternative<WindowsError>(error))
|
|
DEBUG_LOG("WinRT error: {}", to_string(std::get<WindowsError>(error).message()));
|
|
#else
|
|
if (std::holds_alternative<String>(error))
|
|
DEBUG_LOG("NowPlaying error: {}", std::get<String>(error));
|
|
#endif
|
|
}
|
|
}
|
|
|
|
return vbox(content) | borderRounded | color(Color::White);
|
|
}
|
|
}
|
|
|
|
fn main() -> i32 {
|
|
const Config& config = Config::getInstance();
|
|
const SystemData data = SystemData::fetchSystemData(config);
|
|
|
|
Element document = vbox({ hbox({ SystemInfoBox(config, data), filler() }), text("") });
|
|
|
|
Screen screen = Screen::Create(Dimension::Full(), Dimension::Fit(document));
|
|
Render(screen, document);
|
|
screen.Print();
|
|
|
|
return 0;
|
|
}
|