If Amazon used this Besos wouldn't have banned PowerPoint.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 
besos-loves-slides/src/slides_ui.cpp

206 lines
5.2 KiB

#include "guecs/sfml/components.hpp"
#include "guecs/ui.hpp"
#include <fmt/xchar.h>
#include <deque>
#include <iostream>
#include <nlohmann/json.hpp>
#include "dbc.hpp"
#include <memory>
#include "slides_ui.hpp"
using std::string, std::wstring, std::shared_ptr, std::make_shared;
using nlohmann::json;
Slide::Slide(const string& title, const string& content, json& config, json& deck_config) :
$title(guecs::to_wstring(title)),
$content(guecs::to_wstring(content)),
$config(config)
{
// first config the text with any font stuff from the deck
config_text($title_font, deck_config);
config_text($content_font, deck_config);
// then config it with the slide's config
config_text($title_font, $config);
config_text($content_font, $config);
}
void Slide::init(lel::Cell& cell) {
if(!$initialized) {
$initialized = true;
$gui.position(cell.x, cell.y, cell.w, cell.h);
$gui.layout(
"[=*%(300,200)title|_|_]"
"[_|_|_]"
"[*%(300,600)content|_|_]"
"[_|_|_]"
"[_|_|_]"
"[_|_|_]"
"[_|_|_]"
"[_|_|_]");
auto title = $gui.entity("title");
$gui.set<guecs::Text>(title, $title_font);
auto content = $gui.entity("content");
$gui.set<guecs::Text>(content, $content_font);
if($config.contains("image")) {
$gui.set<guecs::Sprite>(content, {$config["image"]});
}
$gui.init();
}
}
void Slide::config_text(guecs::Text &result, nlohmann::json& config) {
if(config.contains("font_size")) result.size = config["font_size"];
if(config.contains("font_color")) {
std::vector<uint8_t> color = config["font_color"];
dbc::check(color.size() == 4, "font_color on slide must have 4 values rgba");
fmt::println("font_color is {},{},{},{}", color[0], color[1], color[2], color[3]);
result.color = sf::Color{
color[0], color[1], color[2], color[3]};
}
if(config.contains("font_padding")) result.padding = config["font_padding"];
if(config.contains("font_centered")) result.centered = config["font_centered"];
}
void Slide::render(sf::RenderWindow& window) {
$gui.render(window);
// $gui.debug_layout(window);
}
SlidesUI::SlidesUI(shared_ptr<SlideDeck> deck) {
dbc::check(deck != nullptr, "deck is null");
dbc::check(deck->slides.size() > 0, "slide deck is empy");
$deck = deck;
$current = 0;
}
void SlidesUI::set_text(const std::string& name) {
if($deck->config.contains(name)) {
auto text = $gui.entity(name);
std::string content = $deck->config[name];
$gui.set<guecs::Text>(text, {
guecs::to_wstring(content),
guecs::THEME.TEXT_SIZE-10,
guecs::THEME.LIGHT_DARK,
});
}
}
void SlidesUI::init() {
$gui.position(0, 0, WINDOW_WIDTH, WINDOW_HEIGHT);
$gui.layout(
"[t_left|t_center|t_right]"
"[*%(300,500)slide|_|_|_|_]"
"[_|_|_|_|_]"
"[_|_|_|_|_]"
"[_|_|_|_|_]"
"[_|_|_|_|_]"
"[_]"
"[title|*%(200,100)description|_|_|_]"
);
set_text("title");
set_text("description");
guecs::Background bg{$gui.$parser};
bg.set_color(guecs::THEME.BG_COLOR_DARK);
$gui.set<guecs::Background>($gui.MAIN, bg);
show_slide();
$gui.init();
}
Slide& SlidesUI::current() {
return $deck->slides.at($current);
}
void SlidesUI::next_slide() {
if($current < $deck->slides.size() - 1) {
$current++;
show_slide();
}
}
void SlidesUI::prev_slide() {
if($current > 0) {
$current--;
show_slide();
}
}
void SlidesUI::set_slide(size_t index) {
$current = index < $deck->slides.size() ? index : 0;
show_slide();
}
void SlidesUI::show_slide() {
auto& slide = current();
// kind of have to set the background for the whole UI using the slide config
auto& bg = $gui.get<guecs::Background>($gui.MAIN);
sf::Color color = guecs::THEME.FILL_COLOR;
if(slide.$config.contains("bg_image")) {
bg.set_sprite(slide.$config["bg_image"], true);
} else if($deck->config.contains("bg_image")) {
bg.set_sprite($deck->config["bg_image"], true);
} else {
bg.sprite = nullptr;
bg.texture = nullptr;
}
if(slide.$config.contains("bg_color")) {
auto color_conf = slide.$config["bg_color"];
color = {color_conf[0], color_conf[1], color_conf[2], color_conf[3]};
} else if($deck->config.contains("bg_color")) {
auto color_conf = $deck->config["bg_color"];
color = {color_conf[0], color_conf[1], color_conf[2], color_conf[3]};
}
bg.set_color(color);
bg.init();
auto& cell = $gui.cell_for("slide");
slide.init(cell);
}
void SlidesUI::render(sf::RenderWindow& window) {
$gui.render(window);
auto& slide = current();
slide.render(window);
// $gui.debug_layout(window);
}
void SlidesUI::mouse(float x, float y, guecs::Modifiers mods) {
$gui.mouse(x, y, mods);
if(mods.test(guecs::ModBit::left)) {
next_slide();
} else if(mods.test(guecs::ModBit::right)) {
prev_slide();
}
}
void SlidesUI::handle_events(sf::RenderWindow& presenter, const sf::Event& event) {
if(const auto* mev = event.getIf<sf::Event::MouseButtonPressed>()) {
sf::Vector2f pos = presenter.mapPixelToCoords(mev->position);
if(mev->button == sf::Mouse::Button::Left) {
mouse(pos.x, pos.y, {1 << guecs::ModBit::left});
} else if(mev->button == sf::Mouse::Button::Right) {
mouse(pos.x, pos.y, {1 << guecs::ModBit::right});
}
}
}