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/main.cpp

250 lines
5.7 KiB

#include "guecs/sfml/backend.hpp"
#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 <filesystem>
#include <fstream>
#include <iostream>
int WINDOW_WIDTH=1280;
int WINDOW_HEIGHT=720;
constexpr const int FRAME_LIMIT=60;
constexpr const bool VSYNC=true;
constexpr const int TITLE_SIZE=84;
constexpr const int CONTENT_SIZE=56;
using std::string, std::wstring, std::shared_ptr, std::make_shared;
using nlohmann::json;
namespace fs = std::filesystem;
struct Slide {
guecs::UI $gui;
wstring $title;
wstring $content;
json $config;
bool $initialized = false;
Slide(const string& title, const string& content, json& config) :
$title(guecs::to_wstring(title)),
$content(guecs::to_wstring(content)),
$config(config)
{
}
void init(lel::Cell& cell) {
if(!$initialized) {
$initialized = true;
$gui.position(cell.x, cell.y, cell.w, cell.h);
$gui.layout(
"[=*%(300,200)title|_|_]"
"[_|_|_]"
"[=*%(300,300)content|_|_]"
"[_|_|_]"
"[_|_|_]");
auto title = $gui.entity("title");
$gui.set<guecs::Text>(title, {
$title, TITLE_SIZE});
auto content = $gui.entity("content");
$gui.set<guecs::Text>(content, {
$content,
CONTENT_SIZE, guecs::THEME.TEXT_COLOR, 20});
$gui.init();
}
}
void render(sf::RenderWindow& window) {
$gui.render(window);
// $gui.debug_layout(window);
}
};
using SlideDeck = std::vector<Slide>;
struct SlidesUI {
guecs::UI $gui;
shared_ptr<SlideDeck> $slides = nullptr;
size_t $current = 0;
SlidesUI(shared_ptr<SlideDeck> slides) {
dbc::check(slides != nullptr, "slides is null");
dbc::check(slides->size() > 0, "slide deck is empy");
$slides = slides;
$current = 0;
$gui.position(0, 0, WINDOW_WIDTH, WINDOW_HEIGHT);
$gui.layout(
"[t_left|t_center|t_right]"
"[m_left|*%(300,400)slide|_|_|m_right]"
"[_|_|_|_|_]"
"[_|_|_|_|_]"
"[_|_|_|_|_]"
"[_|_|_|_|_]"
"[b_left|b_center|b_right]"
);
}
void init() {
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& current() {
return $slides->at($current);
}
void next_slide() {
if($current < $slides->size() - 1) {
$current++;
show_slide();
}
}
void prev_slide() {
if($current > 0) {
$current--;
show_slide();
}
}
void show_slide() {
auto& slide = current();
auto& bg = $gui.get<guecs::Background>($gui.MAIN);
sf::Color color = guecs::THEME.FILL_COLOR;
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]};
}
bg.set_color(color);
bg.init();
auto& cell = $gui.cell_for("slide");
slide.init(cell);
}
void render(sf::RenderWindow& window) {
$gui.render(window);
auto& slide = current();
slide.render(window);
// $gui.debug_layout(window);
}
void 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();
}
}
};
shared_ptr<SlideDeck> parse_slides(const string& md_file) {
shared_ptr<SlideDeck> slides = make_shared<SlideDeck>();
dbc::check(fs::exists(md_file), "md file missing");
auto size = fs::file_size(md_file);
string line(size, '\0');
bool started = false;
json config;
if(std::ifstream in_file{md_file, std::ios::binary}) {
while(std::getline(in_file, line)) {
if(line == "{") {
string json_data;
do {
json_data += line;
} while (std::getline(in_file, line) && line != "}");
json_data += "}";
config = json::parse(json_data);
std::cout << "JSON: " << config << '\n';
} else if(line == "===") {
fmt::println("START");
started = true;
} else if(line == "---") {
fmt::println("START SLIDE");
} else {
if(started) {
string title = line;
string content;
while(std::getline(in_file, line) && line != "---") {
content += line + "\n";
}
slides->emplace_back(title, content, config);
config = json::parse("{}");
} else {
fmt::println("JUNK: {}", line);
}
}
}
}
return slides;
}
int main(int argc, char *argv[]) {
dbc::check(argc == 2, "USAGE: besos my_fucking_slides.md");
sfml::Backend backend;
guecs::init(&backend);
auto& modes = sf::VideoMode::getFullscreenModes();
WINDOW_WIDTH = modes[1].size.x;
WINDOW_HEIGHT = modes[1].size.y;
sf::RenderWindow window(modes[1], "Besos Loves Slides", sf::Style::None);
window.setFramerateLimit(FRAME_LIMIT);
window.setVerticalSyncEnabled(VSYNC);
auto data = parse_slides(argv[1]);
SlidesUI slides(data);
slides.init();
while(window.isOpen()) {
while (const auto event = window.pollEvent()) {
if(event->is<sf::Event::Closed>()) {
window.close();
}
if(const auto* mouse = event->getIf<sf::Event::MouseButtonPressed>()) {
sf::Vector2f pos = window.mapPixelToCoords(mouse->position);
if(mouse->button == sf::Mouse::Button::Left) {
slides.mouse(pos.x, pos.y, {1 << guecs::ModBit::left});
} else if(mouse->button == sf::Mouse::Button::Right) {
slides.mouse(pos.x, pos.y, {1 << guecs::ModBit::right});
}
}
}
slides.render(window);
window.display();
}
}