21#include <winpr/sysinfo.h>
22#include <winpr/assert.h>
24#include <freerdp/gdi/gdi.h>
28#include "sdl_disp.hpp"
29#include "sdl_input.hpp"
30#include "sdl_context.hpp"
32#include <freerdp/log.h>
33#define TAG CLIENT_TAG("sdl.disp")
35static constexpr UINT64 RESIZE_MIN_DELAY = 200;
36static constexpr unsigned MAX_RETRIES = 5;
41 if (a.Flags != b.Flags)
47 if (a.Width != b.Width)
49 if (a.Height != b.Height)
51 if (a.PhysicalWidth != b.PhysicalWidth)
53 if (a.PhysicalHeight != b.PhysicalHeight)
55 if (a.Orientation != b.Orientation)
57 if (a.DesktopScaleFactor != b.DesktopScaleFactor)
59 if (a.DeviceScaleFactor != b.DeviceScaleFactor)
64bool sdlDispContext::settings_changed(
const std::vector<DISPLAY_CONTROL_MONITOR_LAYOUT>& layout)
66 return (layout != _last_sent_layout);
69bool sdlDispContext::sendResize()
71 auto settings = _sdl->context()->settings;
76 if (!_activated || !_disp)
79 if (GetTickCount64() - _lastSentDate < RESIZE_MIN_DELAY)
82 _lastSentDate = GetTickCount64();
85 auto monitors =
static_cast<const rdpMonitor*
>(
87 return sendLayout(monitors, mcount);
90bool sdlDispContext::setWindowResizeable()
92 return _sdl->setResizeable(
true);
96 rdpSettings** ppSettings)
101 auto sdl = get_context(context);
105 if (!sdl->context()->settings)
109 *ppsdlDisp = &sdl->getDisplayChannelContext();
110 *ppSettings = sdl->context()->settings;
114void sdlDispContext::OnActivated(
void* context,
const ActivatedEventArgs* e)
118 rdpSettings* settings =
nullptr;
120 if (!sdl_disp_check_context(context, &sdl, &sdlDisp, &settings))
123 sdlDisp->_waitingResize =
false;
127 if (!sdlDisp->setWindowResizeable())
130 if (e->firstActivation)
133 std::ignore = sdlDisp->addTimer();
137void sdlDispContext::OnGraphicsReset(
void* context,
const GraphicsResetEventArgs* e)
141 rdpSettings* settings =
nullptr;
144 if (!sdl_disp_check_context(context, &sdl, &sdlDisp, &settings))
147 sdlDisp->_waitingResize =
false;
151 if (sdlDisp->setWindowResizeable())
152 std::ignore = sdlDisp->addTimer();
156Uint32 sdlDispContext::OnTimer(
void* param, [[maybe_unused]] SDL_TimerID timerID, Uint32 interval)
167 rdpSettings* settings =
nullptr;
169 if (!sdl_disp_check_context(sdl->context(), &sdl, &sdlDisp, &settings))
172 WLog_Print(sdl->getWLog(), WLOG_TRACE,
"checking for display changes...");
174 auto rc = sdlDisp->sendResize();
176 WLog_Print(sdl->getWLog(), WLOG_TRACE,
"sent new display layout, result %d", rc);
178 if (sdlDisp->_timer_retries++ >= MAX_RETRIES)
180 WLog_Print(sdl->getWLog(), WLOG_TRACE,
"deactivate timer, retries exceeded");
184 WLog_Print(sdl->getWLog(), WLOG_TRACE,
"fire timer one more time");
188bool sdlDispContext::sendLayout(
const rdpMonitor* monitors,
size_t nmonitors)
190 WINPR_ASSERT(monitors);
191 WINPR_ASSERT(nmonitors > 0);
193 auto settings = _sdl->context()->settings;
194 WINPR_ASSERT(settings);
196 std::vector<DISPLAY_CONTROL_MONITOR_LAYOUT> layouts;
197 layouts.reserve(nmonitors);
199 for (
size_t i = 0; i < nmonitors; i++)
201 auto monitor = &monitors[i];
204 layout.Flags = (monitor->is_primary ? DISPLAY_CONTROL_MONITOR_PRIMARY : 0);
205 layout.Left = monitor->x;
206 layout.Top = monitor->y;
207 layout.Width = WINPR_ASSERTING_INT_CAST(uint32_t, monitor->width);
208 layout.Height = WINPR_ASSERTING_INT_CAST(uint32_t, monitor->height);
209 layout.Orientation = ORIENTATION_LANDSCAPE;
210 layout.PhysicalWidth = monitor->attributes.physicalWidth;
211 layout.PhysicalHeight = monitor->attributes.physicalHeight;
213 switch (monitor->attributes.orientation)
215 case ORIENTATION_PORTRAIT:
216 layout.Orientation = ORIENTATION_PORTRAIT;
219 case ORIENTATION_LANDSCAPE_FLIPPED:
220 layout.Orientation = ORIENTATION_LANDSCAPE_FLIPPED;
223 case ORIENTATION_PORTRAIT_FLIPPED:
224 layout.Orientation = ORIENTATION_PORTRAIT_FLIPPED;
227 case ORIENTATION_LANDSCAPE:
236 layout.Orientation = ORIENTATION_LANDSCAPE;
240 layout.DesktopScaleFactor = monitor->attributes.desktopScaleFactor;
241 layout.DeviceScaleFactor = monitor->attributes.deviceScaleFactor;
244 if ((mask & FREERDP_MONITOR_OVERRIDE_ORIENTATION) != 0)
246 if ((mask & FREERDP_MONITOR_OVERRIDE_DESKTOP_SCALE) != 0)
247 layout.DesktopScaleFactor =
249 if ((mask & FREERDP_MONITOR_OVERRIDE_DEVICE_SCALE) != 0)
250 layout.DeviceScaleFactor =
252 layouts.emplace_back(layout);
255 if (!settings_changed(layouts))
259 const size_t len = layouts.size();
260 WINPR_ASSERT(len <= UINT32_MAX);
261 const auto ret = IFCALLRESULT(CHANNEL_RC_OK, _disp->SendMonitorLayout, _disp,
262 static_cast<UINT32
>(len), layouts.data());
263 if (ret != CHANNEL_RC_OK)
265 _last_sent_layout = layouts;
269bool sdlDispContext::addTimer()
271 if (SDL_WasInit(SDL_INIT_EVENTS) == 0)
274 SDL_RemoveTimer(_timer);
275 WLog_Print(_sdl->getWLog(), WLOG_TRACE,
"adding new display check timer");
280 _timer = SDL_AddTimer(1000, sdlDispContext::OnTimer,
this);
284bool sdlDispContext::updateMonitor([[maybe_unused]] SDL_WindowID
id)
289 if (!_sdl->updateWindowList())
295bool sdlDispContext::updateMonitors(SDL_EventType type, SDL_DisplayID displayID)
297 auto settings = _sdl->context()->settings;
306 case SDL_EVENT_DISPLAY_ADDED:
307 if (!_sdl->addDisplayWindow(displayID))
310 case SDL_EVENT_DISPLAY_REMOVED:
311 if (!_sdl->removeDisplay(displayID))
318 if (!_sdl->updateWindowList())
323bool sdlDispContext::handleEvent(
const SDL_DisplayEvent& ev)
327 case SDL_EVENT_DISPLAY_ADDED:
328 SDL_Log(
"A new display with id %u was connected", ev.displayID);
329 return updateMonitors(ev.type, ev.displayID);
330 case SDL_EVENT_DISPLAY_REMOVED:
331 SDL_Log(
"The display with id %u was disconnected", ev.displayID);
332 return updateMonitors(ev.type, ev.displayID);
333 case SDL_EVENT_DISPLAY_ORIENTATION:
334 SDL_Log(
"The orientation of display with id %u was changed", ev.displayID);
335 return updateMonitors(ev.type, ev.displayID);
336 case SDL_EVENT_DISPLAY_MOVED:
337 SDL_Log(
"The display with id %u was moved", ev.displayID);
338 return updateMonitors(ev.type, ev.displayID);
339 case SDL_EVENT_DISPLAY_CONTENT_SCALE_CHANGED:
340 SDL_Log(
"The display with id %u changed scale", ev.displayID);
341 return updateMonitors(ev.type, ev.displayID);
342 case SDL_EVENT_DISPLAY_CURRENT_MODE_CHANGED:
343 SDL_Log(
"The display with id %u changed mode", ev.displayID);
344 return updateMonitors(ev.type, ev.displayID);
345 case SDL_EVENT_DISPLAY_DESKTOP_MODE_CHANGED:
346 SDL_Log(
"The display with id %u changed desktop mode", ev.displayID);
347 return updateMonitors(ev.type, ev.displayID);
353bool sdlDispContext::handleEvent(
const SDL_WindowEvent& ev)
355 auto window = _sdl->getWindowForId(ev.windowID);
360 window->setBordered(bordered);
364 case SDL_EVENT_WINDOW_HIDDEN:
365 case SDL_EVENT_WINDOW_MINIMIZED:
366 return _sdl->redraw(
true);
367 case SDL_EVENT_WINDOW_ENTER_FULLSCREEN:
368 return updateMonitor(ev.windowID);
369 case SDL_EVENT_WINDOW_LEAVE_FULLSCREEN:
370 return updateMonitor(ev.windowID);
372 case SDL_EVENT_WINDOW_EXPOSED:
373 case SDL_EVENT_WINDOW_SHOWN:
374 case SDL_EVENT_WINDOW_MAXIMIZED:
375 case SDL_EVENT_WINDOW_RESTORED:
381 case SDL_EVENT_WINDOW_DISPLAY_SCALE_CHANGED:
382 case SDL_EVENT_WINDOW_PIXEL_SIZE_CHANGED:
383 case SDL_EVENT_WINDOW_RESIZED:
384 return updateMonitor(ev.windowID);
385 case SDL_EVENT_WINDOW_MOUSE_LEAVE:
387 return _sdl->getInputChannelContext().keyboard_grab(ev.windowID,
false);
388 case SDL_EVENT_WINDOW_MOUSE_ENTER:
390 if (!_sdl->getInputChannelContext().keyboard_grab(ev.windowID,
true))
392 return _sdl->getInputChannelContext().keyboard_focus_in();
393 case SDL_EVENT_WINDOW_FOCUS_GAINED:
394 return _sdl->getInputChannelContext().keyboard_focus_in();
401UINT sdlDispContext::DisplayControlCaps(DispClientContext* disp, UINT32 maxNumMonitors,
402 UINT32 maxMonitorAreaFactorA, UINT32 maxMonitorAreaFactorB)
408 return sdlDisp->DisplayControlCaps(maxNumMonitors, maxMonitorAreaFactorA,
409 maxMonitorAreaFactorB);
412UINT sdlDispContext::DisplayControlCaps(UINT32 maxNumMonitors, UINT32 maxMonitorAreaFactorA,
413 UINT32 maxMonitorAreaFactorB)
415 auto settings = _sdl->context()->settings;
416 WINPR_ASSERT(settings);
419 "DisplayControlCapsPdu: MaxNumMonitors: %" PRIu32
" MaxMonitorAreaFactorA: %" PRIu32
420 " MaxMonitorAreaFactorB: %" PRIu32
"",
421 maxNumMonitors, maxMonitorAreaFactorA, maxMonitorAreaFactorB);
425 return CHANNEL_RC_OK;
427 WLog_DBG(TAG,
"DisplayControlCapsPdu: setting the window as resizable");
428 return setWindowResizeable() ? CHANNEL_RC_OK : CHANNEL_RC_NO_MEMORY;
431bool sdlDispContext::init(DispClientContext* disp)
436 auto settings = _sdl->context()->settings;
446 disp->DisplayControlCaps = sdlDispContext::DisplayControlCaps;
449 return _sdl->setResizeable(
true);
452bool sdlDispContext::uninit(DispClientContext* disp)
458 return _sdl->setResizeable(
false);
461sdlDispContext::sdlDispContext(
SdlContext* sdl) : _sdl(sdl)
464 WINPR_ASSERT(_sdl->context()->settings);
465 WINPR_ASSERT(_sdl->context()->pubSub);
467 auto pubSub = _sdl->context()->pubSub;
469 PubSub_SubscribeActivated(pubSub, sdlDispContext::OnActivated);
470 PubSub_SubscribeGraphicsReset(pubSub, sdlDispContext::OnGraphicsReset);
471 std::ignore = addTimer();
474sdlDispContext::~sdlDispContext()
476 wPubSub* pubSub = _sdl->context()->pubSub;
477 WINPR_ASSERT(pubSub);
479 PubSub_UnsubscribeActivated(pubSub, sdlDispContext::OnActivated);
480 PubSub_UnsubscribeGraphicsReset(pubSub, sdlDispContext::OnGraphicsReset);
481 SDL_RemoveTimer(_timer);
WINPR_ATTR_NODISCARD FREERDP_API const void * freerdp_settings_get_pointer(const rdpSettings *settings, FreeRDP_Settings_Keys_Pointer id)
Returns a immutable pointer settings value.
WINPR_ATTR_NODISCARD FREERDP_API UINT16 freerdp_settings_get_uint16(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt16 id)
Returns a UINT16 settings value.
WINPR_ATTR_NODISCARD FREERDP_API UINT64 freerdp_settings_get_uint64(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt64 id)
Returns a UINT64 settings value.
WINPR_ATTR_NODISCARD FREERDP_API UINT32 freerdp_settings_get_uint32(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id)
Returns a UINT32 settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.