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(SDL_WindowID
id)
289 if (!_sdl->updateWindow(
id))
292 if (!_sdl->updateWindowList())
298bool sdlDispContext::updateMonitors(SDL_EventType type, SDL_DisplayID displayID)
300 auto settings = _sdl->context()->settings;
309 case SDL_EVENT_DISPLAY_ADDED:
310 if (!_sdl->addDisplayWindow(displayID))
313 case SDL_EVENT_DISPLAY_REMOVED:
314 if (!_sdl->removeDisplayWindow(displayID))
321 if (!_sdl->updateWindowList())
326bool sdlDispContext::handleEvent(
const SDL_DisplayEvent& ev)
328 const auto cat = SDL_LOG_CATEGORY_APPLICATION;
331 case SDL_EVENT_DISPLAY_ADDED:
332 SDL_LogDebug(cat,
"A new display with id %u was connected", ev.displayID);
333 return updateMonitors(ev.type, ev.displayID);
334 case SDL_EVENT_DISPLAY_REMOVED:
335 SDL_LogDebug(cat,
"The display with id %u was disconnected", ev.displayID);
336 return updateMonitors(ev.type, ev.displayID);
337 case SDL_EVENT_DISPLAY_ORIENTATION:
338 SDL_LogDebug(cat,
"The orientation of display with id %u was changed", ev.displayID);
339 return updateMonitors(ev.type, ev.displayID);
340 case SDL_EVENT_DISPLAY_MOVED:
341 SDL_LogDebug(cat,
"The display with id %u was moved", ev.displayID);
342 return updateMonitors(ev.type, ev.displayID);
343 case SDL_EVENT_DISPLAY_CONTENT_SCALE_CHANGED:
344 SDL_LogDebug(cat,
"The display with id %u changed scale", ev.displayID);
345 return updateMonitors(ev.type, ev.displayID);
346 case SDL_EVENT_DISPLAY_CURRENT_MODE_CHANGED:
347 SDL_LogDebug(cat,
"The display with id %u changed mode", ev.displayID);
348 return updateMonitors(ev.type, ev.displayID);
349 case SDL_EVENT_DISPLAY_DESKTOP_MODE_CHANGED:
350 SDL_LogDebug(cat,
"The display with id %u changed desktop mode", ev.displayID);
351 return updateMonitors(ev.type, ev.displayID);
357bool sdlDispContext::handleEvent(
const SDL_WindowEvent& ev)
359 auto window = _sdl->getWindowForId(ev.windowID);
364 window->setBordered(bordered);
368 case SDL_EVENT_WINDOW_HIDDEN:
369 case SDL_EVENT_WINDOW_MINIMIZED:
370 return _sdl->redraw(
true);
371 case SDL_EVENT_WINDOW_ENTER_FULLSCREEN:
372 return updateMonitor(ev.windowID);
373 case SDL_EVENT_WINDOW_LEAVE_FULLSCREEN:
374 return updateMonitor(ev.windowID);
376 case SDL_EVENT_WINDOW_EXPOSED:
377 case SDL_EVENT_WINDOW_SHOWN:
378 case SDL_EVENT_WINDOW_MAXIMIZED:
379 case SDL_EVENT_WINDOW_RESTORED:
385 case SDL_EVENT_WINDOW_DISPLAY_SCALE_CHANGED:
386 case SDL_EVENT_WINDOW_PIXEL_SIZE_CHANGED:
387 case SDL_EVENT_WINDOW_RESIZED:
388 return updateMonitor(ev.windowID);
389 case SDL_EVENT_WINDOW_MOUSE_LEAVE:
391 return _sdl->getInputChannelContext().keyboard_grab(ev.windowID,
false);
392 case SDL_EVENT_WINDOW_MOUSE_ENTER:
394 if (!_sdl->getInputChannelContext().keyboard_grab(ev.windowID,
true))
396 return _sdl->getInputChannelContext().keyboard_focus_in();
397 case SDL_EVENT_WINDOW_FOCUS_GAINED:
398 return _sdl->getInputChannelContext().keyboard_focus_in();
405UINT sdlDispContext::DisplayControlCaps(DispClientContext* disp, UINT32 maxNumMonitors,
406 UINT32 maxMonitorAreaFactorA, UINT32 maxMonitorAreaFactorB)
412 return sdlDisp->DisplayControlCaps(maxNumMonitors, maxMonitorAreaFactorA,
413 maxMonitorAreaFactorB);
416UINT sdlDispContext::DisplayControlCaps(UINT32 maxNumMonitors, UINT32 maxMonitorAreaFactorA,
417 UINT32 maxMonitorAreaFactorB)
419 auto settings = _sdl->context()->settings;
420 WINPR_ASSERT(settings);
423 "DisplayControlCapsPdu: MaxNumMonitors: %" PRIu32
" MaxMonitorAreaFactorA: %" PRIu32
424 " MaxMonitorAreaFactorB: %" PRIu32
"",
425 maxNumMonitors, maxMonitorAreaFactorA, maxMonitorAreaFactorB);
429 return CHANNEL_RC_OK;
431 WLog_DBG(TAG,
"DisplayControlCapsPdu: setting the window as resizable");
432 return setWindowResizeable() ? CHANNEL_RC_OK : CHANNEL_RC_NO_MEMORY;
435bool sdlDispContext::init(DispClientContext* disp)
440 auto settings = _sdl->context()->settings;
450 disp->DisplayControlCaps = sdlDispContext::DisplayControlCaps;
453 return _sdl->setResizeable(
true);
456bool sdlDispContext::uninit(DispClientContext* disp)
462 return _sdl->setResizeable(
false);
465sdlDispContext::sdlDispContext(
SdlContext* sdl) : _sdl(sdl)
468 WINPR_ASSERT(_sdl->context()->settings);
469 WINPR_ASSERT(_sdl->context()->pubSub);
471 auto pubSub = _sdl->context()->pubSub;
473 if (PubSub_SubscribeActivated(pubSub, sdlDispContext::OnActivated) < 0)
474 throw std::exception();
475 if (PubSub_SubscribeGraphicsReset(pubSub, sdlDispContext::OnGraphicsReset) < 0)
476 throw std::exception();
477 std::ignore = addTimer();
480sdlDispContext::~sdlDispContext()
482 wPubSub* pubSub = _sdl->context()->pubSub;
483 WINPR_ASSERT(pubSub);
485 PubSub_UnsubscribeActivated(pubSub, sdlDispContext::OnActivated);
486 PubSub_UnsubscribeGraphicsReset(pubSub, sdlDispContext::OnGraphicsReset);
487 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.