TGUI 1.8
Loading...
Searching...
No Matches
BackendGuiSFML.hpp
1
2//
3// TGUI - Texus' Graphical User Interface
4// Copyright (C) 2012-2025 Bruno Van de Velde (vdv_b@tgui.eu)
5//
6// This software is provided 'as-is', without any express or implied warranty.
7// In no event will the authors be held liable for any damages arising from the use of this software.
8//
9// Permission is granted to anyone to use this software for any purpose,
10// including commercial applications, and to alter it and redistribute it freely,
11// subject to the following restrictions:
12//
13// 1. The origin of this software must not be misrepresented;
14// you must not claim that you wrote the original software.
15// If you use this software in a product, an acknowledgment
16// in the product documentation would be appreciated but is not required.
17//
18// 2. Altered source versions must be plainly marked as such,
19// and must not be misrepresented as being the original software.
20//
21// 3. This notice may not be removed or altered from any source distribution.
22//
24
25#ifndef TGUI_BACKEND_GUI_SFML_HPP
26#define TGUI_BACKEND_GUI_SFML_HPP
27
28#include <TGUI/Config.hpp>
29#if !TGUI_BUILD_AS_CXX_MODULE
30 #include <TGUI/Backend/Window/BackendGui.hpp>
31#endif
32
33#include <SFML/Window.hpp>
34
36
37TGUI_MODULE_EXPORT namespace tgui
38{
40
41 class TGUI_API BackendGuiSFML : public BackendGui
42 {
43 public:
44
51
56
87 bool handleEvent(sf::Event event);
89
90#if SFML_VERSION_MAJOR >= 3
143 template <typename... Ts>
144 void handleWindowEvents(Ts&&... handlers)
145 {
146 if (!m_window)
147 return;
148
149 m_window->handleEvents([this,&handlers...](auto&& event) {
150 using EventType = decltype(event);
151
152 // We can't properly detect whether the handlers are valid (e.g. when using an invalid event type),
153 // but we can do some tests that will give errors on handlers that are wrong but almost correct.
154 static_assert(!std::disjunction_v<std::is_invocable_r<bool, Ts, EventType, bool>...>, "Handler for handleWindowEvents can't have both an extra bool argument and a bool return type");
155
156 // If any of the handlers return a bool then they will be executed before the event is passed to the gui.
157 // The returned boolean then decides whether the gui should still process the event or not.
158 // Only one of the handlers should match at most, but letting the compiler figure out which handler to
159 // call by creating an overload set is tricky because only handlers with a bool returns should be called here.
160 auto callIfMatchesAndReturnsBool = [&event](auto&& handler)
161 {
162 // std::is_invocable_r_v would still return true if the type is convertable to bool, so we use std::invoke_result_t to test the return type
163 using FuncType = decltype(handler);
164 if constexpr (std::is_invocable_v<FuncType, EventType>)
165 {
166 if constexpr (std::is_same_v<std::invoke_result_t<FuncType, EventType>, bool>)
167 return std::invoke(std::forward<FuncType>(handler), std::forward<EventType>(event));
168 else
169 {
170 static_assert(std::is_same_v<std::invoke_result_t<FuncType, EventType>, void>, "Handler for handleWindowEvents must have either 'void' or 'bool' return type");
171 return true;
172 }
173 }
174 else
175 return true;
176 };
177 const bool passEventToGui = (callIfMatchesAndReturnsBool(std::forward<Ts>(handlers)) && ...);
178
179 // Let the gui handle the event
180 bool eventHandledByGui = false;
181 if (passEventToGui)
182 eventHandledByGui = handleEvent(std::forward<EventType>(event));
183
184 // After the gui has handled the events, we call the handlers that return nothing.
185 // These handlers can have an optional bool parameter that indicates whether the event was processed by the gui.
186 auto callIfMatchesAndReturnsVoid = [&event](auto&& handler, auto&&... extraArgs)
187 {
188 using FuncType = decltype(handler);
189 if constexpr (std::is_invocable_v<FuncType, EventType, decltype(extraArgs)...>)
190 {
191 if constexpr (std::is_same_v<std::invoke_result_t<FuncType, EventType, decltype(extraArgs)...>, void>)
192 std::invoke(std::forward<FuncType>(handler), std::forward<EventType>(event), std::forward<decltype(extraArgs)>(extraArgs)...);
193 }
194 };
195 (callIfMatchesAndReturnsVoid(std::forward<Ts>(handlers)), ...);
196 (callIfMatchesAndReturnsVoid(std::forward<Ts>(handlers), eventHandledByGui), ...);
197 });
198 }
199#endif
200
246 void mainLoop(Color clearColor = {240, 240, 240}) override;
247
256 TGUI_NODISCARD bool convertEvent(const sf::Event& eventSFML, Event& eventTGUI);
257
263 sf::Window* getWindow() const;
264
271 void startTextInput(FloatRect inputRect) override;
272
277 void stopTextInput() override;
278
288 void updateTextCursorPosition(FloatRect inputRect, Vector2f caretPos) override;
289
297 TGUI_NODISCARD bool isKeyboardModifierPressed(Event::KeyModifier modifierKey) const override;
298
300 protected:
301
307 void setGuiWindow(sf::Window& window);
308
310 // Updates the view and changes the size of the root container when needed
312 void updateContainerSize() override;
313
315 protected:
316
317 sf::Window* m_window = nullptr;
318
319 bool m_modifierKeySystemPressed = false;
320 bool m_modifierKeyControlPressed = false;
321 bool m_modifierKeyShiftPressed = false;
322 bool m_modifierKeyAltPressed = false;
323
325 };
326}
327
329
330#endif // TGUI_BACKEND_GUI_SFML_HPP
~BackendGuiSFML()
Destructor.
bool convertEvent(const sf::Event &eventSFML, Event &eventTGUI)
Helper function that converts an SFML event to a TGUI event.
void setGuiWindow(sf::Window &window)
Sets the window which the gui should use.
sf::Window * getWindow() const
Returns the window that was provided to the gui.
bool handleEvent(sf::Event event)
Passes the event to the widgets.
void updateTextCursorPosition(FloatRect inputRect, Vector2f caretPos) override
This function is called by TGUI when the position of the caret changes in a text field (EditBox or Te...
void handleWindowEvents(Ts &&... handlers)
Handle all pending window events at once using callbacks, as alternative to polling events.
Definition BackendGuiSFML.hpp:144
BackendGuiSFML()
Default constructor.
void mainLoop(Color clearColor={240, 240, 240}) override
Give the gui control over the main loop.
void startTextInput(FloatRect inputRect) override
This function is called by TGUI when focusing a text field (EditBox or TextArea). It will result in t...
void stopTextInput() override
This function is called by TGUI when unfocusing a text field (EditBox or TextArea)....
bool isKeyboardModifierPressed(Event::KeyModifier modifierKey) const override
Checks the state for one of the modifier keys.
bool handleEvent(Event event)
Passes the event to the widgets.
BackendGui()
Default constructor.
Wrapper for colors.
Definition Color.hpp:73
Namespace that contains all TGUI functions and classes.
Definition AbsoluteOrRelativeValue.hpp:38
Definition Event.hpp:38
KeyModifier
Modifiers keys.
Definition Event.hpp:157