/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved. * * 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. */ #include #include #include #include #include #include #include #include using namespace rfb; CMsgWriter::CMsgWriter(ConnParams* cp_, rdr::OutStream* os_) : cp(cp_), os(os_) { } CMsgWriter::~CMsgWriter() { } void CMsgWriter::writeSetPixelFormat(const PixelFormat& pf) { startMsg(msgTypeSetPixelFormat); os->pad(3); pf.write(os); endMsg(); } void CMsgWriter::writeSetEncodings(int nEncodings, rdr::U32* encodings) { startMsg(msgTypeSetEncodings); os->skip(1); os->writeU16(nEncodings); for (int i = 0; i < nEncodings; i++) os->writeU32(encodings[i]); endMsg(); } // Ask for encodings based on which decoders are supported. Assumes higher // encoding numbers are more desirable. void CMsgWriter::writeSetEncodings(int preferredEncoding, bool useCopyRect) { int nEncodings = 0; rdr::U32 encodings[encodingMax+3]; if (cp->supportsLocalCursor) encodings[nEncodings++] = pseudoEncodingCursor; if (cp->supportsDesktopResize) encodings[nEncodings++] = pseudoEncodingDesktopSize; if (cp->supportsExtendedDesktopSize) encodings[nEncodings++] = pseudoEncodingExtendedDesktopSize; if (cp->supportsDesktopRename) encodings[nEncodings++] = pseudoEncodingDesktopName; if (Decoder::supported(preferredEncoding)) { encodings[nEncodings++] = preferredEncoding; } if (useCopyRect) { encodings[nEncodings++] = encodingCopyRect; } /* * Prefer encodings in this order: * * Tight, ZRLE, Hextile, * */ if ((preferredEncoding != encodingTight) && Decoder::supported(encodingTight)) encodings[nEncodings++] = encodingTight; if ((preferredEncoding != encodingZRLE) && Decoder::supported(encodingZRLE)) encodings[nEncodings++] = encodingZRLE; if ((preferredEncoding != encodingHextile) && Decoder::supported(encodingHextile)) encodings[nEncodings++] = encodingHextile; // Remaining encodings for (int i = encodingMax; i >= 0; i--) { switch (i) { case encodingTight: case encodingZRLE: case encodingHextile: break; default: if ((i != preferredEncoding) && Decoder::supported(i)) encodings[nEncodings++] = i; } } encodings[nEncodings++] = pseudoEncodingLastRect; if (cp->customCompressLevel && cp->compressLevel >= 0 && cp->compressLevel <= 9) encodings[nEncodings++] = pseudoEncodingCompressLevel0 + cp->compressLevel; if (!cp->noJpeg && cp->qualityLevel >= 0 && cp->qualityLevel <= 9) encodings[nEncodings++] = pseudoEncodingQualityLevel0 + cp->qualityLevel; writeSetEncodings(nEncodings, encodings); } void CMsgWriter::writeFramebufferUpdateRequest(const Rect& r, bool incremental) { startMsg(msgTypeFramebufferUpdateRequest); os->writeU8(incremental); os->writeU16(r.tl.x); os->writeU16(r.tl.y); os->writeU16(r.width()); os->writeU16(r.height()); endMsg(); } void CMsgWriter::keyEvent(rdr::U32 key, bool down) { startMsg(msgTypeKeyEvent); os->writeU8(down); os->pad(2); os->writeU32(key); endMsg(); } void CMsgWriter::pointerEvent(const Point& pos, int buttonMask) { Point p(pos); if (p.x < 0) p.x = 0; if (p.y < 0) p.y = 0; if (p.x >= cp->width) p.x = cp->width - 1; if (p.y >= cp->height) p.y = cp->height - 1; startMsg(msgTypePointerEvent); os->writeU8(buttonMask); os->writeU16(p.x); os->writeU16(p.y); endMsg(); } void CMsgWriter::clientCutText(const char* str, rdr::U32 len) { startMsg(msgTypeClientCutText); os->pad(3); os->writeU32(len); os->writeBytes(str, len); endMsg(); }