2011-08-16 08:22:00 +04:00
|
|
|
/**
|
|
|
|
* FreeRDP: A Remote Desktop Protocol Client
|
|
|
|
* X11 RAIL
|
|
|
|
*
|
|
|
|
* Copyright 2011 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2012-08-15 01:20:53 +04:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2011-10-27 21:29:16 +04:00
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/Xutil.h>
|
|
|
|
|
2011-08-18 05:33:22 +04:00
|
|
|
#include <freerdp/utils/event.h>
|
2011-08-20 02:46:10 +04:00
|
|
|
#include <freerdp/utils/hexdump.h>
|
2011-08-22 03:04:21 +04:00
|
|
|
#include <freerdp/utils/rail.h>
|
2011-08-17 10:14:02 +04:00
|
|
|
#include <freerdp/rail/rail.h>
|
|
|
|
|
2011-08-20 02:46:10 +04:00
|
|
|
#include "xf_window.h"
|
2011-08-16 08:22:00 +04:00
|
|
|
#include "xf_rail.h"
|
|
|
|
|
2012-01-02 13:27:04 +04:00
|
|
|
void xf_rail_enable_remoteapp_mode(xfInfo* xfi)
|
|
|
|
{
|
|
|
|
if (xfi->remote_app == false)
|
|
|
|
{
|
|
|
|
xfi->remote_app = true;
|
|
|
|
xfi->drawable = DefaultRootWindow(xfi->display);
|
|
|
|
xf_DestroyWindow(xfi, xfi->window);
|
|
|
|
xfi->window = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-04 02:35:17 +04:00
|
|
|
void xf_rail_disable_remoteapp_mode(xfInfo* xfi)
|
|
|
|
{
|
2012-08-14 23:59:11 +04:00
|
|
|
if (xfi->remote_app == true)
|
|
|
|
{
|
|
|
|
xfi->remote_app = false;
|
|
|
|
xf_create_window(xfi);
|
|
|
|
}
|
2012-08-04 02:35:17 +04:00
|
|
|
}
|
|
|
|
|
2011-12-06 03:16:45 +04:00
|
|
|
void xf_rail_paint(xfInfo* xfi, rdpRail* rail, sint32 uleft, sint32 utop, uint32 uright, uint32 ubottom)
|
2011-08-17 10:14:02 +04:00
|
|
|
{
|
2012-05-08 00:07:34 +04:00
|
|
|
xfWindow* xfw;
|
2012-05-18 01:34:52 +04:00
|
|
|
rdpWindow* window;
|
2011-08-19 20:43:44 +04:00
|
|
|
boolean intersect;
|
2011-09-23 05:23:01 +04:00
|
|
|
uint32 iwidth, iheight;
|
2011-12-06 03:16:45 +04:00
|
|
|
sint32 ileft, itop;
|
|
|
|
uint32 iright, ibottom;
|
|
|
|
sint32 wleft, wtop;
|
|
|
|
uint32 wright, wbottom;
|
2011-08-19 20:43:44 +04:00
|
|
|
|
2011-08-17 10:14:02 +04:00
|
|
|
window_list_rewind(rail->list);
|
|
|
|
|
|
|
|
while (window_list_has_next(rail->list))
|
|
|
|
{
|
|
|
|
window = window_list_get_next(rail->list);
|
|
|
|
xfw = (xfWindow*) window->extra;
|
|
|
|
|
2012-01-14 04:07:27 +04:00
|
|
|
// RDP can have zero width or height windows. X cannot, so we ignore these.
|
|
|
|
|
|
|
|
if (window->windowWidth == 0 || window->windowHeight == 0)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2012-08-04 02:35:17 +04:00
|
|
|
wleft = window->visibleOffsetX;
|
|
|
|
wtop = window->visibleOffsetY;
|
|
|
|
wright = window->visibleOffsetX + window->windowWidth - 1;
|
|
|
|
wbottom = window->visibleOffsetY + window->windowHeight - 1;
|
2011-08-19 20:43:44 +04:00
|
|
|
|
2011-09-23 05:23:01 +04:00
|
|
|
ileft = MAX(uleft, wleft);
|
|
|
|
itop = MAX(utop, wtop);
|
|
|
|
iright = MIN(uright, wright);
|
|
|
|
ibottom = MIN(ubottom, wbottom);
|
2011-08-17 10:14:02 +04:00
|
|
|
|
2011-09-23 05:23:01 +04:00
|
|
|
iwidth = iright - ileft + 1;
|
|
|
|
iheight = ibottom - itop + 1;
|
|
|
|
|
2011-11-19 21:19:16 +04:00
|
|
|
intersect = ((iright > ileft) && (ibottom > itop)) ? true : false;
|
2011-08-17 10:14:02 +04:00
|
|
|
|
2011-08-19 20:43:44 +04:00
|
|
|
if (intersect)
|
|
|
|
{
|
2012-05-18 01:34:52 +04:00
|
|
|
xf_UpdateWindowArea(xfi, xfw, ileft - wleft, itop - wtop, iwidth, iheight);
|
2011-08-19 20:43:44 +04:00
|
|
|
}
|
2011-08-17 10:14:02 +04:00
|
|
|
}
|
2011-09-23 05:23:01 +04:00
|
|
|
}
|
2011-08-19 20:43:44 +04:00
|
|
|
|
2012-08-04 02:35:17 +04:00
|
|
|
void xf_rail_DesktopNonMonitored(rdpRail *rail, rdpWindow* window)
|
|
|
|
{
|
|
|
|
xfInfo* xfi;
|
|
|
|
xfi = (xfInfo*) rail->extra;
|
|
|
|
xf_rail_disable_remoteapp_mode(xfi);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-25 20:00:13 +04:00
|
|
|
static void xf_rail_CreateWindow(rdpRail* rail, rdpWindow* window)
|
2011-08-17 10:14:02 +04:00
|
|
|
{
|
2011-08-18 07:26:12 +04:00
|
|
|
xfInfo* xfi;
|
2011-08-17 10:14:02 +04:00
|
|
|
xfWindow* xfw;
|
2011-08-17 22:13:44 +04:00
|
|
|
|
2011-08-18 07:26:12 +04:00
|
|
|
xfi = (xfInfo*) rail->extra;
|
|
|
|
|
2012-01-02 13:27:04 +04:00
|
|
|
xf_rail_enable_remoteapp_mode(xfi);
|
|
|
|
|
2011-09-23 05:23:01 +04:00
|
|
|
xfw = xf_CreateWindow((xfInfo*) rail->extra, window,
|
2011-08-23 05:22:05 +04:00
|
|
|
window->windowOffsetX, window->windowOffsetY,
|
2011-08-22 07:54:02 +04:00
|
|
|
window->windowWidth, window->windowHeight,
|
|
|
|
window->windowId);
|
|
|
|
|
2011-08-25 01:16:57 +04:00
|
|
|
xf_SetWindowStyle(xfi, xfw, window->style, window->extendedStyle);
|
|
|
|
|
2012-05-08 00:07:34 +04:00
|
|
|
xf_SetWindowText(xfi, xfw, window->title);
|
2011-08-30 23:06:50 +04:00
|
|
|
|
2011-08-17 10:14:02 +04:00
|
|
|
window->extra = (void*) xfw;
|
2011-08-18 00:45:09 +04:00
|
|
|
window->extraId = (void*) xfw->handle;
|
2011-08-17 10:14:02 +04:00
|
|
|
}
|
|
|
|
|
2012-05-25 20:00:13 +04:00
|
|
|
static void xf_rail_MoveWindow(rdpRail* rail, rdpWindow* window)
|
2011-08-18 06:50:49 +04:00
|
|
|
{
|
2011-08-18 07:26:12 +04:00
|
|
|
xfInfo* xfi;
|
2011-08-18 06:50:49 +04:00
|
|
|
xfWindow* xfw;
|
2011-08-18 07:26:12 +04:00
|
|
|
|
|
|
|
xfi = (xfInfo*) rail->extra;
|
2011-08-18 06:50:49 +04:00
|
|
|
xfw = (xfWindow*) window->extra;
|
|
|
|
|
2012-08-04 02:35:17 +04:00
|
|
|
//The rail server like to set the window to a small size when it is minimized even though it is hidden
|
|
|
|
//in some cases this can cause the window not to restore back to its original size. Therefore we dont update
|
|
|
|
//our local window when that rail window state is minimized
|
|
|
|
if (xfw->rail_state == WINDOW_SHOW_MINIMIZED)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Do nothing if window is already in the correct position
|
|
|
|
if ( xfw->left == window->visibleOffsetX &&
|
|
|
|
xfw->top == window->visibleOffsetY &&
|
|
|
|
xfw->width == window->windowWidth &&
|
|
|
|
xfw->height == window->windowHeight)
|
|
|
|
{
|
|
|
|
//Just ensure entire window area is updated to
|
|
|
|
//handle cases where we have drawn locally before getting new bitmap
|
|
|
|
//from the server
|
|
|
|
xf_UpdateWindowArea(xfi, xfw, 0, 0, window->windowWidth, window->windowHeight);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-08 00:07:34 +04:00
|
|
|
xf_MoveWindow(xfi, xfw,
|
2012-08-04 02:35:17 +04:00
|
|
|
window->visibleOffsetX, window->visibleOffsetY,
|
2012-05-08 00:07:34 +04:00
|
|
|
window->windowWidth, window->windowHeight);
|
2011-08-18 06:50:49 +04:00
|
|
|
}
|
|
|
|
|
2012-05-25 20:00:13 +04:00
|
|
|
static void xf_rail_ShowWindow(rdpRail* rail, rdpWindow* window, uint8 state)
|
2011-08-22 07:15:19 +04:00
|
|
|
{
|
|
|
|
xfInfo* xfi;
|
|
|
|
xfWindow* xfw;
|
|
|
|
|
|
|
|
xfi = (xfInfo*) rail->extra;
|
|
|
|
xfw = (xfWindow*) window->extra;
|
|
|
|
|
2012-05-08 00:07:34 +04:00
|
|
|
xf_ShowWindow(xfi, xfw, state);
|
2011-08-22 07:15:19 +04:00
|
|
|
}
|
|
|
|
|
2012-05-25 20:00:13 +04:00
|
|
|
static void xf_rail_SetWindowText(rdpRail* rail, rdpWindow* window)
|
2011-08-20 06:21:09 +04:00
|
|
|
{
|
|
|
|
xfInfo* xfi;
|
|
|
|
xfWindow* xfw;
|
|
|
|
|
|
|
|
xfi = (xfInfo*) rail->extra;
|
|
|
|
xfw = (xfWindow*) window->extra;
|
|
|
|
|
2012-05-08 00:07:34 +04:00
|
|
|
xf_SetWindowText(xfi, xfw, window->title);
|
2011-08-20 06:21:09 +04:00
|
|
|
}
|
|
|
|
|
2012-05-25 20:00:13 +04:00
|
|
|
static void xf_rail_SetWindowIcon(rdpRail* rail, rdpWindow* window, rdpIcon* icon)
|
2011-08-22 01:32:18 +04:00
|
|
|
{
|
|
|
|
xfInfo* xfi;
|
|
|
|
xfWindow* xfw;
|
|
|
|
|
|
|
|
xfi = (xfInfo*) rail->extra;
|
|
|
|
xfw = (xfWindow*) window->extra;
|
|
|
|
|
2011-09-14 03:03:06 +04:00
|
|
|
icon->extra = freerdp_icon_convert(icon->entry->bitsColor, NULL, icon->entry->bitsMask,
|
2011-08-22 06:30:49 +04:00
|
|
|
icon->entry->width, icon->entry->height, icon->entry->bpp, rail->clrconv);
|
|
|
|
|
2011-08-22 01:32:18 +04:00
|
|
|
xf_SetWindowIcon(xfi, xfw, icon);
|
|
|
|
}
|
|
|
|
|
2012-05-25 20:00:13 +04:00
|
|
|
static void xf_rail_SetWindowRects(rdpRail* rail, rdpWindow* window)
|
2011-08-23 00:06:12 +04:00
|
|
|
{
|
|
|
|
xfInfo* xfi;
|
|
|
|
xfWindow* xfw;
|
|
|
|
|
|
|
|
xfi = (xfInfo*) rail->extra;
|
|
|
|
xfw = (xfWindow*) window->extra;
|
|
|
|
|
2011-08-25 22:11:45 +04:00
|
|
|
xf_SetWindowRects(xfi, xfw, window->windowRects, window->numWindowRects);
|
2011-08-23 00:06:12 +04:00
|
|
|
}
|
|
|
|
|
2012-05-25 20:00:13 +04:00
|
|
|
static void xf_rail_SetWindowVisibilityRects(rdpRail* rail, rdpWindow* window)
|
2011-08-30 22:21:16 +04:00
|
|
|
{
|
|
|
|
xfInfo* xfi;
|
|
|
|
xfWindow* xfw;
|
|
|
|
|
|
|
|
xfi = (xfInfo*) rail->extra;
|
|
|
|
xfw = (xfWindow*) window->extra;
|
|
|
|
|
|
|
|
xf_SetWindowVisibilityRects(xfi, xfw, window->windowRects, window->numWindowRects);
|
|
|
|
}
|
|
|
|
|
2012-05-25 20:00:13 +04:00
|
|
|
static void xf_rail_DestroyWindow(rdpRail* rail, rdpWindow* window)
|
2011-08-18 06:31:27 +04:00
|
|
|
{
|
|
|
|
xfWindow* xfw;
|
|
|
|
xfw = (xfWindow*) window->extra;
|
|
|
|
xf_DestroyWindow((xfInfo*) rail->extra, xfw);
|
|
|
|
}
|
|
|
|
|
2011-08-17 10:14:02 +04:00
|
|
|
void xf_rail_register_callbacks(xfInfo* xfi, rdpRail* rail)
|
|
|
|
{
|
|
|
|
rail->extra = (void*) xfi;
|
2011-08-31 10:01:49 +04:00
|
|
|
rail->rail_CreateWindow = xf_rail_CreateWindow;
|
|
|
|
rail->rail_MoveWindow = xf_rail_MoveWindow;
|
|
|
|
rail->rail_ShowWindow = xf_rail_ShowWindow;
|
|
|
|
rail->rail_SetWindowText = xf_rail_SetWindowText;
|
|
|
|
rail->rail_SetWindowIcon = xf_rail_SetWindowIcon;
|
|
|
|
rail->rail_SetWindowRects = xf_rail_SetWindowRects;
|
|
|
|
rail->rail_SetWindowVisibilityRects = xf_rail_SetWindowVisibilityRects;
|
|
|
|
rail->rail_DestroyWindow = xf_rail_DestroyWindow;
|
2012-08-04 02:35:17 +04:00
|
|
|
rail->rail_DesktopNonMonitored = xf_rail_DesktopNonMonitored;
|
2011-08-17 10:14:02 +04:00
|
|
|
}
|
|
|
|
|
2011-08-22 03:04:21 +04:00
|
|
|
static void xf_on_free_rail_client_event(RDP_EVENT* event)
|
|
|
|
{
|
|
|
|
if (event->event_class == RDP_EVENT_CLASS_RAIL)
|
|
|
|
{
|
|
|
|
rail_free_cloned_order(event->event_type, event->user_data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
static void xf_send_rail_client_event(rdpChannels* channels, uint16 event_type, void* param)
|
2011-08-22 03:04:21 +04:00
|
|
|
{
|
|
|
|
RDP_EVENT* out_event = NULL;
|
|
|
|
void * payload = NULL;
|
|
|
|
|
|
|
|
payload = rail_clone_order(event_type, param);
|
|
|
|
if (payload != NULL)
|
|
|
|
{
|
|
|
|
out_event = freerdp_event_new(RDP_EVENT_CLASS_RAIL, event_type,
|
|
|
|
xf_on_free_rail_client_event, payload);
|
2011-10-17 04:10:09 +04:00
|
|
|
freerdp_channels_send_event(channels, out_event);
|
2011-08-22 03:04:21 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-25 02:22:06 +04:00
|
|
|
void xf_rail_send_activate(xfInfo* xfi, Window xwindow, boolean enabled)
|
|
|
|
{
|
2011-10-16 08:50:10 +04:00
|
|
|
rdpRail* rail;
|
2011-10-17 04:10:09 +04:00
|
|
|
rdpChannels* channels;
|
2011-08-25 02:22:06 +04:00
|
|
|
rdpWindow* rail_window;
|
|
|
|
RAIL_ACTIVATE_ORDER activate;
|
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
rail = xfi->_context->rail;
|
|
|
|
channels = xfi->_context->channels;
|
2011-08-25 02:22:06 +04:00
|
|
|
|
2011-10-16 08:50:10 +04:00
|
|
|
rail_window = window_list_get_by_extra_id(rail->list, (void*) xwindow);
|
|
|
|
|
|
|
|
if (rail_window == NULL)
|
|
|
|
return;
|
2011-08-25 02:22:06 +04:00
|
|
|
|
|
|
|
activate.windowId = rail_window->windowId;
|
|
|
|
activate.enabled = enabled;
|
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
xf_send_rail_client_event(channels, RDP_EVENT_TYPE_RAIL_CLIENT_ACTIVATE, &activate);
|
2011-08-25 02:22:06 +04:00
|
|
|
}
|
|
|
|
|
2011-08-22 07:39:56 +04:00
|
|
|
void xf_rail_send_client_system_command(xfInfo* xfi, uint32 windowId, uint16 command)
|
|
|
|
{
|
2011-10-17 04:10:09 +04:00
|
|
|
rdpChannels* channels;
|
2011-08-22 20:14:11 +04:00
|
|
|
RAIL_SYSCOMMAND_ORDER syscommand;
|
2011-08-22 07:39:56 +04:00
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
channels = xfi->_context->channels;
|
2011-08-22 07:39:56 +04:00
|
|
|
|
2011-08-22 20:14:11 +04:00
|
|
|
syscommand.windowId = windowId;
|
|
|
|
syscommand.command = command;
|
2011-08-22 07:39:56 +04:00
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
xf_send_rail_client_event(channels, RDP_EVENT_TYPE_RAIL_CLIENT_SYSCOMMAND, &syscommand);
|
2011-08-22 07:39:56 +04:00
|
|
|
}
|
|
|
|
|
2011-12-03 02:51:30 +04:00
|
|
|
/**
|
|
|
|
* The position of the X window can become out of sync with the RDP window
|
|
|
|
* if the X window is moved locally by the window manager. In this event
|
|
|
|
* send an update to the RDP server informing it of the new window position
|
|
|
|
* and size.
|
|
|
|
*/
|
|
|
|
void xf_rail_adjust_position(xfInfo* xfi, rdpWindow *window)
|
|
|
|
{
|
|
|
|
xfWindow* xfw;
|
|
|
|
rdpChannels* channels;
|
|
|
|
RAIL_WINDOW_MOVE_ORDER window_move;
|
|
|
|
|
|
|
|
xfw = (xfWindow*) window->extra;
|
|
|
|
channels = xfi->_context->channels;
|
|
|
|
|
|
|
|
if (! xfw->is_mapped || xfw->local_move.state != LMS_NOT_ACTIVE)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// If current window position disagrees with RDP window position, send
|
|
|
|
// update to RDP server
|
2012-08-04 02:35:17 +04:00
|
|
|
if ( xfw->left != window->visibleOffsetX ||
|
|
|
|
xfw->top != window->visibleOffsetY ||
|
2011-12-03 02:51:30 +04:00
|
|
|
xfw->width != window->windowWidth ||
|
|
|
|
xfw->height != window->windowHeight)
|
|
|
|
{
|
2012-08-04 02:35:17 +04:00
|
|
|
//Although the rail server can give negative window coordinates when updating windowOffsetX and windowOffsetY,
|
|
|
|
//we can only send unsigned integers to the rail server. Therefore, we always bring negative coordinates up to 0 when
|
|
|
|
//attempting to adjust the rail window.
|
|
|
|
uint32 offsetX = 0;
|
|
|
|
uint32 offsetY = 0;
|
|
|
|
|
|
|
|
if (window->windowOffsetX < 0)
|
|
|
|
offsetX = offsetX - window->windowOffsetX;
|
|
|
|
|
|
|
|
if (window->windowOffsetY < 0)
|
|
|
|
offsetY = offsetY - window->windowOffsetY;
|
|
|
|
//windowOffset corresponds to the window location on the rail server
|
|
|
|
//but our local window is based on the visibleOffset since using the windowOffset
|
|
|
|
//can result in blank areas for a maximized window
|
2011-12-03 02:51:30 +04:00
|
|
|
window_move.windowId = window->windowId;
|
2012-08-04 02:35:17 +04:00
|
|
|
|
|
|
|
//Calculate new offsets for the rail server window
|
|
|
|
//Negative offset correction + rail server window offset + (difference in visibleOffset and new window local offset)
|
|
|
|
window_move.left = offsetX + window->windowOffsetX + (xfw->left - window->visibleOffsetX);
|
|
|
|
window_move.top = offsetY + window->windowOffsetY + (xfw->top - window->visibleOffsetY);
|
|
|
|
|
|
|
|
window_move.right = window_move.left + xfw->width;
|
|
|
|
window_move.bottom = window_move.top + xfw->height;
|
2011-12-03 02:51:30 +04:00
|
|
|
|
2011-12-08 01:49:37 +04:00
|
|
|
DEBUG_X11_LMS("window=0x%X rc={l=%d t=%d r=%d b=%d} w=%u h=%u"
|
|
|
|
" RDP=0x%X rc={l=%d t=%d} w=%d h=%d",
|
2012-08-04 02:35:17 +04:00
|
|
|
(uint32) xfw->handle, window_move.left, window_move.top,
|
|
|
|
window_move.right, window_move.bottom, xfw->width, xfw->height,
|
2011-12-08 01:49:37 +04:00
|
|
|
window->windowId,
|
|
|
|
window->windowOffsetX, window->windowOffsetY,
|
|
|
|
window->windowWidth, window->windowHeight);
|
|
|
|
|
2011-12-03 02:51:30 +04:00
|
|
|
xf_send_rail_client_event(channels, RDP_EVENT_TYPE_RAIL_CLIENT_WINDOW_MOVE, &window_move);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void xf_rail_end_local_move(xfInfo* xfi, rdpWindow *window)
|
|
|
|
{
|
|
|
|
xfWindow* xfw;
|
|
|
|
rdpChannels* channels;
|
|
|
|
RAIL_WINDOW_MOVE_ORDER window_move;
|
|
|
|
rdpInput* input = xfi->instance->input;
|
2012-05-26 02:37:38 +04:00
|
|
|
int x,y;
|
|
|
|
Window root_window;
|
|
|
|
Window child_window;
|
|
|
|
unsigned int mask;
|
|
|
|
int child_x;
|
|
|
|
int child_y;
|
2011-12-03 02:51:30 +04:00
|
|
|
|
|
|
|
xfw = (xfWindow*) window->extra;
|
|
|
|
channels = xfi->_context->channels;
|
|
|
|
|
2011-12-17 01:09:21 +04:00
|
|
|
DEBUG_X11_LMS("window=0x%X rc={l=%d t=%d r=%d b=%d} w=%d h=%d",
|
|
|
|
(uint32) xfw->handle,
|
|
|
|
xfw->left, xfw->top, xfw->right, xfw->bottom,
|
|
|
|
xfw->width, xfw->height);
|
2011-12-03 02:51:30 +04:00
|
|
|
|
2012-08-04 02:35:17 +04:00
|
|
|
//Although the rail server can give negative window coordinates when updating windowOffsetX and windowOffsetY,
|
|
|
|
//we can only send unsigned integers to the rail server. Therefore, we always bring negative coordinates up to 0 when
|
|
|
|
//attempting to adjust the rail window.
|
|
|
|
uint32 offsetX = 0;
|
|
|
|
uint32 offsetY = 0;
|
|
|
|
|
|
|
|
if (window->windowOffsetX < 0)
|
|
|
|
offsetX = offsetX - window->windowOffsetX;
|
|
|
|
|
|
|
|
if (window->windowOffsetY < 0)
|
|
|
|
offsetY = offsetY - window->windowOffsetY;
|
|
|
|
|
2012-05-26 02:37:38 +04:00
|
|
|
/*
|
|
|
|
* For keyboard moves send and explicit update to RDP server
|
|
|
|
*/
|
|
|
|
window_move.windowId = window->windowId;
|
2012-08-04 02:35:17 +04:00
|
|
|
|
|
|
|
//Calculate new offsets for the rail server window
|
|
|
|
//Negative offset correction + rail server window offset + (difference in visibleOffset and new window local offset)
|
|
|
|
window_move.left = offsetX + window->windowOffsetX + (xfw->left - window->visibleOffsetX);
|
|
|
|
window_move.top = offsetY + window->windowOffsetY + (xfw->top - window->visibleOffsetY);
|
|
|
|
|
|
|
|
window_move.right = window_move.left + xfw->width; // In the update to RDP the position is one past the window
|
|
|
|
window_move.bottom = window_move.top + xfw->height;
|
|
|
|
|
2011-12-03 02:51:30 +04:00
|
|
|
|
2012-05-26 02:37:38 +04:00
|
|
|
xf_send_rail_client_event(channels,
|
|
|
|
RDP_EVENT_TYPE_RAIL_CLIENT_WINDOW_MOVE, &window_move);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Simulate button up at new position to end the local move (per RDP spec)
|
|
|
|
*/
|
2011-12-03 02:51:30 +04:00
|
|
|
|
2012-05-26 02:37:38 +04:00
|
|
|
XQueryPointer(xfi->display, xfw->handle,
|
|
|
|
&root_window, &child_window,
|
|
|
|
&x, &y, &child_x, &child_y, &mask);
|
2011-12-03 02:51:30 +04:00
|
|
|
input->MouseEvent(input, PTR_FLAGS_BUTTON1, x, y);
|
2012-08-04 02:35:17 +04:00
|
|
|
|
|
|
|
//only send the mouse coordinates if not a keyboard move or size
|
|
|
|
if ((xfw->local_move.direction != _NET_WM_MOVERESIZE_MOVE_KEYBOARD) &&
|
|
|
|
(xfw->local_move.direction != _NET_WM_MOVERESIZE_SIZE_KEYBOARD))
|
|
|
|
{
|
|
|
|
input->MouseEvent(input, PTR_FLAGS_BUTTON1, x, y);
|
|
|
|
DEBUG_X11_LMS("Mouse coordinates. x= %i, y= %i", x, y);
|
|
|
|
}
|
2012-05-26 02:37:38 +04:00
|
|
|
|
2012-01-14 04:07:27 +04:00
|
|
|
// Proactively update the RAIL window dimensions. There is a race condition where
|
|
|
|
// we can start to receive GDI orders for the new window dimensions before we
|
|
|
|
// receive the RAIL ORDER for the new window size. This avoids that race condition.
|
|
|
|
|
2012-08-04 02:35:17 +04:00
|
|
|
window->windowOffsetX = offsetX + window->windowOffsetX + (xfw->left - window->visibleOffsetX);
|
|
|
|
window->windowOffsetY = offsetY + window->windowOffsetY + (xfw->top - window->visibleOffsetY);
|
2012-01-14 04:07:27 +04:00
|
|
|
window->windowWidth = xfw->width;
|
|
|
|
window->windowHeight = xfw->height;
|
|
|
|
|
2011-12-03 02:51:30 +04:00
|
|
|
xfw->local_move.state = LMS_TERMINATING;
|
|
|
|
}
|
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
void xf_process_rail_get_sysparams_event(xfInfo* xfi, rdpChannels* channels, RDP_EVENT* event)
|
2011-08-16 08:22:00 +04:00
|
|
|
{
|
2011-08-18 05:33:22 +04:00
|
|
|
RAIL_SYSPARAM_ORDER* sysparam;
|
|
|
|
|
|
|
|
sysparam = (RAIL_SYSPARAM_ORDER*) event->user_data;
|
|
|
|
|
|
|
|
sysparam->workArea.left = xfi->workArea.x;
|
|
|
|
sysparam->workArea.top = xfi->workArea.y;
|
|
|
|
sysparam->workArea.right = xfi->workArea.x + xfi->workArea.width;
|
|
|
|
sysparam->workArea.bottom = xfi->workArea.y + xfi->workArea.height;
|
|
|
|
|
|
|
|
sysparam->taskbarPos.left = 0;
|
|
|
|
sysparam->taskbarPos.top = 0;
|
|
|
|
sysparam->taskbarPos.right = 0;
|
|
|
|
sysparam->taskbarPos.bottom = 0;
|
|
|
|
|
2011-11-19 21:19:16 +04:00
|
|
|
sysparam->dragFullWindows = false;
|
2011-08-25 23:05:07 +04:00
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
xf_send_rail_client_event(channels, RDP_EVENT_TYPE_RAIL_CLIENT_SET_SYSPARAMS, sysparam);
|
2011-08-18 05:33:22 +04:00
|
|
|
}
|
|
|
|
|
2011-08-20 06:21:09 +04:00
|
|
|
const char* error_code_names[] =
|
|
|
|
{
|
|
|
|
"RAIL_EXEC_S_OK",
|
|
|
|
"RAIL_EXEC_E_HOOK_NOT_LOADED",
|
|
|
|
"RAIL_EXEC_E_DECODE_FAILED",
|
|
|
|
"RAIL_EXEC_E_NOT_IN_ALLOWLIST",
|
|
|
|
"RAIL_EXEC_E_FILE_NOT_FOUND",
|
|
|
|
"RAIL_EXEC_E_FAIL",
|
|
|
|
"RAIL_EXEC_E_SESSION_LOCKED"
|
|
|
|
};
|
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
void xf_process_rail_exec_result_event(xfInfo* xfi, rdpChannels* channels, RDP_EVENT* event)
|
2011-08-19 18:10:08 +04:00
|
|
|
{
|
2011-08-20 06:21:09 +04:00
|
|
|
RAIL_EXEC_RESULT_ORDER* exec_result;
|
|
|
|
|
|
|
|
exec_result = (RAIL_EXEC_RESULT_ORDER*) event->user_data;
|
2011-08-19 18:10:08 +04:00
|
|
|
|
|
|
|
if (exec_result->execResult != RAIL_EXEC_S_OK)
|
|
|
|
{
|
|
|
|
printf("RAIL exec error: execResult=%s NtError=0x%X\n",
|
|
|
|
error_code_names[exec_result->execResult], exec_result->rawResult);
|
2012-01-02 13:27:04 +04:00
|
|
|
xfi->disconnect = True;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
xf_rail_enable_remoteapp_mode(xfi);
|
2011-08-19 18:10:08 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
void xf_process_rail_server_sysparam_event(xfInfo* xfi, rdpChannels* channels, RDP_EVENT* event)
|
2011-08-20 02:46:10 +04:00
|
|
|
{
|
2011-09-02 21:29:17 +04:00
|
|
|
RAIL_SYSPARAM_ORDER* sysparam = (RAIL_SYSPARAM_ORDER*) event->user_data;
|
2011-08-20 02:46:10 +04:00
|
|
|
|
|
|
|
switch (sysparam->param)
|
|
|
|
{
|
2011-08-22 01:32:18 +04:00
|
|
|
case SPI_SET_SCREEN_SAVE_ACTIVE:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SPI_SET_SCREEN_SAVE_SECURE:
|
|
|
|
break;
|
2011-08-20 02:46:10 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
void xf_process_rail_server_minmaxinfo_event(xfInfo* xfi, rdpChannels* channels, RDP_EVENT* event)
|
2011-08-20 02:46:10 +04:00
|
|
|
{
|
2011-10-16 08:50:10 +04:00
|
|
|
rdpRail* rail;
|
2011-09-01 02:02:22 +04:00
|
|
|
rdpWindow* rail_window = NULL;
|
2011-10-16 08:50:10 +04:00
|
|
|
RAIL_MINMAXINFO_ORDER* minmax = (RAIL_MINMAXINFO_ORDER*) event->user_data;
|
2011-08-20 02:46:10 +04:00
|
|
|
|
2011-10-16 08:50:10 +04:00
|
|
|
rail = ((rdpContext*) xfi->context)->rail;
|
|
|
|
rail_window = window_list_get_by_id(rail->list, minmax->windowId);
|
2011-09-01 02:02:22 +04:00
|
|
|
|
|
|
|
if (rail_window != NULL)
|
|
|
|
{
|
|
|
|
xfWindow * window = NULL;
|
|
|
|
window = (xfWindow *) rail_window->extra;
|
|
|
|
|
2011-09-04 00:38:01 +04:00
|
|
|
DEBUG_X11_LMS("windowId=0x%X maxWidth=%d maxHeight=%d maxPosX=%d maxPosY=%d "
|
2011-09-03 20:50:26 +04:00
|
|
|
"minTrackWidth=%d minTrackHeight=%d maxTrackWidth=%d maxTrackHeight=%d",
|
2011-09-01 02:02:22 +04:00
|
|
|
minmax->windowId, minmax->maxWidth, minmax->maxHeight,
|
|
|
|
(sint16)minmax->maxPosX, (sint16)minmax->maxPosY,
|
|
|
|
minmax->minTrackWidth, minmax->minTrackHeight,
|
|
|
|
minmax->maxTrackWidth, minmax->maxTrackHeight);
|
|
|
|
|
|
|
|
xf_SetWindowMinMaxInfo(xfi, window, minmax->maxWidth, minmax->maxHeight, minmax->maxPosX, minmax->maxPosY,
|
2011-09-03 20:50:26 +04:00
|
|
|
minmax->minTrackWidth, minmax->minTrackHeight, minmax->maxTrackWidth, minmax->maxTrackHeight);
|
2011-09-01 02:02:22 +04:00
|
|
|
}
|
2011-08-20 02:46:10 +04:00
|
|
|
}
|
|
|
|
|
2011-08-22 01:32:18 +04:00
|
|
|
const char* movetype_names[] =
|
2011-08-20 02:46:10 +04:00
|
|
|
{
|
|
|
|
"(invalid)",
|
|
|
|
"RAIL_WMSZ_LEFT",
|
|
|
|
"RAIL_WMSZ_RIGHT",
|
|
|
|
"RAIL_WMSZ_TOP",
|
|
|
|
"RAIL_WMSZ_TOPLEFT",
|
|
|
|
"RAIL_WMSZ_TOPRIGHT",
|
|
|
|
"RAIL_WMSZ_BOTTOM",
|
|
|
|
"RAIL_WMSZ_BOTTOMLEFT",
|
|
|
|
"RAIL_WMSZ_BOTTOMRIGHT",
|
|
|
|
"RAIL_WMSZ_MOVE",
|
|
|
|
"RAIL_WMSZ_KEYMOVE",
|
2011-08-22 01:32:18 +04:00
|
|
|
"RAIL_WMSZ_KEYSIZE"
|
|
|
|
};
|
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
void xf_process_rail_server_localmovesize_event(xfInfo* xfi, rdpChannels* channels, RDP_EVENT* event)
|
2011-08-22 01:32:18 +04:00
|
|
|
{
|
2011-10-16 08:50:10 +04:00
|
|
|
rdpRail* rail;
|
2011-09-01 02:02:22 +04:00
|
|
|
rdpWindow* rail_window = NULL;
|
2011-09-23 05:23:01 +04:00
|
|
|
RAIL_LOCALMOVESIZE_ORDER* movesize = (RAIL_LOCALMOVESIZE_ORDER*) event->user_data;
|
2011-11-29 01:50:08 +04:00
|
|
|
int direction = 0;
|
2011-12-03 02:51:30 +04:00
|
|
|
Window child_window;
|
|
|
|
int x,y;
|
2011-09-01 02:02:22 +04:00
|
|
|
|
2011-10-16 08:50:10 +04:00
|
|
|
rail = ((rdpContext*) xfi->context)->rail;
|
|
|
|
rail_window = window_list_get_by_id(rail->list, movesize->windowId);
|
2011-09-01 02:02:22 +04:00
|
|
|
|
|
|
|
if (rail_window != NULL)
|
|
|
|
{
|
2011-12-03 02:51:30 +04:00
|
|
|
xfWindow* xfw = NULL;
|
|
|
|
xfw = (xfWindow*) rail_window->extra;
|
2011-09-01 02:02:22 +04:00
|
|
|
|
2011-09-04 00:38:01 +04:00
|
|
|
DEBUG_X11_LMS("windowId=0x%X isMoveSizeStart=%d moveSizeType=%s PosX=%d PosY=%d",
|
2011-09-01 02:02:22 +04:00
|
|
|
movesize->windowId, movesize->isMoveSizeStart,
|
2011-09-23 05:23:01 +04:00
|
|
|
movetype_names[movesize->moveSizeType], (sint16) movesize->posX, (sint16) movesize->posY);
|
2011-09-01 02:02:22 +04:00
|
|
|
|
2011-11-29 01:50:08 +04:00
|
|
|
switch (movesize->moveSizeType)
|
|
|
|
{
|
|
|
|
case RAIL_WMSZ_LEFT: //0x1
|
|
|
|
direction = _NET_WM_MOVERESIZE_SIZE_LEFT;
|
2011-12-03 02:51:30 +04:00
|
|
|
x = movesize->posX;
|
|
|
|
y = movesize->posY;
|
2011-11-29 01:50:08 +04:00
|
|
|
break;
|
|
|
|
case RAIL_WMSZ_RIGHT: //0x2
|
|
|
|
direction = _NET_WM_MOVERESIZE_SIZE_RIGHT;
|
2011-12-03 02:51:30 +04:00
|
|
|
x = movesize->posX;
|
|
|
|
y = movesize->posY;
|
2011-11-29 01:50:08 +04:00
|
|
|
break;
|
|
|
|
case RAIL_WMSZ_TOP: //0x3
|
|
|
|
direction = _NET_WM_MOVERESIZE_SIZE_TOP;
|
2011-12-03 02:51:30 +04:00
|
|
|
x = movesize->posX;
|
|
|
|
y = movesize->posY;
|
2011-11-29 01:50:08 +04:00
|
|
|
break;
|
|
|
|
case RAIL_WMSZ_TOPLEFT: //0x4
|
|
|
|
direction = _NET_WM_MOVERESIZE_SIZE_TOPLEFT;
|
2011-12-03 02:51:30 +04:00
|
|
|
x = movesize->posX;
|
|
|
|
y = movesize->posY;
|
2011-11-29 01:50:08 +04:00
|
|
|
break;
|
|
|
|
case RAIL_WMSZ_TOPRIGHT: //0x5
|
|
|
|
direction = _NET_WM_MOVERESIZE_SIZE_TOPRIGHT;
|
2011-12-03 02:51:30 +04:00
|
|
|
x = movesize->posX;
|
|
|
|
y = movesize->posY;
|
2011-11-29 01:50:08 +04:00
|
|
|
break;
|
|
|
|
case RAIL_WMSZ_BOTTOM: //0x6
|
|
|
|
direction = _NET_WM_MOVERESIZE_SIZE_BOTTOM;
|
2011-12-03 02:51:30 +04:00
|
|
|
x = movesize->posX;
|
|
|
|
y = movesize->posY;
|
2011-11-29 01:50:08 +04:00
|
|
|
break;
|
|
|
|
case RAIL_WMSZ_BOTTOMLEFT: //0x7
|
|
|
|
direction = _NET_WM_MOVERESIZE_SIZE_BOTTOMLEFT;
|
2011-12-03 02:51:30 +04:00
|
|
|
x = movesize->posX;
|
|
|
|
y = movesize->posY;
|
2011-11-29 01:50:08 +04:00
|
|
|
break;
|
|
|
|
case RAIL_WMSZ_BOTTOMRIGHT: //0x8
|
|
|
|
direction = _NET_WM_MOVERESIZE_SIZE_BOTTOMRIGHT;
|
2011-12-03 02:51:30 +04:00
|
|
|
x = movesize->posX;
|
|
|
|
y = movesize->posY;
|
2011-11-29 01:50:08 +04:00
|
|
|
break;
|
|
|
|
case RAIL_WMSZ_MOVE: //0x9
|
|
|
|
direction = _NET_WM_MOVERESIZE_MOVE;
|
2011-12-08 01:49:37 +04:00
|
|
|
XTranslateCoordinates(xfi->display, xfw->handle,
|
|
|
|
RootWindowOfScreen(xfi->screen),
|
2011-12-03 02:51:30 +04:00
|
|
|
movesize->posX, movesize->posY, &x, &y, &child_window);
|
2011-11-29 01:50:08 +04:00
|
|
|
break;
|
|
|
|
case RAIL_WMSZ_KEYMOVE: //0xA
|
|
|
|
direction = _NET_WM_MOVERESIZE_MOVE_KEYBOARD;
|
2011-12-03 02:51:30 +04:00
|
|
|
x = movesize->posX;
|
|
|
|
y = movesize->posY;
|
2012-05-26 02:37:38 +04:00
|
|
|
/* FIXME: local keyboard moves not working */
|
|
|
|
return;
|
2011-11-29 01:50:08 +04:00
|
|
|
case RAIL_WMSZ_KEYSIZE: //0xB
|
|
|
|
direction = _NET_WM_MOVERESIZE_SIZE_KEYBOARD;
|
2011-12-03 02:51:30 +04:00
|
|
|
x = movesize->posX;
|
|
|
|
y = movesize->posY;
|
2012-05-26 02:37:38 +04:00
|
|
|
/* FIXME: local keyboard moves not working */
|
|
|
|
return;
|
2011-11-29 01:50:08 +04:00
|
|
|
}
|
|
|
|
|
2011-09-01 02:02:22 +04:00
|
|
|
if (movesize->isMoveSizeStart)
|
2011-11-29 01:50:08 +04:00
|
|
|
{
|
2011-12-03 02:51:30 +04:00
|
|
|
xf_StartLocalMoveSize(xfi, xfw, direction, x, y);
|
2011-11-29 01:50:08 +04:00
|
|
|
} else {
|
2011-12-17 01:09:21 +04:00
|
|
|
xf_EndLocalMoveSize(xfi, xfw);
|
2011-11-29 01:50:08 +04:00
|
|
|
}
|
2011-09-01 02:02:22 +04:00
|
|
|
}
|
2011-08-20 02:46:10 +04:00
|
|
|
}
|
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
void xf_process_rail_appid_resp_event(xfInfo* xfi, rdpChannels* channels, RDP_EVENT* event)
|
2011-08-20 02:46:10 +04:00
|
|
|
{
|
|
|
|
RAIL_GET_APPID_RESP_ORDER* appid_resp =
|
|
|
|
(RAIL_GET_APPID_RESP_ORDER*)event->user_data;
|
|
|
|
|
|
|
|
printf("Server Application ID Response PDU: windowId=0x%X "
|
|
|
|
"applicationId=(length=%d dump)\n",
|
2011-08-22 01:32:18 +04:00
|
|
|
appid_resp->windowId, appid_resp->applicationId.length);
|
|
|
|
|
2011-08-20 02:46:10 +04:00
|
|
|
freerdp_hexdump(appid_resp->applicationId.string, appid_resp->applicationId.length);
|
|
|
|
}
|
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
void xf_process_rail_langbarinfo_event(xfInfo* xfi, rdpChannels* channels, RDP_EVENT* event)
|
2011-08-20 02:46:10 +04:00
|
|
|
{
|
|
|
|
RAIL_LANGBAR_INFO_ORDER* langbar =
|
2011-08-22 01:32:18 +04:00
|
|
|
(RAIL_LANGBAR_INFO_ORDER*) event->user_data;
|
2011-08-20 02:46:10 +04:00
|
|
|
|
|
|
|
printf("Language Bar Information PDU: languageBarStatus=0x%X\n",
|
2011-08-22 01:32:18 +04:00
|
|
|
langbar->languageBarStatus);
|
2011-08-20 02:46:10 +04:00
|
|
|
}
|
|
|
|
|
2011-10-17 04:10:09 +04:00
|
|
|
void xf_process_rail_event(xfInfo* xfi, rdpChannels* channels, RDP_EVENT* event)
|
2011-08-18 05:33:22 +04:00
|
|
|
{
|
|
|
|
switch (event->event_type)
|
|
|
|
{
|
|
|
|
case RDP_EVENT_TYPE_RAIL_CHANNEL_GET_SYSPARAMS:
|
2011-10-17 04:10:09 +04:00
|
|
|
xf_process_rail_get_sysparams_event(xfi, channels, event);
|
2011-08-18 05:33:22 +04:00
|
|
|
break;
|
|
|
|
|
2011-08-19 18:10:08 +04:00
|
|
|
case RDP_EVENT_TYPE_RAIL_CHANNEL_EXEC_RESULTS:
|
2011-10-17 04:10:09 +04:00
|
|
|
xf_process_rail_exec_result_event(xfi, channels, event);
|
2011-08-19 18:10:08 +04:00
|
|
|
break;
|
|
|
|
|
2011-08-20 02:46:10 +04:00
|
|
|
case RDP_EVENT_TYPE_RAIL_CHANNEL_SERVER_SYSPARAM:
|
2011-10-17 04:10:09 +04:00
|
|
|
xf_process_rail_server_sysparam_event(xfi, channels, event);
|
2011-08-20 02:46:10 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case RDP_EVENT_TYPE_RAIL_CHANNEL_SERVER_MINMAXINFO:
|
2011-10-17 04:10:09 +04:00
|
|
|
xf_process_rail_server_minmaxinfo_event(xfi, channels, event);
|
2011-08-20 02:46:10 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case RDP_EVENT_TYPE_RAIL_CHANNEL_SERVER_LOCALMOVESIZE:
|
2011-10-17 04:10:09 +04:00
|
|
|
xf_process_rail_server_localmovesize_event(xfi, channels, event);
|
2011-08-20 02:46:10 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case RDP_EVENT_TYPE_RAIL_CHANNEL_APPID_RESP:
|
2011-10-17 04:10:09 +04:00
|
|
|
xf_process_rail_appid_resp_event(xfi, channels, event);
|
2011-08-20 02:46:10 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case RDP_EVENT_TYPE_RAIL_CHANNEL_LANGBARINFO:
|
2011-10-17 04:10:09 +04:00
|
|
|
xf_process_rail_langbarinfo_event(xfi, channels, event);
|
2011-08-20 02:46:10 +04:00
|
|
|
break;
|
|
|
|
|
2011-08-18 05:33:22 +04:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2011-08-16 08:22:00 +04:00
|
|
|
}
|