/** * FreeRDP: A Remote Desktop Protocol Implementation * RAIL Windows * * Copyright 2011 Marc-Andre Moreau * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include "librail.h" #include struct _WINDOW_STYLE { UINT32 style; const char* name; BOOL multi; }; typedef struct _WINDOW_STYLE WINDOW_STYLE; static const WINDOW_STYLE WINDOW_STYLES[] = { { WS_BORDER, "WS_BORDER", FALSE }, { WS_CAPTION, "WS_CAPTION", FALSE }, { WS_CHILD, "WS_CHILD", FALSE }, { WS_CLIPCHILDREN, "WS_CLIPCHILDREN", FALSE }, { WS_CLIPSIBLINGS, "WS_CLIPSIBLINGS", FALSE }, { WS_DISABLED, "WS_DISABLED", FALSE }, { WS_DLGFRAME, "WS_DLGFRAME", FALSE }, { WS_GROUP, "WS_GROUP", FALSE }, { WS_HSCROLL, "WS_HSCROLL", FALSE }, { WS_ICONIC, "WS_ICONIC", FALSE }, { WS_MAXIMIZE, "WS_MAXIMIZE", FALSE }, { WS_MAXIMIZEBOX, "WS_MAXIMIZEBOX", FALSE }, { WS_MINIMIZE, "WS_MINIMIZE", FALSE }, { WS_MINIMIZEBOX, "WS_MINIMIZEBOX", FALSE }, { WS_OVERLAPPED, "WS_OVERLAPPED", FALSE }, { WS_OVERLAPPEDWINDOW, "WS_OVERLAPPEDWINDOW", TRUE }, { WS_POPUP, "WS_POPUP", FALSE }, { WS_POPUPWINDOW, "WS_POPUPWINDOW", TRUE }, { WS_SIZEBOX, "WS_SIZEBOX", FALSE }, { WS_SYSMENU, "WS_SYSMENU", FALSE }, { WS_TABSTOP, "WS_TABSTOP", FALSE }, { WS_THICKFRAME, "WS_THICKFRAME", FALSE }, { WS_VISIBLE, "WS_VISIBLE", FALSE } }; static const WINDOW_STYLE EXTENDED_WINDOW_STYLES[] = { { WS_EX_ACCEPTFILES, "WS_EX_ACCEPTFILES", FALSE }, { WS_EX_APPWINDOW, "WS_EX_APPWINDOW", FALSE }, { WS_EX_CLIENTEDGE, "WS_EX_CLIENTEDGE", FALSE }, { WS_EX_COMPOSITED, "WS_EX_COMPOSITED", FALSE }, { WS_EX_CONTEXTHELP, "WS_EX_CONTEXTHELP", FALSE }, { WS_EX_CONTROLPARENT, "WS_EX_CONTROLPARENT", FALSE }, { WS_EX_DLGMODALFRAME, "WS_EX_DLGMODALFRAME", FALSE }, { WS_EX_LAYERED, "WS_EX_LAYERED", FALSE }, { WS_EX_LAYOUTRTL, "WS_EX_LAYOUTRTL", FALSE }, { WS_EX_LEFT, "WS_EX_LEFT", FALSE }, { WS_EX_LEFTSCROLLBAR, "WS_EX_LEFTSCROLLBAR", FALSE }, { WS_EX_LTRREADING, "WS_EX_LTRREADING", FALSE }, { WS_EX_MDICHILD, "WS_EX_MDICHILD", FALSE }, { WS_EX_NOACTIVATE, "WS_EX_NOACTIVATE", FALSE }, { WS_EX_NOINHERITLAYOUT, "WS_EX_NOINHERITLAYOUT", FALSE }, { WS_EX_NOPARENTNOTIFY, "WS_EX_NOPARENTNOTIFY", FALSE }, { WS_EX_OVERLAPPEDWINDOW, "WS_EX_OVERLAPPEDWINDOW", TRUE }, { WS_EX_PALETTEWINDOW, "WS_EX_PALETTEWINDOW", TRUE }, { WS_EX_RIGHT, "WS_EX_RIGHT", FALSE }, { WS_EX_RIGHTSCROLLBAR, "WS_EX_RIGHTSCROLLBAR", FALSE }, { WS_EX_RTLREADING, "WS_EX_RTLREADING", FALSE }, { WS_EX_STATICEDGE, "WS_EX_STATICEDGE", FALSE }, { WS_EX_TOOLWINDOW, "WS_EX_TOOLWINDOW", FALSE }, { WS_EX_TOPMOST, "WS_EX_TOPMOST", FALSE }, { WS_EX_TRANSPARENT, "WS_EX_TRANSPARENT", FALSE }, { WS_EX_WINDOWEDGE, "WS_EX_WINDOWEDGE", FALSE } }; void print_window_styles(UINT32 style) { int i; fprintf(stderr, "Window Styles:\n{\n"); for (i = 0; i < ARRAYSIZE(WINDOW_STYLES); i++) { if (style & WINDOW_STYLES[i].style) { if (WINDOW_STYLES[i].multi) { if ((style & WINDOW_STYLES[i].style) != WINDOW_STYLES[i].style) continue; } fprintf(stderr, "\t%s\n", WINDOW_STYLES[i].name); } } fprintf(stderr, "}\n"); } void print_extended_window_styles(UINT32 style) { int i; fprintf(stderr, "Extended Window Styles:\n{\n"); for (i = 0; i < ARRAYSIZE(EXTENDED_WINDOW_STYLES); i++) { if (style & EXTENDED_WINDOW_STYLES[i].style) { if (EXTENDED_WINDOW_STYLES[i].multi) { if ((style & EXTENDED_WINDOW_STYLES[i].style) != EXTENDED_WINDOW_STYLES[i].style) continue; } fprintf(stderr, "\t%s\n", EXTENDED_WINDOW_STYLES[i].name); } } fprintf(stderr, "}\n"); } void window_state_update(rdpWindow* window, WINDOW_ORDER_INFO* orderInfo, WINDOW_STATE_ORDER* window_state) { window->fieldFlags = orderInfo->fieldFlags; if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_OWNER) { window->ownerWindowId = window_state->ownerWindowId; DEBUG_RAIL("ownerWindowId:0x%08X", window->ownerWindowId); } DEBUG_RAIL("windowId=0x%X ownerWindowId=0x%X", window->windowId, window->ownerWindowId); if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_STYLE) { window->style = window_state->style; window->extendedStyle = window_state->extendedStyle; #ifdef WITH_DEBUG_RAIL print_window_styles(window->style); print_extended_window_styles(window->extendedStyle); #endif } if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_SHOW) { window->showState = window_state->showState; DEBUG_RAIL("ShowState:%d", window->showState); } if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_TITLE) { window->titleInfo.length = window_state->titleInfo.length; window->titleInfo.string = malloc(window_state->titleInfo.length); memcpy(window->titleInfo.string, window_state->titleInfo.string, window->titleInfo.length); #ifdef WITH_DEBUG_RAIL winpr_HexDump(window->titleInfo.string, window->titleInfo.length); #endif } if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_OFFSET) { window->clientOffsetX = window_state->clientOffsetX; window->clientOffsetY = window_state->clientOffsetY; DEBUG_RAIL("Client Area Offset: (%d, %d)", window->clientOffsetX, window->clientOffsetY); } if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_SIZE) { window->clientAreaWidth = window_state->clientAreaWidth; window->clientAreaHeight = window_state->clientAreaHeight; DEBUG_RAIL("Client Area Size: (%d, %d)", window->clientAreaWidth, window->clientAreaHeight); } if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_RP_CONTENT) { window->RPContent = window_state->RPContent; } if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_ROOT_PARENT) { window->rootParentHandle = window_state->rootParentHandle; } if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_WND_OFFSET) { window->windowOffsetX = window_state->windowOffsetX; window->windowOffsetY = window_state->windowOffsetY; DEBUG_RAIL("Window Offset: (%d, %d)", window->windowOffsetX, window->windowOffsetY); } if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_WND_CLIENT_DELTA) { window->windowClientDeltaX = window_state->windowClientDeltaX; window->windowClientDeltaY = window_state->windowClientDeltaY; DEBUG_RAIL("Window Client Delta: (%d, %d)", window->windowClientDeltaX, window->windowClientDeltaY); } if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_WND_SIZE) { window->windowWidth = window_state->windowWidth; window->windowHeight = window_state->windowHeight; DEBUG_RAIL("Window Size: (%d, %d)", window->windowWidth, window->windowHeight); } if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_WND_RECTS) { int i; if (window->windowRects != NULL) free(window->windowRects); window->windowRects = window_state->windowRects; window->numWindowRects = window_state->numWindowRects; for (i = 0; i < (int) window_state->numWindowRects; i++) { DEBUG_RAIL("Window Rect #%d: left:%d top:%d right:%d bottom:%d", i, window_state->windowRects[i].left, window_state->windowRects[i].top, window_state->windowRects[i].right, window_state->windowRects[i].bottom); } } if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_VIS_OFFSET) { window->visibleOffsetX = window_state->visibleOffsetX; window->visibleOffsetY = window_state->visibleOffsetY; DEBUG_RAIL("Window Visible Offset: (%d, %d)", window->visibleOffsetX, window->visibleOffsetY); } if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_VISIBILITY) { int i; if (window->visibilityRects != NULL) free(window->visibilityRects); window->visibilityRects = window_state->visibilityRects; window->numVisibilityRects = window_state->numVisibilityRects; for (i = 0; i < (int) window_state->numVisibilityRects; i++) { DEBUG_RAIL("Visibility Rect #%d: left:%d top:%d right:%d bottom:%d", i, window_state->visibilityRects[i].left, window_state->visibilityRects[i].top, window_state->visibilityRects[i].right, window_state->visibilityRects[i].bottom); } } } void rail_CreateWindow(rdpRail* rail, rdpWindow* window) { if (window->titleInfo.length > 0) { ConvertFromUnicode(CP_UTF8, 0, (WCHAR*) window->titleInfo.string, window->titleInfo.length / 2, &window->title, 0, NULL, NULL); } else { window->title = (char*) malloc(sizeof("RAIL")); memcpy(window->title, "RAIL", sizeof("RAIL")); } IFCALL(rail->rail_CreateWindow, rail, window); if (window->fieldFlags & WINDOW_ORDER_FIELD_WND_RECTS) { IFCALL(rail->rail_SetWindowRects, rail, window); } if (window->fieldFlags & WINDOW_ORDER_FIELD_VISIBILITY) { IFCALL(rail->rail_SetWindowVisibilityRects, rail, window); } } void rail_UpdateWindow(rdpRail* rail, rdpWindow* window) { if (window->fieldFlags & WINDOW_ORDER_FIELD_OWNER) { } if (window->fieldFlags & WINDOW_ORDER_FIELD_STYLE) { } if (window->fieldFlags & WINDOW_ORDER_FIELD_SHOW) { IFCALL(rail->rail_ShowWindow, rail, window, window->showState); } if (window->fieldFlags & WINDOW_ORDER_FIELD_TITLE) { if (window->title != NULL) { free(window->title); window->title = NULL; } ConvertFromUnicode(CP_UTF8, 0, (WCHAR*) window->titleInfo.string, window->titleInfo.length / 2, &window->title, 0, NULL, NULL); IFCALL(rail->rail_SetWindowText, rail, window); } if (window->fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_OFFSET) { } if (window->fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_SIZE) { } if (window->fieldFlags & WINDOW_ORDER_FIELD_RP_CONTENT) { } if (window->fieldFlags & WINDOW_ORDER_FIELD_ROOT_PARENT) { } if ((window->fieldFlags & WINDOW_ORDER_FIELD_WND_OFFSET) || (window->fieldFlags & WINDOW_ORDER_FIELD_WND_SIZE)) { IFCALL(rail->rail_MoveWindow, rail, window); } if (window->fieldFlags & WINDOW_ORDER_FIELD_WND_CLIENT_DELTA) { } if (window->fieldFlags & WINDOW_ORDER_FIELD_WND_RECTS) { IFCALL(rail->rail_SetWindowRects, rail, window); } if (window->fieldFlags & WINDOW_ORDER_FIELD_VIS_OFFSET) { } if (window->fieldFlags & WINDOW_ORDER_FIELD_VISIBILITY) { IFCALL(rail->rail_SetWindowVisibilityRects, rail, window); } } void rail_DestroyWindow(rdpRail* rail, rdpWindow* window) { IFCALL(rail->rail_DestroyWindow, rail, window); if (window) { free(window); } }