20#include <freerdp/config.h>
26#include <winpr/cast.h>
27#include <winpr/assert.h>
28#include <winpr/wlog.h>
29#include <winpr/print.h>
31#include <freerdp/client/rail.h>
37#include <freerdp/log.h>
38#define TAG CLIENT_TAG("x11")
40static const char* error_code2str(UINT32 code)
47 EVCASE(RAIL_EXEC_S_OK);
48 EVCASE(RAIL_EXEC_E_HOOK_NOT_LOADED);
49 EVCASE(RAIL_EXEC_E_DECODE_FAILED);
50 EVCASE(RAIL_EXEC_E_NOT_IN_ALLOWLIST);
51 EVCASE(RAIL_EXEC_E_FILE_NOT_FOUND);
52 EVCASE(RAIL_EXEC_E_FAIL);
53 EVCASE(RAIL_EXEC_E_SESSION_LOCKED);
55 return "RAIL_EXEC_E_UNKNOWN";
60static const char* movetype2str(UINT32 code)
69 EVCASE(RAIL_WMSZ_LEFT);
70 EVCASE(RAIL_WMSZ_RIGHT);
71 EVCASE(RAIL_WMSZ_TOP);
72 EVCASE(RAIL_WMSZ_TOPLEFT);
73 EVCASE(RAIL_WMSZ_TOPRIGHT);
74 EVCASE(RAIL_WMSZ_BOTTOM);
75 EVCASE(RAIL_WMSZ_BOTTOMLEFT);
76 EVCASE(RAIL_WMSZ_BOTTOMRIGHT);
77 EVCASE(RAIL_WMSZ_MOVE);
78 EVCASE(RAIL_WMSZ_KEYMOVE);
79 EVCASE(RAIL_WMSZ_KEYSIZE);
81 return "RAIL_WMSZ_INVALID";
91typedef struct xf_rail_icon xfRailIcon;
93struct xf_rail_icon_cache
97 UINT32 numCacheEntries;
105} rail_paint_fn_arg_t;
107BOOL xf_rail_enable_remoteapp_mode(xfContext* xfc)
110 if (!xfc->remote_app)
112 rdpGdi* gdi = xfc->common.context.gdi;
115 const BOOL old = gdi->suppressOutput;
116 gdi->suppressOutput = TRUE;
117 xfc->remote_app = TRUE;
118 xfc->drawable = xf_CreateDummyWindow(xfc);
119 xf_DestroyDesktopWindow(xfc, xfc->window);
120 xfc->window =
nullptr;
122 gdi->suppressOutput = old;
127BOOL xf_rail_disable_remoteapp_mode(xfContext* xfc)
132 rdpGdi* gdi = xfc->common.context.gdi;
135 const BOOL old = gdi->suppressOutput;
136 gdi->suppressOutput = TRUE;
138 xfc->remote_app = FALSE;
139 xf_DestroyDummyWindow(xfc, xfc->drawable);
140 xf_destroy_window(xfc);
141 xf_create_window(xfc);
142 xf_create_image(xfc);
144 gdi->suppressOutput = old;
149BOOL xf_rail_send_activate(xfContext* xfc, Window xwindow, BOOL enabled)
152 xfAppWindow* appWindow = xf_AppWindowFromX11Window(xfc, xwindow);
156 WLog_Print(xfc->log, WLOG_DEBUG,
"xf_rail_send_activate: ignoring unknown window 0x%08lx",
162 xf_SetWindowStyle(xfc, appWindow, appWindow->dwStyle, appWindow->dwExStyle);
164 WINPR_ASSERT(appWindow->windowId <= UINT32_MAX);
165 activate.windowId = (UINT32)appWindow->windowId;
166 xf_rail_return_window(appWindow, FALSE);
168 activate.enabled = enabled;
169 const UINT rc = xfc->rail->ClientActivate(xfc->rail, &activate);
170 return rc == CHANNEL_RC_OK;
173BOOL xf_rail_send_client_system_command(xfContext* xfc, UINT64 windowId, UINT16 command)
176 WINPR_ASSERT(xfc->rail);
177 WINPR_ASSERT(xfc->rail->ClientSystemCommand);
178 if (windowId > UINT32_MAX)
182 const UINT rc = xfc->rail->ClientSystemCommand(xfc->rail, &syscommand);
183 return rc == CHANNEL_RC_OK;
192BOOL xf_rail_adjust_position(xfContext* xfc, xfAppWindow* appWindow)
197 WINPR_ASSERT(appWindow);
198 if (!appWindow->is_mapped || appWindow->local_move.state != LMS_NOT_ACTIVE)
202 if (appWindow->x != appWindow->windowOffsetX || appWindow->y != appWindow->windowOffsetY ||
203 appWindow->width != (INT64)appWindow->windowWidth ||
204 appWindow->height != (INT64)appWindow->windowHeight)
206 WINPR_ASSERT(appWindow->windowId <= UINT32_MAX);
207 windowMove.windowId = (UINT32)appWindow->windowId;
212 const INT16 left = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginLeft);
213 const INT16 right = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginRight);
214 const INT16 top = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginTop);
215 const INT16 bottom = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginBottom);
216 windowMove.left = WINPR_ASSERTING_INT_CAST(INT16, appWindow->x - left);
217 windowMove.top = WINPR_ASSERTING_INT_CAST(INT16, appWindow->y - top);
218 windowMove.right = WINPR_ASSERTING_INT_CAST(INT16, appWindow->x + appWindow->width + right);
220 WINPR_ASSERTING_INT_CAST(INT16, appWindow->y + appWindow->height + bottom);
221 const UINT rc = xfc->rail->ClientWindowMove(xfc->rail, &windowMove);
222 return rc == CHANNEL_RC_OK;
227BOOL xf_rail_end_local_move(xfContext* xfc, xfAppWindow* appWindow)
233 unsigned int mask = 0;
234 Window root_window = 0;
235 Window child_window = 0;
238 WINPR_ASSERT(appWindow);
240 if ((appWindow->local_move.direction == NET_WM_MOVERESIZE_MOVE_KEYBOARD) ||
241 (appWindow->local_move.direction == NET_WM_MOVERESIZE_SIZE_KEYBOARD))
248 WINPR_ASSERT(appWindow->windowId <= UINT32_MAX);
249 windowMove.windowId = (UINT32)appWindow->windowId;
255 const INT16 left = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginLeft);
256 const INT16 right = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginRight);
257 const INT16 top = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginTop);
258 const INT16 bottom = WINPR_ASSERTING_INT_CAST(INT16, appWindow->resizeMarginBottom);
259 const INT16 w = WINPR_ASSERTING_INT_CAST(INT16, appWindow->width + right);
260 const INT16 h = WINPR_ASSERTING_INT_CAST(INT16, appWindow->height + bottom);
261 windowMove.left = WINPR_ASSERTING_INT_CAST(INT16, appWindow->x - left);
262 windowMove.top = WINPR_ASSERTING_INT_CAST(INT16, appWindow->y - top);
263 windowMove.right = WINPR_ASSERTING_INT_CAST(INT16, appWindow->x + w);
265 windowMove.bottom = WINPR_ASSERTING_INT_CAST(INT16, appWindow->y + h);
266 const UINT rc = xfc->rail->ClientWindowMove(xfc->rail, &windowMove);
267 if (rc != CHANNEL_RC_OK)
274 XQueryPointer(xfc->display, appWindow->handle, &root_window, &child_window, &x, &y, &child_x,
278 if ((appWindow->local_move.direction != NET_WM_MOVERESIZE_MOVE_KEYBOARD) &&
279 (appWindow->local_move.direction != NET_WM_MOVERESIZE_SIZE_KEYBOARD))
281 if (!freerdp_client_send_button_event(&xfc->common, FALSE, PTR_FLAGS_BUTTON1, x, y))
290 appWindow->windowOffsetX = appWindow->x;
291 appWindow->windowOffsetY = appWindow->y;
292 appWindow->windowWidth = WINPR_ASSERTING_INT_CAST(uint32_t, appWindow->width);
293 appWindow->windowHeight = WINPR_ASSERTING_INT_CAST(uint32_t, appWindow->height);
294 appWindow->local_move.state = LMS_TERMINATING;
298BOOL xf_rail_paint_surface(xfContext* xfc, UINT64 windowId,
const RECTANGLE_16* rect)
300 xfAppWindow* appWindow = xf_rail_get_window(xfc, windowId, FALSE);
308 .left = WINPR_ASSERTING_INT_CAST(UINT16, MAX(appWindow->x, 0)),
309 .top = WINPR_ASSERTING_INT_CAST(UINT16, MAX(appWindow->y, 0)),
310 .right = WINPR_ASSERTING_INT_CAST(UINT16, MAX(appWindow->x + appWindow->width, 0)),
311 .bottom = WINPR_ASSERTING_INT_CAST(UINT16, MAX(appWindow->y + appWindow->height, 0))
314 REGION16 windowInvalidRegion = WINPR_C_ARRAY_INIT;
315 region16_init(&windowInvalidRegion);
316 if (!region16_union_rect(&windowInvalidRegion, &windowInvalidRegion, &windowRect))
318 if (!region16_intersect_rect(&windowInvalidRegion, &windowInvalidRegion, rect))
321 if (!region16_is_empty(&windowInvalidRegion))
323 const RECTANGLE_16* extents = region16_extents(&windowInvalidRegion);
326 .left = WINPR_ASSERTING_INT_CAST(UINT16, extents->left - appWindow->x),
327 .top = WINPR_ASSERTING_INT_CAST(UINT16, extents->top - appWindow->y),
328 .right = WINPR_ASSERTING_INT_CAST(UINT16, extents->right - appWindow->x),
329 .bottom = WINPR_ASSERTING_INT_CAST(UINT16, extents->bottom - appWindow->y)
332 xf_UpdateWindowArea(xfc, appWindow, updateRect.left, updateRect.top,
333 updateRect.right - updateRect.left, updateRect.bottom - updateRect.top);
335 region16_uninit(&windowInvalidRegion);
336 xf_rail_return_window(appWindow, FALSE);
340static BOOL rail_paint_fn(
const void* pvkey, WINPR_ATTR_UNUSED
void* value,
void* pvarg)
342 rail_paint_fn_arg_t* arg = pvarg;
346 const UINT64 key = *(
const UINT64*)pvkey;
347 return xf_rail_paint_surface(arg->xfc, key, arg->rect);
350BOOL xf_rail_paint(xfContext* xfc,
const RECTANGLE_16* rect)
352 rail_paint_fn_arg_t arg = { .xfc = xfc, .rect = rect };
357 if (!xfc->railWindows)
360 return HashTable_Foreach(xfc->railWindows, rail_paint_fn, &arg);
363#define window_state_log_style(log, windowState) \
364 window_state_log_style_int((log), (windowState), __FILE__, __func__, __LINE__)
365static void window_state_log_style_int(wLog* log,
const WINDOW_STATE_ORDER* windowState,
366 const char* file,
const char* fkt,
size_t line)
368 const DWORD log_level = WLOG_DEBUG;
371 WINPR_ASSERT(windowState);
372 if (WLog_IsLevelActive(log, log_level))
374 char buffer1[128] = WINPR_C_ARRAY_INIT;
375 char buffer2[128] = WINPR_C_ARRAY_INIT;
377 window_styles_to_string(windowState->style, buffer1,
sizeof(buffer1));
378 window_styles_ex_to_string(windowState->extendedStyle, buffer2,
sizeof(buffer2));
379 WLog_PrintTextMessage(log, log_level, line, file, fkt,
"windowStyle={%s, %s}", buffer1,
386static BOOL xf_rail_window_common(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo,
390 xfContext* xfc = (xfContext*)context;
393 WINPR_ASSERT(orderInfo);
394 WINPR_ASSERT(windowState);
396 UINT32 fieldFlags = orderInfo->fieldFlags;
397 BOOL position_or_size_updated = FALSE;
398 xfAppWindow* appWindow = xf_rail_get_window(xfc, orderInfo->windowId, FALSE);
400 if (fieldFlags & WINDOW_ORDER_STATE_NEW)
403 appWindow = xf_rail_add_window(xfc, orderInfo->windowId, windowState->windowOffsetX,
404 windowState->windowOffsetY, windowState->windowWidth,
405 windowState->windowHeight, 0xFFFFFFFF);
410 appWindow->dwStyle = windowState->style;
411 appWindow->dwExStyle = windowState->extendedStyle;
412 window_state_log_style(xfc->log, windowState);
415 if (fieldFlags & WINDOW_ORDER_FIELD_TITLE)
422 char* title =
nullptr;
424 cnv.b = windowState->titleInfo.string;
425 if (windowState->titleInfo.length == 0)
427 if (!(title = _strdup(
"")))
429 WLog_ERR(TAG,
"failed to duplicate empty window title string");
433 else if (!(title = ConvertWCharNToUtf8Alloc(
434 cnv.wc, windowState->titleInfo.length /
sizeof(WCHAR),
nullptr)))
436 WLog_ERR(TAG,
"failed to convert window title");
440 appWindow->title = title;
444 if (!(appWindow->title = _strdup(
"RdpRailWindow")))
445 WLog_ERR(TAG,
"failed to duplicate default window title string");
448 if (!appWindow->title)
451 xf_AppWindowInit(xfc, appWindow);
458 if ((fieldFlags & WINDOW_ORDER_FIELD_WND_OFFSET) ||
459 (fieldFlags & WINDOW_ORDER_FIELD_WND_SIZE) ||
460 (fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_OFFSET) ||
461 (fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_SIZE) ||
462 (fieldFlags & WINDOW_ORDER_FIELD_WND_CLIENT_DELTA) ||
463 (fieldFlags & WINDOW_ORDER_FIELD_VIS_OFFSET) ||
464 (fieldFlags & WINDOW_ORDER_FIELD_VISIBILITY))
466 position_or_size_updated = TRUE;
471 if (fieldFlags & WINDOW_ORDER_FIELD_WND_OFFSET)
473 appWindow->windowOffsetX = windowState->windowOffsetX;
474 appWindow->windowOffsetY = windowState->windowOffsetY;
477 if (fieldFlags & WINDOW_ORDER_FIELD_WND_SIZE)
479 appWindow->windowWidth = windowState->windowWidth;
480 appWindow->windowHeight = windowState->windowHeight;
483 if (fieldFlags & WINDOW_ORDER_FIELD_RESIZE_MARGIN_X)
485 appWindow->resizeMarginLeft = windowState->resizeMarginLeft;
486 appWindow->resizeMarginRight = windowState->resizeMarginRight;
489 if (fieldFlags & WINDOW_ORDER_FIELD_RESIZE_MARGIN_Y)
491 appWindow->resizeMarginTop = windowState->resizeMarginTop;
492 appWindow->resizeMarginBottom = windowState->resizeMarginBottom;
495 if (fieldFlags & WINDOW_ORDER_FIELD_OWNER)
497 appWindow->ownerWindowId = windowState->ownerWindowId;
500 if (fieldFlags & WINDOW_ORDER_FIELD_STYLE)
502 appWindow->dwStyle = windowState->style;
503 appWindow->dwExStyle = windowState->extendedStyle;
504 window_state_log_style(xfc->log, windowState);
507 if (fieldFlags & WINDOW_ORDER_FIELD_SHOW)
509 appWindow->showState = windowState->showState;
512 if (fieldFlags & WINDOW_ORDER_FIELD_TITLE)
514 char* title =
nullptr;
521 cnv.b = windowState->titleInfo.string;
522 if (windowState->titleInfo.length == 0)
524 if (!(title = _strdup(
"")))
526 WLog_ERR(TAG,
"failed to duplicate empty window title string");
530 else if (!(title = ConvertWCharNToUtf8Alloc(
531 cnv.wc, windowState->titleInfo.length /
sizeof(WCHAR),
nullptr)))
533 WLog_ERR(TAG,
"failed to convert window title");
537 free(appWindow->title);
538 appWindow->title = title;
541 if (fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_OFFSET)
543 appWindow->clientOffsetX = windowState->clientOffsetX;
544 appWindow->clientOffsetY = windowState->clientOffsetY;
547 if (fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_SIZE)
549 appWindow->clientAreaWidth = windowState->clientAreaWidth;
550 appWindow->clientAreaHeight = windowState->clientAreaHeight;
553 if (fieldFlags & WINDOW_ORDER_FIELD_WND_CLIENT_DELTA)
555 appWindow->windowClientDeltaX = windowState->windowClientDeltaX;
556 appWindow->windowClientDeltaY = windowState->windowClientDeltaY;
559 if (fieldFlags & WINDOW_ORDER_FIELD_WND_RECTS)
561 if (appWindow->windowRects)
563 free(appWindow->windowRects);
564 appWindow->windowRects =
nullptr;
567 appWindow->numWindowRects = windowState->numWindowRects;
569 if (appWindow->numWindowRects)
571 appWindow->windowRects =
574 if (!appWindow->windowRects)
577 CopyMemory(appWindow->windowRects, windowState->windowRects,
582 if (fieldFlags & WINDOW_ORDER_FIELD_VIS_OFFSET)
584 appWindow->visibleOffsetX = windowState->visibleOffsetX;
585 appWindow->visibleOffsetY = windowState->visibleOffsetY;
588 if (fieldFlags & WINDOW_ORDER_FIELD_VISIBILITY)
590 if (appWindow->visibilityRects)
592 free(appWindow->visibilityRects);
593 appWindow->visibilityRects =
nullptr;
596 appWindow->numVisibilityRects = windowState->numVisibilityRects;
598 if (appWindow->numVisibilityRects)
600 appWindow->visibilityRects =
603 if (!appWindow->visibilityRects)
606 CopyMemory(appWindow->visibilityRects, windowState->visibilityRects,
613 if (fieldFlags & WINDOW_ORDER_FIELD_STYLE)
617 if (fieldFlags & WINDOW_ORDER_FIELD_SHOW)
619 xf_ShowWindow(xfc, appWindow, WINPR_ASSERTING_INT_CAST(UINT8, appWindow->showState));
622 if (fieldFlags & WINDOW_ORDER_FIELD_TITLE)
624 if (appWindow->title)
625 xf_SetWindowText(xfc, appWindow, appWindow->title);
628 if (position_or_size_updated)
630 const INT32 visibilityRectsOffsetX =
631 (appWindow->visibleOffsetX -
632 (appWindow->clientOffsetX - appWindow->windowClientDeltaX));
633 const INT32 visibilityRectsOffsetY =
634 (appWindow->visibleOffsetY -
635 (appWindow->clientOffsetY - appWindow->windowClientDeltaY));
642 if (appWindow->rail_state != WINDOW_SHOW_MINIMIZED)
645 if (appWindow->x == (INT64)appWindow->windowOffsetX &&
646 appWindow->y == (INT64)appWindow->windowOffsetY &&
647 appWindow->width == (INT64)appWindow->windowWidth &&
648 appWindow->height == (INT64)appWindow->windowHeight)
650 xf_UpdateWindowArea(xfc, appWindow, 0, 0,
651 WINPR_ASSERTING_INT_CAST(
int, appWindow->windowWidth),
652 WINPR_ASSERTING_INT_CAST(
int, appWindow->windowHeight));
656 xf_MoveWindow(xfc, appWindow, appWindow->windowOffsetX, appWindow->windowOffsetY,
657 WINPR_ASSERTING_INT_CAST(
int, appWindow->windowWidth),
658 WINPR_ASSERTING_INT_CAST(
int, appWindow->windowHeight));
661 xf_SetWindowVisibilityRects(
662 xfc, appWindow, WINPR_ASSERTING_INT_CAST(uint32_t, visibilityRectsOffsetX),
663 WINPR_ASSERTING_INT_CAST(uint32_t, visibilityRectsOffsetY),
664 appWindow->visibilityRects,
665 WINPR_ASSERTING_INT_CAST(
int, appWindow->numVisibilityRects));
668 if (appWindow->rail_state == WINDOW_SHOW_MAXIMIZED)
670 xf_SendClientEvent(xfc, appWindow->handle, xfc->NET_WM_STATE, 4, NET_WM_STATE_ADD,
671 xfc->NET_WM_STATE_MAXIMIZED_VERT, xfc->NET_WM_STATE_MAXIMIZED_HORZ,
676 if (fieldFlags & (WINDOW_ORDER_STATE_NEW | WINDOW_ORDER_FIELD_STYLE))
677 xf_SetWindowStyle(xfc, appWindow, appWindow->dwStyle, appWindow->dwExStyle);
686 xf_rail_return_window(appWindow, FALSE);
690static BOOL xf_rail_window_delete(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo)
692 xfContext* xfc = (xfContext*)context;
694 return xf_rail_del_window(xfc, orderInfo->windowId);
697static xfRailIconCache* RailIconCache_New(rdpSettings* settings)
699 xfRailIconCache* cache = calloc(1,
sizeof(xfRailIconCache));
705 cache->numCacheEntries =
707 cache->entries = calloc(1ull * cache->numCaches * cache->numCacheEntries,
sizeof(xfRailIcon));
711 WLog_ERR(TAG,
"failed to allocate icon cache %" PRIu32
" x %" PRIu32
" entries",
712 cache->numCaches, cache->numCacheEntries);
720static void RailIconCache_Free(xfRailIconCache* cache)
725 for (UINT32 i = 0; i < cache->numCaches * cache->numCacheEntries; i++)
727 xfRailIcon* cur = &cache->entries[i];
731 free(cache->scratch.data);
732 free(cache->entries);
736static xfRailIcon* RailIconCache_Lookup(xfRailIconCache* cache, UINT8 cacheId, UINT16 cacheEntry)
749 return &cache->scratch;
751 if (cacheId >= cache->numCaches)
754 if (cacheEntry >= cache->numCacheEntries)
757 return &cache->entries[cache->numCacheEntries * cacheId + cacheEntry];
770static BOOL convert_rail_icon(
const ICON_INFO* iconInfo, xfRailIcon* railIcon)
772 WINPR_ASSERT(iconInfo);
773 WINPR_ASSERT(railIcon);
775 BYTE* nextPixel =
nullptr;
776 long* pixels =
nullptr;
777 BYTE* argbPixels = calloc(1ull * iconInfo->width * iconInfo->height, 4);
782 if (!freerdp_image_copy_from_icon_data(
783 argbPixels, PIXEL_FORMAT_ARGB32, 0, 0, 0,
784 WINPR_ASSERTING_INT_CAST(UINT16, iconInfo->width),
785 WINPR_ASSERTING_INT_CAST(UINT16, iconInfo->height), iconInfo->bitsColor,
786 WINPR_ASSERTING_INT_CAST(UINT16, iconInfo->cbBitsColor), iconInfo->bitsMask,
787 WINPR_ASSERTING_INT_CAST(UINT16, iconInfo->cbBitsMask), iconInfo->colorTable,
788 WINPR_ASSERTING_INT_CAST(UINT16, iconInfo->cbColorTable), iconInfo->bpp))
792 const UINT32 nelements = 2 + iconInfo->width * iconInfo->height;
793 pixels = realloc(railIcon->data, nelements *
sizeof(
long));
798 railIcon->data = pixels;
800 railIcon->length = WINPR_ASSERTING_INT_CAST(
int, nelements);
801 pixels[0] = iconInfo->width;
802 pixels[1] = iconInfo->height;
803 nextPixel = argbPixels;
805 for (UINT32 i = 2; i < nelements; i++)
807 pixels[i] = FreeRDPReadColor(nextPixel, PIXEL_FORMAT_BGRA32);
819static void xf_rail_set_window_icon(xfContext* xfc, xfAppWindow* railWindow, xfRailIcon* icon,
824 LogDynAndXChangeProperty(xfc->log, xfc->display, railWindow->handle, xfc->NET_WM_ICON,
825 XA_CARDINAL, 32, replace ? PropModeReplace : PropModeAppend,
826 (unsigned char*)icon->data, icon->length);
827 LogDynAndXFlush(xfc->log, xfc->display);
830static BOOL xf_rail_window_icon(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo,
834 xfContext* xfc = (xfContext*)context;
835 BOOL replaceIcon = 0;
836 xfAppWindow* railWindow = xf_rail_get_window(xfc, orderInfo->windowId, FALSE);
841 WINPR_ASSERT(windowIcon);
842 WINPR_ASSERT(windowIcon->iconInfo);
843 xfRailIcon* icon = RailIconCache_Lookup(
844 xfc->railIconCache, WINPR_ASSERTING_INT_CAST(UINT8, windowIcon->iconInfo->cacheId),
845 WINPR_ASSERTING_INT_CAST(UINT16, windowIcon->iconInfo->cacheEntry));
849 WLog_Print(xfc->log, WLOG_WARN,
"failed to get icon from cache %02X:%04X",
850 windowIcon->iconInfo->cacheId, windowIcon->iconInfo->cacheEntry);
852 else if (!convert_rail_icon(windowIcon->iconInfo, icon))
854 WLog_Print(xfc->log, WLOG_WARN,
"failed to convert icon for window %08X",
855 orderInfo->windowId);
859 replaceIcon = !!(orderInfo->fieldFlags & WINDOW_ORDER_STATE_NEW);
860 xf_rail_set_window_icon(xfc, railWindow, icon, replaceIcon);
863 xf_rail_return_window(railWindow, FALSE);
867static BOOL xf_rail_window_cached_icon(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo,
871 xfContext* xfc = (xfContext*)context;
872 WINPR_ASSERT(orderInfo);
874 BOOL replaceIcon = 0;
875 xfAppWindow* railWindow = xf_rail_get_window(xfc, orderInfo->windowId, FALSE);
880 WINPR_ASSERT(windowCachedIcon);
882 xfRailIcon* icon = RailIconCache_Lookup(
883 xfc->railIconCache, WINPR_ASSERTING_INT_CAST(UINT8, windowCachedIcon->cachedIcon.cacheId),
884 WINPR_ASSERTING_INT_CAST(UINT16, windowCachedIcon->cachedIcon.cacheEntry));
888 WLog_Print(xfc->log, WLOG_WARN,
"failed to get icon from cache %02X:%04X",
889 windowCachedIcon->cachedIcon.cacheId, windowCachedIcon->cachedIcon.cacheEntry);
893 replaceIcon = !!(orderInfo->fieldFlags & WINDOW_ORDER_STATE_NEW);
894 xf_rail_set_window_icon(xfc, railWindow, icon, replaceIcon);
897 xf_rail_return_window(railWindow, FALSE);
902xf_rail_notify_icon_common(WINPR_ATTR_UNUSED rdpContext* context,
906 WLog_ERR(
"TODO",
"TODO: implement");
907 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_VERSION)
911 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_TIP)
915 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_INFO_TIP)
919 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_STATE)
923 if (orderInfo->fieldFlags & WINDOW_ORDER_ICON)
927 if (orderInfo->fieldFlags & WINDOW_ORDER_CACHED_ICON)
934static BOOL xf_rail_notify_icon_create(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo,
937 return xf_rail_notify_icon_common(context, orderInfo, notifyIconState);
940static BOOL xf_rail_notify_icon_update(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo,
943 return xf_rail_notify_icon_common(context, orderInfo, notifyIconState);
946static BOOL xf_rail_notify_icon_delete(WINPR_ATTR_UNUSED rdpContext* context,
949 WLog_ERR(
"TODO",
"TODO: implement");
954xf_rail_monitored_desktop(WINPR_ATTR_UNUSED rdpContext* context,
958 const UINT32 mask = WINDOW_ORDER_TYPE_DESKTOP | WINDOW_ORDER_FIELD_DESKTOP_HOOKED |
959 WINDOW_ORDER_FIELD_DESKTOP_ARC_BEGAN |
960 WINDOW_ORDER_FIELD_DESKTOP_ARC_COMPLETED |
961 WINDOW_ORDER_FIELD_DESKTOP_ACTIVE_WND | WINDOW_ORDER_FIELD_DESKTOP_ZORDER;
962 xfContext* xfc = (xfContext*)context;
964 if (!context || !orderInfo || !monitoredDesktop)
967 if ((orderInfo->fieldFlags & WINDOW_ORDER_TYPE_DESKTOP) == 0)
969 WLog_Print(xfc->log, WLOG_WARN,
"WINDOW_ORDER_TYPE_DESKTOP flag missing!");
973 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ARC_BEGAN) &&
974 (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_HOOKED))
977 WLog_Print(xfc->log, WLOG_WARN,
978 "TODO: implement WINDOW_ORDER_FIELD_DESKTOP_ARC_BEGAN && "
979 "WINDOW_ORDER_FIELD_DESKTOP_HOOKED");
981 else if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_HOOKED)
983 WLog_Print(xfc->log, WLOG_WARN,
"TODO: implement WINDOW_ORDER_FIELD_DESKTOP_HOOKED");
985 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ARC_COMPLETED)
987 WLog_DBG(TAG,
"WINDOW_ORDER_FIELD_DESKTOP_ARC_COMPLETED -> switch to RAILS mode");
988 if (!xf_rail_enable_remoteapp_mode(xfc))
993 if ((app !=
nullptr) && (strnlen(app, 1) > 0))
995 if (client_rail_server_start_cmd(xfc->rail) != CHANNEL_RC_OK)
999 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ACTIVE_WND)
1001 WLog_Print(xfc->log, WLOG_WARN,
"TODO: implement WINDOW_ORDER_FIELD_DESKTOP_ACTIVE_WND");
1003 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ZORDER)
1005 WLog_Print(xfc->log, WLOG_WARN,
"TODO: implement WINDOW_ORDER_FIELD_DESKTOP_ZORDER");
1007 if (orderInfo->fieldFlags & ~mask)
1009 WLog_Print(xfc->log, WLOG_WARN,
"unknown flags 0x%08" PRIx32
"!", orderInfo->fieldFlags);
1014static BOOL xf_rail_non_monitored_desktop(rdpContext* context,
1017 xfContext* xfc = (xfContext*)context;
1018 const UINT32 mask = WINDOW_ORDER_TYPE_DESKTOP | WINDOW_ORDER_FIELD_DESKTOP_NONE;
1020 if (!context || !orderInfo)
1023 if ((orderInfo->fieldFlags & WINDOW_ORDER_TYPE_DESKTOP) == 0)
1025 WLog_Print(xfc->log, WLOG_WARN,
"TODO: implement WINDOW_ORDER_TYPE_DESKTOP");
1028 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_NONE)
1030 WLog_Print(xfc->log, WLOG_WARN,
"TODO: implement WINDOW_ORDER_FIELD_DESKTOP_NONE");
1032 if (orderInfo->fieldFlags & ~mask)
1034 WLog_Print(xfc->log, WLOG_WARN,
"unknown flags 0x%08" PRIx32
"!", orderInfo->fieldFlags);
1037 return xf_rail_disable_remoteapp_mode(xfc);
1040static void xf_rail_register_update_callbacks(rdpUpdate* update)
1042 WINPR_ASSERT(update);
1044 rdpWindowUpdate* window = update->window;
1045 WINPR_ASSERT(window);
1047 window->WindowCreate = xf_rail_window_common;
1048 window->WindowUpdate = xf_rail_window_common;
1049 window->WindowDelete = xf_rail_window_delete;
1050 window->WindowIcon = xf_rail_window_icon;
1051 window->WindowCachedIcon = xf_rail_window_cached_icon;
1052 window->NotifyIconCreate = xf_rail_notify_icon_create;
1053 window->NotifyIconUpdate = xf_rail_notify_icon_update;
1054 window->NotifyIconDelete = xf_rail_notify_icon_delete;
1055 window->MonitoredDesktop = xf_rail_monitored_desktop;
1056 window->NonMonitoredDesktop = xf_rail_non_monitored_desktop;
1066static UINT xf_rail_server_execute_result(RailClientContext* context,
1069 WINPR_ASSERT(context);
1070 WINPR_ASSERT(execResult);
1072 xfContext* xfc = (xfContext*)context->custom;
1075 if (execResult->execResult != RAIL_EXEC_S_OK)
1077 WLog_ERR(TAG,
"RAIL exec error: execResult=%s [0x%08" PRIx32
"] NtError=0x%X\n",
1078 error_code2str(execResult->execResult), execResult->execResult,
1079 execResult->rawResult);
1080 freerdp_abort_connect_context(&xfc->common.context);
1083 return CHANNEL_RC_OK;
1091static UINT xf_rail_server_system_param(WINPR_ATTR_UNUSED RailClientContext* context,
1095 WLog_ERR(
"TODO",
"TODO: implement");
1096 return CHANNEL_RC_OK;
1104static UINT xf_rail_server_local_move_size(RailClientContext* context,
1110 Window child_window = 0;
1111 WINPR_ASSERT(context);
1112 WINPR_ASSERT(localMoveSize);
1114 xfContext* xfc = (xfContext*)context->custom;
1115 xfAppWindow* appWindow = xf_rail_get_window(xfc, localMoveSize->windowId, FALSE);
1118 return ERROR_INTERNAL_ERROR;
1120 WLog_Print(xfc->log, WLOG_TRACE,
"%s [0x%08" PRIx32
"]",
1121 movetype2str(localMoveSize->moveSizeType), localMoveSize->moveSizeType);
1122 switch (localMoveSize->moveSizeType)
1124 case RAIL_WMSZ_LEFT:
1125 direction = NET_WM_MOVERESIZE_SIZE_LEFT;
1126 x = localMoveSize->posX;
1127 y = localMoveSize->posY;
1130 case RAIL_WMSZ_RIGHT:
1131 direction = NET_WM_MOVERESIZE_SIZE_RIGHT;
1132 x = localMoveSize->posX;
1133 y = localMoveSize->posY;
1137 direction = NET_WM_MOVERESIZE_SIZE_TOP;
1138 x = localMoveSize->posX;
1139 y = localMoveSize->posY;
1142 case RAIL_WMSZ_TOPLEFT:
1143 direction = NET_WM_MOVERESIZE_SIZE_TOPLEFT;
1144 x = localMoveSize->posX;
1145 y = localMoveSize->posY;
1148 case RAIL_WMSZ_TOPRIGHT:
1149 direction = NET_WM_MOVERESIZE_SIZE_TOPRIGHT;
1150 x = localMoveSize->posX;
1151 y = localMoveSize->posY;
1154 case RAIL_WMSZ_BOTTOM:
1155 direction = NET_WM_MOVERESIZE_SIZE_BOTTOM;
1156 x = localMoveSize->posX;
1157 y = localMoveSize->posY;
1160 case RAIL_WMSZ_BOTTOMLEFT:
1161 direction = NET_WM_MOVERESIZE_SIZE_BOTTOMLEFT;
1162 x = localMoveSize->posX;
1163 y = localMoveSize->posY;
1166 case RAIL_WMSZ_BOTTOMRIGHT:
1167 direction = NET_WM_MOVERESIZE_SIZE_BOTTOMRIGHT;
1168 x = localMoveSize->posX;
1169 y = localMoveSize->posY;
1172 case RAIL_WMSZ_MOVE:
1173 direction = NET_WM_MOVERESIZE_MOVE;
1174 XTranslateCoordinates(xfc->display, appWindow->handle, RootWindowOfScreen(xfc->screen),
1175 localMoveSize->posX, localMoveSize->posY, &x, &y, &child_window);
1178 case RAIL_WMSZ_KEYMOVE:
1179 direction = NET_WM_MOVERESIZE_MOVE_KEYBOARD;
1180 x = localMoveSize->posX;
1181 y = localMoveSize->posY;
1185 case RAIL_WMSZ_KEYSIZE:
1186 direction = NET_WM_MOVERESIZE_SIZE_KEYBOARD;
1187 x = localMoveSize->posX;
1188 y = localMoveSize->posY;
1195 if (localMoveSize->isMoveSizeStart)
1196 xf_StartLocalMoveSize(xfc, appWindow, direction, x, y);
1198 xf_EndLocalMoveSize(xfc, appWindow);
1200 xf_rail_return_window(appWindow, FALSE);
1201 return CHANNEL_RC_OK;
1209static UINT xf_rail_server_min_max_info(RailClientContext* context,
1212 WINPR_ASSERT(context);
1213 WINPR_ASSERT(minMaxInfo);
1215 xfContext* xfc = (xfContext*)context->custom;
1216 xfAppWindow* appWindow = xf_rail_get_window(xfc, minMaxInfo->windowId, FALSE);
1220 xf_SetWindowMinMaxInfo(xfc, appWindow, minMaxInfo->maxWidth, minMaxInfo->maxHeight,
1221 minMaxInfo->maxPosX, minMaxInfo->maxPosY, minMaxInfo->minTrackWidth,
1222 minMaxInfo->minTrackHeight, minMaxInfo->maxTrackWidth,
1223 minMaxInfo->maxTrackHeight);
1225 xf_rail_return_window(appWindow, FALSE);
1227 return CHANNEL_RC_OK;
1236xf_rail_server_language_bar_info(WINPR_ATTR_UNUSED RailClientContext* context,
1239 WLog_ERR(
"TODO",
"TODO: implement");
1240 return CHANNEL_RC_OK;
1249xf_rail_server_get_appid_response(WINPR_ATTR_UNUSED RailClientContext* context,
1252 WLog_ERR(
"TODO",
"TODO: implement");
1253 return CHANNEL_RC_OK;
1256static BOOL rail_window_key_equals(
const void* key1,
const void* key2)
1258 const UINT64* k1 = (
const UINT64*)key1;
1259 const UINT64* k2 = (
const UINT64*)key2;
1267static UINT32 rail_window_key_hash(
const void* key)
1269 const UINT64* k1 = (
const UINT64*)key;
1273static void rail_window_free(
void* value)
1275 xfAppWindow* appWindow = (xfAppWindow*)value;
1280 xf_DestroyWindow(appWindow->xfc, appWindow);
1283int xf_rail_init(xfContext* xfc, RailClientContext* rail)
1285 rdpContext* context = (rdpContext*)xfc;
1291 xf_rail_register_update_callbacks(context->update);
1292 rail->custom = (
void*)xfc;
1293 rail->ServerExecuteResult = xf_rail_server_execute_result;
1294 rail->ServerSystemParam = xf_rail_server_system_param;
1295 rail->ServerLocalMoveSize = xf_rail_server_local_move_size;
1296 rail->ServerMinMaxInfo = xf_rail_server_min_max_info;
1297 rail->ServerLanguageBarInfo = xf_rail_server_language_bar_info;
1298 rail->ServerGetAppIdResponse = xf_rail_server_get_appid_response;
1299 xfc->railWindows = HashTable_New(TRUE);
1301 if (!xfc->railWindows)
1304 if (!HashTable_SetHashFunction(xfc->railWindows, rail_window_key_hash))
1307 wObject* obj = HashTable_KeyObject(xfc->railWindows);
1311 wObject* obj = HashTable_ValueObject(xfc->railWindows);
1314 xfc->railIconCache = RailIconCache_New(xfc->common.context.settings);
1316 if (!xfc->railIconCache)
1322 HashTable_Free(xfc->railWindows);
1326int xf_rail_uninit(xfContext* xfc, RailClientContext* rail)
1332 xfc->rail->custom =
nullptr;
1333 xfc->rail =
nullptr;
1336 if (xfc->railWindows)
1338 HashTable_Free(xfc->railWindows);
1339 xfc->railWindows =
nullptr;
1342 if (xfc->railIconCache)
1344 RailIconCache_Free(xfc->railIconCache);
1345 xfc->railIconCache =
nullptr;
1351xfAppWindow* xf_rail_add_window(xfContext* xfc, UINT64
id, INT32 x, INT32 y, UINT32 width,
1352 UINT32 height, UINT32 surfaceId)
1357 xfAppWindow* appWindow = (xfAppWindow*)calloc(1,
sizeof(xfAppWindow));
1362 appWindow->xfc = xfc;
1363 appWindow->windowId = id;
1364 appWindow->surfaceId = surfaceId;
1367 appWindow->width = WINPR_ASSERTING_INT_CAST(
int, width);
1368 appWindow->height = WINPR_ASSERTING_INT_CAST(
int, height);
1370 xf_AppWindowsLock(xfc);
1371 if (!xf_AppWindowCreate(xfc, appWindow))
1374 if (!HashTable_Insert(xfc->railWindows, &appWindow->windowId, (
void*)appWindow))
1378 rail_window_free(appWindow);
1379 xf_AppWindowsUnlock(xfc);
1383BOOL xf_rail_del_window(xfContext* xfc, UINT64
id)
1388 if (!xfc->railWindows)
1392 const BOOL res = HashTable_Remove(xfc->railWindows, &
id);
1397void xf_rail_return_windowFrom(xfAppWindow* window, BOOL alreadyLocked,
const char* file,
1398 const char* fkt,
size_t line)
1406 xfAppWindowsUnlockFrom(window->xfc, file, fkt, line);
WINPR_ATTR_NODISCARD FREERDP_API const char * freerdp_settings_get_string(const rdpSettings *settings, FreeRDP_Settings_Keys_String id)
Returns a immutable string 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.
This struct contains function pointer to initialize/free objects.
OBJECT_FREE_FN fnObjectFree
WINPR_ATTR_NODISCARD OBJECT_EQUALS_FN fnObjectEquals