mirror of
https://github.com/kasmtech/KasmVNC.git
synced 2025-01-12 17:08:28 +01:00
25b8e64adb
This change adds support for the VMware Mouse Position pseudo-encoding[1], which is used to notify VNC clients when X11 clients call `XWarpPointer()`[2]. This function is called by SDL (and other similar libraries) when they detect that the server does not support native relative motion, like some RFB clients. With this, RFB clients can choose to adjust the local cursor position under certain circumstances to match what the server has set. For instance, if pointer lock has been enabled on the client's machine and the cursor is not being drawn locally, the local position of the cursor is irrelevant, so the RFB client can use what the server sends as the canonical absolute position of the cursor. This ultimately enables the possibility of games (especially FPS games) to behave how users expect (if the clients implement the corresponding change). Part of: #619 1: https://github.com/rfbproto/rfbproto/blob/master/rfbproto.rst#vmware-cursor-position-pseudo-encoding 2: https://tronche.com/gui/x/xlib/input/XWarpPointer.html 3: https://hg.libsdl.org/SDL/file/28e3b60e2131/src/events/SDL_mouse.c#l804
174 lines
5.9 KiB
C++
174 lines
5.9 KiB
C++
/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved.
|
|
* Copyright 2009-2014 Pierre Ossman for Cendio AB
|
|
*
|
|
* This is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This software is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this software; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
|
|
* USA.
|
|
*/
|
|
//
|
|
// SMsgWriter - class for writing RFB messages on the server side.
|
|
//
|
|
|
|
#ifndef __RFB_SMSGWRITER_H__
|
|
#define __RFB_SMSGWRITER_H__
|
|
|
|
#include <rdr/types.h>
|
|
#include <rfb/encodings.h>
|
|
#include <rfb/ScreenSet.h>
|
|
|
|
namespace rdr { class OutStream; }
|
|
|
|
namespace rfb {
|
|
|
|
class ConnParams;
|
|
struct ScreenSet;
|
|
|
|
class SMsgWriter {
|
|
public:
|
|
SMsgWriter(ConnParams* cp, rdr::OutStream* os);
|
|
virtual ~SMsgWriter();
|
|
|
|
// writeServerInit() must only be called at the appropriate time in the
|
|
// protocol initialisation.
|
|
void writeServerInit();
|
|
|
|
// Methods to write normal protocol messages
|
|
|
|
// writeSetColourMapEntries() writes a setColourMapEntries message, using
|
|
// the given colour entries.
|
|
void writeSetColourMapEntries(int firstColour, int nColours,
|
|
const rdr::U16 red[],
|
|
const rdr::U16 green[],
|
|
const rdr::U16 blue[]);
|
|
|
|
// writeBell() and writeServerCutText() do the obvious thing.
|
|
void writeBell();
|
|
void writeServerCutText(const char* str, int len);
|
|
void writeStats(const char* str, int len);
|
|
|
|
// writeFence() sends a new fence request or response to the client.
|
|
void writeFence(rdr::U32 flags, unsigned len, const char data[]);
|
|
|
|
// writeEndOfContinuousUpdates() indicates that we have left continuous
|
|
// updates mode.
|
|
void writeEndOfContinuousUpdates();
|
|
|
|
// writeSetDesktopSize() won't actually write immediately, but will
|
|
// write the relevant pseudo-rectangle as part of the next update.
|
|
bool writeSetDesktopSize();
|
|
// Same thing for the extended version. The first version queues up a
|
|
// generic update of the current server state, but the second queues a
|
|
// specific message.
|
|
bool writeExtendedDesktopSize();
|
|
bool writeExtendedDesktopSize(rdr::U16 reason, rdr::U16 result,
|
|
int fb_width, int fb_height,
|
|
const ScreenSet& layout);
|
|
|
|
bool writeSetDesktopName();
|
|
|
|
// Like setDesktopSize, we can't just write out a cursor message
|
|
// immediately.
|
|
bool writeSetCursor();
|
|
bool writeSetXCursor();
|
|
bool writeSetCursorWithAlpha();
|
|
|
|
// Notifies the client that the cursor pointer was moved by the server.
|
|
void writeCursorPos();
|
|
|
|
// Same for LED state message
|
|
bool writeLEDState();
|
|
|
|
// And QEMU keyboard event handshake
|
|
bool writeQEMUKeyEvent();
|
|
|
|
// needFakeUpdate() returns true when an immediate update is needed in
|
|
// order to flush out pseudo-rectangles to the client.
|
|
bool needFakeUpdate();
|
|
|
|
// needNoDataUpdate() returns true when an update without any
|
|
// framebuffer changes need to be sent (using writeNoDataUpdate()).
|
|
// Commonly this is an update that modifies the size of the framebuffer
|
|
// or the screen layout.
|
|
bool needNoDataUpdate();
|
|
|
|
// writeNoDataUpdate() write a framebuffer update containing only
|
|
// pseudo-rectangles.
|
|
void writeNoDataUpdate();
|
|
|
|
// writeFramebufferUpdateStart() initiates an update which you can fill
|
|
// in using writeCopyRect() and encoders. Finishing the update by calling
|
|
// writeFramebufferUpdateEnd().
|
|
void writeFramebufferUpdateStart(int nRects);
|
|
void writeFramebufferUpdateEnd();
|
|
|
|
// There is no explicit encoder for CopyRect rects.
|
|
void writeCopyRect(const Rect& r, int srcX, int srcY);
|
|
|
|
// Encoders should call these to mark the start and stop of individual
|
|
// rects.
|
|
void startRect(const Rect& r, int enc);
|
|
void endRect();
|
|
|
|
protected:
|
|
void startMsg(int type);
|
|
void endMsg();
|
|
|
|
void writePseudoRects();
|
|
void writeNoDataRects();
|
|
|
|
void writeSetDesktopSizeRect(int width, int height);
|
|
void writeExtendedDesktopSizeRect(rdr::U16 reason, rdr::U16 result,
|
|
int fb_width, int fb_height,
|
|
const ScreenSet& layout);
|
|
void writeSetDesktopNameRect(const char *name);
|
|
void writeSetCursorRect(int width, int height,
|
|
int hotspotX, int hotspotY,
|
|
const void* data, const void* mask);
|
|
void writeSetXCursorRect(int width, int height,
|
|
int hotspotX, int hotspotY,
|
|
const void* data, const void* mask);
|
|
void writeSetCursorWithAlphaRect(int width, int height,
|
|
int hotspotX, int hotspotY,
|
|
const rdr::U8* data);
|
|
void writeSetVMwareCursorPositionRect(int hotspotX, int hotspotY);
|
|
void writeLEDStateRect(rdr::U8 state);
|
|
void writeQEMUKeyEventRect();
|
|
|
|
ConnParams* cp;
|
|
rdr::OutStream* os;
|
|
|
|
int nRectsInUpdate;
|
|
int nRectsInHeader;
|
|
|
|
bool needSetDesktopSize;
|
|
bool needExtendedDesktopSize;
|
|
bool needSetDesktopName;
|
|
bool needSetCursor;
|
|
bool needSetXCursor;
|
|
bool needSetCursorWithAlpha;
|
|
bool needCursorPos;
|
|
bool needLEDState;
|
|
bool needQEMUKeyEvent;
|
|
|
|
typedef struct {
|
|
rdr::U16 reason, result;
|
|
int fb_width, fb_height;
|
|
ScreenSet layout;
|
|
} ExtendedDesktopSizeMsg;
|
|
|
|
std::list<ExtendedDesktopSizeMsg> extendedDesktopSizeMsgs;
|
|
};
|
|
}
|
|
#endif
|