slint/api/cpp/tests/platform_eventloop.cpp
Montel Laurent 301892d008
Some checks failed
autofix.ci / format_fix (push) Has been cancelled
autofix.ci / lint_typecheck (push) Has been cancelled
CI / files-changed (push) Has been cancelled
CI / ffi_32bit_build (push) Has been cancelled
CI / build_and_test (--exclude ffmpeg --exclude gstreamer-player, macos-14, stable) (push) Has been cancelled
CI / build_and_test (--exclude ffmpeg --exclude gstreamer-player, windows-2022, 1.85) (push) Has been cancelled
CI / build_and_test (--exclude ffmpeg --exclude gstreamer-player, windows-2022, beta) (push) Has been cancelled
CI / build_and_test (--exclude ffmpeg --exclude gstreamer-player, windows-2022, stable) (push) Has been cancelled
CI / build_and_test (ubuntu-22.04, 1.85) (push) Has been cancelled
CI / build_and_test (ubuntu-22.04, nightly) (push) Has been cancelled
CI / node_test (macos-14) (push) Has been cancelled
CI / node_test (ubuntu-22.04) (push) Has been cancelled
CI / node_test (windows-2022) (push) Has been cancelled
CI / python_test (macos-14) (push) Has been cancelled
CI / python_test (ubuntu-22.04) (push) Has been cancelled
CI / python_test (windows-2022) (push) Has been cancelled
CI / cpp_test_driver (macos-13) (push) Has been cancelled
CI / cpp_test_driver (ubuntu-22.04) (push) Has been cancelled
CI / cpp_test_driver (windows-2022) (push) Has been cancelled
CI / docs (push) Has been cancelled
CI / cpp_cmake (macos-14, 1.85) (push) Has been cancelled
CI / cpp_cmake (ubuntu-22.04, stable) (push) Has been cancelled
CI / cpp_cmake (windows-2022, nightly) (push) Has been cancelled
CI / cpp_package_test (push) Has been cancelled
CI / wasm (push) Has been cancelled
CI / wasm_demo (push) Has been cancelled
CI / vsce_build_test (push) Has been cancelled
CI / mcu (pico-st7789, thumbv6m-none-eabi) (push) Has been cancelled
CI / mcu (pico2-st7789, thumbv8m.main-none-eabihf) (push) Has been cancelled
CI / mcu (stm32h735g, thumbv7em-none-eabihf) (push) Has been cancelled
CI / mcu-embassy (push) Has been cancelled
CI / tree-sitter (push) Has been cancelled
CI / updater_test (0.3.0) (push) Has been cancelled
CI / fmt_test (push) Has been cancelled
CI / esp-idf-quick (push) Has been cancelled
CI / android (push) Has been cancelled
CI / miri (push) Has been cancelled
CI / test-figma-inspector (push) Has been cancelled
Updgrade Catch2 dependency (#8846)
CMake 4.0 remove some deprecated code < 3.5 

Catch2 old version still depend against cmake 3.0 + deprecated method.
Necessary to increase version otherwise when we try to compile
example with cmake 4.0 it will failed to configure it
2025-07-05 08:54:24 +02:00

364 lines
10 KiB
C++

// Copyright © SixtyFPS GmbH <info@slint.dev>
// SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-Slint-Royalty-free-2.0 OR LicenseRef-Slint-Software-3.0
// cSpell: ignore singleshot
#define CATCH_CONFIG_MAIN
#include "catch2/catch_all.hpp"
#include <slint-platform.h>
#include <thread>
#include <deque>
#include <memory>
#include <mutex>
#include <chrono>
#include <optional>
struct TestPlatform : slint::platform::Platform
{
std::mutex the_mutex;
std::deque<slint::platform::Platform::Task> queue;
bool quit = false;
std::condition_variable cv;
std::chrono::time_point<std::chrono::steady_clock> start = std::chrono::steady_clock::now();
/// Returns a new WindowAdapter
virtual std::unique_ptr<slint::platform::WindowAdapter> create_window_adapter() override
{
#ifdef SLINT_FEATURE_RENDERER_SOFTWARE
struct TestWindowAdapter : slint::platform::WindowAdapter
{
slint::platform::SoftwareRenderer r { { } };
slint::PhysicalSize size() override { return slint::PhysicalSize({}); }
slint::platform::AbstractRenderer &renderer() override { return r; }
};
return std::make_unique<TestWindowAdapter>();
#else
assert(!"creating window in this test");
return nullptr;
#endif
};
/// Spins an event loop and renders the visible windows.
virtual void run_event_loop() override
{
quit = false;
while (true) {
slint::platform::update_timers_and_animations();
std::optional<slint::platform::Platform::Task> event;
{
std::unique_lock lock(the_mutex);
if (queue.empty()) {
if (quit) {
quit = false;
break;
}
if (auto duration = slint::platform::duration_until_next_timer_update()) {
cv.wait_for(lock, *duration);
} else {
cv.wait(lock);
}
continue;
} else {
event = std::move(queue.front());
queue.pop_front();
}
}
if (event) {
std::move(*event).run();
event.reset();
}
}
}
virtual void quit_event_loop() override
{
const std::unique_lock lock(the_mutex);
quit = true;
cv.notify_all();
}
virtual void run_in_event_loop(slint::platform::Platform::Task event) override
{
const std::unique_lock lock(the_mutex);
queue.push_back(std::move(event));
cv.notify_all();
}
#ifdef SLINT_FEATURE_FREESTANDING
virtual std::chrono::milliseconds duration_since_start() override
{
return std::chrono::duration_cast<std::chrono::milliseconds>(
std::chrono::steady_clock::now() - start);
}
#endif
};
bool init_platform = (slint::platform::set_platform(std::make_unique<TestPlatform>()), true);
TEST_CASE("C++ Singleshot Timers")
{
using namespace slint;
int called = 0;
Timer testTimer(std::chrono::milliseconds(16), [&]() {
slint::quit_event_loop();
called += 10;
});
REQUIRE(called == 0);
slint::run_event_loop();
REQUIRE(called == 10);
}
TEST_CASE("C++ Repeated Timer")
{
int timer_triggered = 0;
slint::Timer timer;
timer.start(slint::TimerMode::Repeated, std::chrono::milliseconds(3),
[&]() { timer_triggered++; });
REQUIRE(timer_triggered == 0);
bool timer_was_running = false;
slint::Timer::single_shot(std::chrono::milliseconds(100), [&]() {
timer_was_running = timer.running();
slint::quit_event_loop();
});
slint::run_event_loop();
REQUIRE(timer_triggered > 1);
REQUIRE(timer_was_running);
}
TEST_CASE("C++ Restart Singleshot Timer")
{
int timer_triggered = 0;
slint::Timer timer;
timer.start(slint::TimerMode::SingleShot, std::chrono::milliseconds(3),
[&]() { timer_triggered++; });
REQUIRE(timer.running());
REQUIRE(timer_triggered == 0);
bool timer_was_running = true;
slint::Timer::single_shot(std::chrono::milliseconds(50), [&]() {
timer_was_running = timer.running();
slint::quit_event_loop();
});
slint::run_event_loop();
REQUIRE(timer_triggered == 1);
REQUIRE(!timer_was_running); // Timer is already stopped at this point
timer_was_running = true;
timer_triggered = 0;
timer.restart();
REQUIRE(timer.running());
slint::Timer::single_shot(std::chrono::milliseconds(50), [&]() {
timer_was_running = timer.running();
slint::quit_event_loop();
});
slint::run_event_loop();
REQUIRE(timer_triggered == 1);
REQUIRE(!timer_was_running);
}
TEST_CASE("C++ Restart Repeated Timer")
{
int timer_triggered = 0;
slint::Timer timer;
timer.start(slint::TimerMode::Repeated, std::chrono::milliseconds(3),
[&]() { timer_triggered++; });
REQUIRE(timer_triggered == 0);
bool timer_was_running = false;
slint::Timer::single_shot(std::chrono::milliseconds(50), [&]() {
timer_was_running = timer.running();
slint::quit_event_loop();
});
slint::run_event_loop();
REQUIRE(timer_triggered > 1);
REQUIRE(timer_was_running);
timer_was_running = false;
timer_triggered = 0;
timer.stop();
slint::Timer::single_shot(std::chrono::milliseconds(50), [&]() {
timer_was_running = timer.running();
slint::quit_event_loop();
});
slint::run_event_loop();
REQUIRE(timer_triggered == 0);
REQUIRE(!timer_was_running);
timer_was_running = false;
timer_triggered = 0;
timer.restart();
slint::Timer::single_shot(std::chrono::milliseconds(50), [&]() {
timer_was_running = timer.running();
slint::quit_event_loop();
});
slint::run_event_loop();
REQUIRE(timer_triggered > 1);
REQUIRE(timer_was_running);
}
TEST_CASE("Quit from event")
{
int called = 0;
slint::invoke_from_event_loop([&] {
slint::quit_event_loop();
called += 10;
});
REQUIRE(called == 0);
slint::run_event_loop();
REQUIRE(called == 10);
}
TEST_CASE("Event from thread")
{
std::atomic<int> called = 0;
auto t = std::thread([&] {
called += 10;
slint::invoke_from_event_loop([&] {
called += 100;
slint::quit_event_loop();
});
});
slint::run_event_loop();
REQUIRE(called == 110);
t.join();
}
TEST_CASE("Blocking Event from thread")
{
std::atomic<int> called = 0;
auto t = std::thread([&] {
// test returning a, unique_ptr because it is movable-only
std::unique_ptr foo =
slint::blocking_invoke_from_event_loop([&] { return std::make_unique<int>(42); });
called = *foo;
int xxx = 123;
slint::blocking_invoke_from_event_loop([&] {
slint::quit_event_loop();
xxx = 888999;
});
REQUIRE(xxx == 888999);
});
slint::run_event_loop();
REQUIRE(called == 42);
t.join();
}
#if defined(SLINT_FEATURE_INTERPRETER) && defined(SLINT_FEATURE_RENDERER_SOFTWARE)
# include <slint-interpreter.h>
TEST_CASE("Quit on last window closed")
{
using namespace slint::interpreter;
using namespace slint;
int ok = 0;
ComponentCompiler compiler;
auto comp_def = compiler.build_from_source("export component App inherits Window { }", "");
REQUIRE(comp_def.has_value());
auto instance = comp_def->create();
instance->hide(); // hide before show should mess the counter
REQUIRE(instance->window().is_visible() == false);
instance->show();
REQUIRE(instance->window().is_visible() == true);
slint::Timer::single_shot(std::chrono::milliseconds(10), [&]() {
REQUIRE(instance->window().is_visible() == true);
instance->hide();
REQUIRE(instance->window().is_visible() == false);
ok = 1;
slint::Timer::single_shot(std::chrono::milliseconds(0), [&]() {
// event loop should be stopped
ok = -1;
});
});
slint::run_event_loop();
REQUIRE(ok == 1);
REQUIRE(instance->window().is_visible() == false);
ok = 0;
slint::Timer::single_shot(std::chrono::milliseconds(5), [&]() {
REQUIRE(ok == -1); // the event we started previously should have been ran first
ok = 1;
REQUIRE(instance->window().is_visible() == false);
instance->show();
instance->show(); // two show shouldn't make the loop alive
slint::Timer::single_shot(std::chrono::milliseconds(0), [&]() {
REQUIRE(instance->window().is_visible() == true);
instance->hide();
ok = 2;
slint::Timer::single_shot(std::chrono::milliseconds(0), [&]() {
// event loop should be stopped
ok = -2;
});
});
});
slint::run_event_loop();
REQUIRE(ok == 2);
ok = 0;
auto instance2 = comp_def->create();
instance2->show();
slint::Timer::single_shot(std::chrono::milliseconds(5), [&]() {
REQUIRE(ok == -2); // the event we started previously should have been ran first
instance->show();
instance2->hide();
slint::Timer::single_shot(std::chrono::milliseconds(0), [&]() {
instance2->show();
instance->hide();
slint::Timer::single_shot(std::chrono::milliseconds(0), [&]() {
instance2->hide();
ok = 3;
});
});
});
slint::run_event_loop();
REQUIRE(ok == 3);
ok = 0;
slint::Timer::single_shot(std::chrono::milliseconds(0), [&]() {
REQUIRE(ok == 0);
instance->show();
slint::Timer::single_shot(std::chrono::milliseconds(0), [&]() {
instance2->hide();
instance->hide();
slint::Timer::single_shot(std::chrono::milliseconds(0), [&]() {
instance2->show();
slint::quit_event_loop();
ok = 4;
});
});
});
slint::run_event_loop(slint::EventLoopMode::RunUntilQuit);
REQUIRE(ok == 4);
}
#endif