fix(cpp/viz,core): bell icon TI_BELL, candlestick Setup-inside-BeginPlot, pie legend, kpi sparkline a la derecha
- toast.cpp: TI_BELL en lugar de \xf0\x9f\x94\x94 (fuera del rango cargado por icon_font, renderizaba como ?) - candlestick.cpp: SetupAxes/SetupAxisScale/SetupAxisLimits movidos dentro de BeginPlot/EndPlot — antes el plot desaparecia al entrar - pie_chart.cpp: SetupLegend(East, Outside, NoButtons), eliminado NoLegend - kpi_card.cpp: layout 2 cols con sparkline a la derecha centrado verticalmente
This commit is contained in:
@@ -1,5 +1,6 @@
|
|||||||
#include "core/toast.h"
|
#include "core/toast.h"
|
||||||
#include "core/tokens.h"
|
#include "core/tokens.h"
|
||||||
|
#include "core/icons_tabler.h"
|
||||||
#include <imgui.h>
|
#include <imgui.h>
|
||||||
#include <chrono>
|
#include <chrono>
|
||||||
#include <deque>
|
#include <deque>
|
||||||
@@ -171,10 +172,11 @@ void toast_inbox_button(const char* id) {
|
|||||||
ImGui::PushStyleVar(ImGuiStyleVar_FrameRounding, radius::sm);
|
ImGui::PushStyleVar(ImGuiStyleVar_FrameRounding, radius::sm);
|
||||||
ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(spacing::xs, spacing::xs));
|
ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(spacing::xs, spacing::xs));
|
||||||
|
|
||||||
// Label: campana UTF-8 (U+1F514 = bell). Si la fuente no lo tiene,
|
// Label: TI_BELL (Tabler icon, dentro del rango 0xE000-0xFCFF cargado por
|
||||||
// fallback a "!".
|
// icon_font.cpp). Antes usabamos un emoji U+1F514 (4-byte UTF-8) que cae
|
||||||
|
// fuera del rango y se renderizaba como '?'.
|
||||||
char label[64];
|
char label[64];
|
||||||
std::snprintf(label, sizeof(label), "\xf0\x9f\x94\x94%s", id);
|
std::snprintf(label, sizeof(label), TI_BELL "%s", id);
|
||||||
|
|
||||||
ImVec2 btn_pos = ImGui::GetCursorScreenPos();
|
ImVec2 btn_pos = ImGui::GetCursorScreenPos();
|
||||||
const bool clicked = ImGui::Button(label, ImVec2(28.0f, 28.0f));
|
const bool clicked = ImGui::Button(label, ImVec2(28.0f, 28.0f));
|
||||||
|
|||||||
@@ -12,16 +12,19 @@ void candlestick(const char* title, const double* dates, const double* opens,
|
|||||||
double spacing = (count > 1) ? (dates[1] - dates[0]) : 1.0;
|
double spacing = (count > 1) ? (dates[1] - dates[0]) : 1.0;
|
||||||
double half_w = spacing * (double)width_percent * 0.5;
|
double half_w = spacing * (double)width_percent * 0.5;
|
||||||
|
|
||||||
ImPlot::SetupAxes("Date", "Price", ImPlotAxisFlags_None, ImPlotAxisFlags_AutoFit);
|
|
||||||
ImPlot::SetupAxisScale(ImAxis_X1, ImPlotScale_Time);
|
|
||||||
|
|
||||||
// Auto-fit X axis to the data range with a small margin.
|
|
||||||
ImPlot::SetupAxisLimits(ImAxis_X1,
|
|
||||||
dates[0] - spacing,
|
|
||||||
dates[count - 1] + spacing,
|
|
||||||
ImGuiCond_Always);
|
|
||||||
|
|
||||||
if (ImPlot::BeginPlot(title, ImVec2(-1, 0))) {
|
if (ImPlot::BeginPlot(title, ImVec2(-1, 0))) {
|
||||||
|
// ImPlot Setup* calls DEBEN ir dentro de BeginPlot/EndPlot. Antes
|
||||||
|
// estaban fuera y dejaban el plot sin ejes, lo que hacia que el demo
|
||||||
|
// de candlestick "desapareciera" al entrar en la galeria.
|
||||||
|
ImPlot::SetupAxes("Date", "Price",
|
||||||
|
ImPlotAxisFlags_None,
|
||||||
|
ImPlotAxisFlags_AutoFit);
|
||||||
|
ImPlot::SetupAxisScale(ImAxis_X1, ImPlotScale_Time);
|
||||||
|
ImPlot::SetupAxisLimits(ImAxis_X1,
|
||||||
|
dates[0] - spacing,
|
||||||
|
dates[count - 1] + spacing,
|
||||||
|
ImGuiCond_Always);
|
||||||
|
|
||||||
ImDrawList* draw = ImPlot::GetPlotDrawList();
|
ImDrawList* draw = ImPlot::GetPlotDrawList();
|
||||||
|
|
||||||
const ImU32 col_bull = IM_COL32(0, 200, 80, 255); // green — close >= open
|
const ImU32 col_bull = IM_COL32(0, 200, 80, 255); // green — close >= open
|
||||||
|
|||||||
@@ -36,6 +36,20 @@ void kpi_card(const char* label, float value, float delta_percent,
|
|||||||
ImGuiChildFlags_Borders,
|
ImGuiChildFlags_Borders,
|
||||||
ImGuiWindowFlags_NoScrollbar | ImGuiWindowFlags_NoScrollWithMouse);
|
ImGuiWindowFlags_NoScrollbar | ImGuiWindowFlags_NoScrollWithMouse);
|
||||||
|
|
||||||
|
const bool has_history = history != nullptr && history_count > 0;
|
||||||
|
const bool has_delta = delta_percent != 0.0f;
|
||||||
|
|
||||||
|
// Layout de dos columnas: izquierda info (label, value, delta), derecha sparkline.
|
||||||
|
// El sparkline se sitia verticalmente centrado a la derecha de la card.
|
||||||
|
constexpr float spark_w = 100.0f;
|
||||||
|
constexpr float spark_h = 36.0f;
|
||||||
|
|
||||||
|
const float inner_w = ImGui::GetContentRegionAvail().x;
|
||||||
|
const float info_w = has_history ? (inner_w - spark_w - spacing::sm) : inner_w;
|
||||||
|
|
||||||
|
ImGui::BeginGroup();
|
||||||
|
ImGui::PushItemWidth(info_w);
|
||||||
|
|
||||||
// Top row: optional icon + label, ambos en text_muted.
|
// Top row: optional icon + label, ambos en text_muted.
|
||||||
ImGui::PushStyleColor(ImGuiCol_Text, colors::text_muted);
|
ImGui::PushStyleColor(ImGuiCol_Text, colors::text_muted);
|
||||||
if (icon && *icon) {
|
if (icon && *icon) {
|
||||||
@@ -53,9 +67,6 @@ void kpi_card(const char* label, float value, float delta_percent,
|
|||||||
ImGui::SetWindowFontScale(1.0f);
|
ImGui::SetWindowFontScale(1.0f);
|
||||||
|
|
||||||
// Delta / trend — SIEMPRE se reserva la linea aunque no haya tendencia.
|
// Delta / trend — SIEMPRE se reserva la linea aunque no haya tendencia.
|
||||||
const bool has_delta = delta_percent != 0.0f;
|
|
||||||
const bool has_history = history != nullptr && history_count > 0;
|
|
||||||
|
|
||||||
if (has_delta) {
|
if (has_delta) {
|
||||||
const bool positive = delta_percent >= 0.0f;
|
const bool positive = delta_percent >= 0.0f;
|
||||||
const ImVec4 delta_color = positive ? colors::success : colors::error;
|
const ImVec4 delta_color = positive ? colors::success : colors::error;
|
||||||
@@ -68,12 +79,6 @@ void kpi_card(const char* label, float value, float delta_percent,
|
|||||||
ImGui::PushStyleColor(ImGuiCol_Text, delta_color);
|
ImGui::PushStyleColor(ImGuiCol_Text, delta_color);
|
||||||
ImGui::TextUnformatted(delta_buf);
|
ImGui::TextUnformatted(delta_buf);
|
||||||
ImGui::PopStyleColor();
|
ImGui::PopStyleColor();
|
||||||
|
|
||||||
if (has_history) {
|
|
||||||
sparkline(label, history, history_count, delta_color, 120.0f, 24.0f);
|
|
||||||
}
|
|
||||||
} else if (has_history) {
|
|
||||||
sparkline(label, history, history_count, colors::primary, 120.0f, 24.0f);
|
|
||||||
} else {
|
} else {
|
||||||
// Placeholder para preservar altura.
|
// Placeholder para preservar altura.
|
||||||
ImGui::PushStyleColor(ImGuiCol_Text, colors::text_dim);
|
ImGui::PushStyleColor(ImGuiCol_Text, colors::text_dim);
|
||||||
@@ -81,6 +86,29 @@ void kpi_card(const char* label, float value, float delta_percent,
|
|||||||
ImGui::PopStyleColor();
|
ImGui::PopStyleColor();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ImGui::PopItemWidth();
|
||||||
|
ImGui::EndGroup();
|
||||||
|
|
||||||
|
// Sparkline a la derecha, centrado verticalmente respecto a la card.
|
||||||
|
if (has_history) {
|
||||||
|
const ImVec4 spark_color = has_delta
|
||||||
|
? (delta_percent >= 0.0f ? colors::success : colors::error)
|
||||||
|
: colors::primary;
|
||||||
|
|
||||||
|
ImGui::SameLine();
|
||||||
|
|
||||||
|
// Centrar verticalmente: la card mide card_height, el sparkline spark_h.
|
||||||
|
// Restamos el padding interno (spacing::sm) ya consumido al inicio.
|
||||||
|
const float card_inner_h = card_height - 2.0f * spacing::sm;
|
||||||
|
float y_offset = (card_inner_h - spark_h) * 0.5f;
|
||||||
|
if (y_offset < 0.0f) y_offset = 0.0f;
|
||||||
|
// Anclamos el sparkline al borde derecho.
|
||||||
|
const float spark_x = inner_w - spark_w;
|
||||||
|
ImGui::SetCursorPos(ImVec2(spacing::sm + spark_x,
|
||||||
|
spacing::sm + y_offset));
|
||||||
|
sparkline(label, history, history_count, spark_color, spark_w, spark_h);
|
||||||
|
}
|
||||||
|
|
||||||
ImGui::EndChild();
|
ImGui::EndChild();
|
||||||
|
|
||||||
ImGui::PopStyleVar(3);
|
ImGui::PopStyleVar(3);
|
||||||
|
|||||||
@@ -52,9 +52,10 @@ void draw_pie(const char* title, const char* const* labels, const T* values,
|
|||||||
|
|
||||||
const ImVec2 plot_size(-1.0f, height > 0.0f ? height : 200.0f);
|
const ImVec2 plot_size(-1.0f, height > 0.0f ? height : 200.0f);
|
||||||
|
|
||||||
|
// Legend SIEMPRE visible para que se distingan las slices por color (los
|
||||||
|
// pies suelen tener muchas categorias).
|
||||||
const ImPlotFlags plot_flags = plot_static::kPlotFlags
|
const ImPlotFlags plot_flags = plot_static::kPlotFlags
|
||||||
| ImPlotFlags_Equal
|
| ImPlotFlags_Equal;
|
||||||
| ImPlotFlags_NoLegend;
|
|
||||||
|
|
||||||
if (ImPlot::BeginPlot(title, plot_size, plot_flags)) {
|
if (ImPlot::BeginPlot(title, plot_size, plot_flags)) {
|
||||||
ImPlot::SetupAxes(nullptr, nullptr,
|
ImPlot::SetupAxes(nullptr, nullptr,
|
||||||
@@ -62,6 +63,8 @@ void draw_pie(const char* title, const char* const* labels, const T* values,
|
|||||||
plot_static::kAxisFlagsHidden);
|
plot_static::kAxisFlagsHidden);
|
||||||
ImPlot::SetupAxisLimits(ImAxis_X1, 0.0, 1.0, ImPlotCond_Always);
|
ImPlot::SetupAxisLimits(ImAxis_X1, 0.0, 1.0, ImPlotCond_Always);
|
||||||
ImPlot::SetupAxisLimits(ImAxis_Y1, 0.0, 1.0, ImPlotCond_Always);
|
ImPlot::SetupAxisLimits(ImAxis_Y1, 0.0, 1.0, ImPlotCond_Always);
|
||||||
|
ImPlot::SetupLegend(ImPlotLocation_East,
|
||||||
|
ImPlotLegendFlags_Outside | ImPlotLegendFlags_NoButtons);
|
||||||
|
|
||||||
ImPlot::PlotPieChart(labels, values, count, 0.5, 0.5, outer, "%.0f", 90.0);
|
ImPlot::PlotPieChart(labels, values, count, 0.5, 0.5, outer, "%.0f", 90.0);
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user