2014-10-18 02:23:07 +04:00
|
|
|
/**
|
|
|
|
* WinPR: Windows Portable Runtime
|
|
|
|
* Clipboard Functions
|
|
|
|
*
|
|
|
|
* Copyright 2014 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.
|
|
|
|
*/
|
|
|
|
|
2022-02-16 12:08:00 +03:00
|
|
|
#include <winpr/config.h>
|
2014-10-18 02:23:07 +04:00
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
#include <errno.h>
|
2014-10-18 02:23:07 +04:00
|
|
|
#include <winpr/crt.h>
|
2016-02-06 00:28:45 +03:00
|
|
|
#include <winpr/user.h>
|
2014-10-18 02:23:07 +04:00
|
|
|
|
|
|
|
#include "clipboard.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Standard Clipboard Formats:
|
|
|
|
* http://msdn.microsoft.com/en-us/library/windows/desktop/ff729168/
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* "CF_TEXT":
|
|
|
|
*
|
|
|
|
* Null-terminated ANSI text with CR/LF line endings.
|
|
|
|
*/
|
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
static void* clipboard_synthesize_cf_text(wClipboard* clipboard, UINT32 formatId, const void* data,
|
2019-11-06 17:24:51 +03:00
|
|
|
UINT32* pSize)
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
|
|
|
int size;
|
|
|
|
char* pDstData = NULL;
|
|
|
|
|
|
|
|
if (formatId == CF_UNICODETEXT)
|
|
|
|
{
|
2018-12-10 14:41:04 +03:00
|
|
|
size_t wsize;
|
2014-10-18 02:23:07 +04:00
|
|
|
char* str = NULL;
|
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
if (*pSize > INT32_MAX)
|
|
|
|
return NULL;
|
|
|
|
|
2018-12-10 14:41:04 +03:00
|
|
|
wsize = _wcsnlen(data, (*pSize) / 2);
|
2019-11-06 17:24:51 +03:00
|
|
|
size = ConvertFromUnicode(CP_UTF8, 0, (LPCWSTR)data, wsize, (CHAR**)&str, 0, NULL, NULL);
|
2014-10-18 02:23:07 +04:00
|
|
|
|
|
|
|
if (!str)
|
|
|
|
return NULL;
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
pDstData = ConvertLineEndingToCRLF((const char*)str, &size);
|
2014-10-18 02:23:07 +04:00
|
|
|
free(str);
|
|
|
|
*pSize = size;
|
|
|
|
return pDstData;
|
|
|
|
}
|
|
|
|
else if ((formatId == CF_TEXT) || (formatId == CF_OEMTEXT) ||
|
2017-11-14 18:10:52 +03:00
|
|
|
(formatId == ClipboardGetFormatId(clipboard, "UTF8_STRING")) ||
|
|
|
|
(formatId == ClipboardGetFormatId(clipboard, "text/plain")) ||
|
|
|
|
(formatId == ClipboardGetFormatId(clipboard, "TEXT")) ||
|
|
|
|
(formatId == ClipboardGetFormatId(clipboard, "STRING")))
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
size = (INT64)*pSize;
|
|
|
|
pDstData = ConvertLineEndingToCRLF((const char*)data, &size);
|
2014-10-18 02:23:07 +04:00
|
|
|
|
|
|
|
if (!pDstData)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
*pSize = size;
|
|
|
|
return pDstData;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* "CF_OEMTEXT":
|
|
|
|
*
|
|
|
|
* Null-terminated OEM text with CR/LF line endings.
|
|
|
|
*/
|
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
static void* clipboard_synthesize_cf_oemtext(wClipboard* clipboard, UINT32 formatId,
|
2019-11-06 17:24:51 +03:00
|
|
|
const void* data, UINT32* pSize)
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
|
|
|
return clipboard_synthesize_cf_text(clipboard, formatId, data, pSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* "CF_LOCALE":
|
|
|
|
*
|
|
|
|
* System locale identifier associated with CF_TEXT
|
|
|
|
*/
|
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
static void* clipboard_synthesize_cf_locale(wClipboard* clipboard, UINT32 formatId,
|
2019-11-06 17:24:51 +03:00
|
|
|
const void* data, UINT32* pSize)
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
|
|
|
UINT32* pDstData = NULL;
|
2019-11-06 17:24:51 +03:00
|
|
|
pDstData = (UINT32*)malloc(sizeof(UINT32));
|
2017-11-14 18:10:52 +03:00
|
|
|
|
2015-04-03 17:21:01 +03:00
|
|
|
if (!pDstData)
|
|
|
|
return NULL;
|
2014-10-18 02:23:07 +04:00
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
*pDstData = 0x0409; /* English - United States */
|
2019-11-06 17:24:51 +03:00
|
|
|
return (void*)pDstData;
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* "CF_UNICODETEXT":
|
|
|
|
*
|
|
|
|
* Null-terminated UTF-16 text with CR/LF line endings.
|
|
|
|
*/
|
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
static void* clipboard_synthesize_cf_unicodetext(wClipboard* clipboard, UINT32 formatId,
|
2019-11-06 17:24:51 +03:00
|
|
|
const void* data, UINT32* pSize)
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
|
|
|
int size;
|
|
|
|
int status;
|
|
|
|
char* crlfStr = NULL;
|
|
|
|
WCHAR* pDstData = NULL;
|
|
|
|
|
|
|
|
if ((formatId == CF_TEXT) || (formatId == CF_OEMTEXT) ||
|
2017-11-14 18:10:52 +03:00
|
|
|
(formatId == ClipboardGetFormatId(clipboard, "UTF8_STRING")) ||
|
|
|
|
(formatId == ClipboardGetFormatId(clipboard, "text/plain")) ||
|
|
|
|
(formatId == ClipboardGetFormatId(clipboard, "TEXT")) ||
|
|
|
|
(formatId == ClipboardGetFormatId(clipboard, "STRING")))
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
2017-11-14 18:10:52 +03:00
|
|
|
if (!pSize || (*pSize > INT32_MAX))
|
|
|
|
return NULL;
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
size = (int)*pSize;
|
2021-08-02 13:13:34 +03:00
|
|
|
crlfStr = ConvertLineEndingToCRLF((const char*)data, &size);
|
2014-10-18 02:23:07 +04:00
|
|
|
|
|
|
|
if (!crlfStr)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
status = ConvertToUnicode(CP_UTF8, 0, crlfStr, size, &pDstData, 0);
|
|
|
|
free(crlfStr);
|
|
|
|
|
|
|
|
if (status <= 0)
|
|
|
|
return NULL;
|
|
|
|
|
2014-12-04 21:19:10 +03:00
|
|
|
*pSize = status * 2;
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
return (void*)pDstData;
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* "UTF8_STRING":
|
|
|
|
*
|
|
|
|
* Null-terminated UTF-8 string with LF line endings.
|
|
|
|
*/
|
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
static void* clipboard_synthesize_utf8_string(wClipboard* clipboard, UINT32 formatId,
|
2019-11-06 17:24:51 +03:00
|
|
|
const void* data, UINT32* pSize)
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
2017-11-14 18:10:52 +03:00
|
|
|
INT64 size;
|
2014-10-18 02:23:07 +04:00
|
|
|
char* pDstData = NULL;
|
|
|
|
|
|
|
|
if (formatId == CF_UNICODETEXT)
|
|
|
|
{
|
2018-12-10 14:41:04 +03:00
|
|
|
size_t wsize = _wcsnlen(data, (*pSize) / 2);
|
2019-11-06 17:24:51 +03:00
|
|
|
size =
|
2021-08-02 13:13:34 +03:00
|
|
|
ConvertFromUnicode(CP_UTF8, 0, (LPCWSTR)data, wsize, (CHAR**)&pDstData, 0, NULL, NULL);
|
2014-10-18 02:23:07 +04:00
|
|
|
|
|
|
|
if (!pDstData)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
size = ConvertLineEndingToLF(pDstData, size);
|
2021-06-16 15:43:07 +03:00
|
|
|
if (size < 0)
|
|
|
|
{
|
|
|
|
free(pDstData);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
*pSize = (UINT32)size;
|
2014-10-18 02:23:07 +04:00
|
|
|
return pDstData;
|
|
|
|
}
|
|
|
|
else if ((formatId == CF_TEXT) || (formatId == CF_OEMTEXT) ||
|
2017-11-14 18:10:52 +03:00
|
|
|
(formatId == ClipboardGetFormatId(clipboard, "text/plain")) ||
|
|
|
|
(formatId == ClipboardGetFormatId(clipboard, "TEXT")) ||
|
|
|
|
(formatId == ClipboardGetFormatId(clipboard, "STRING")))
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
2021-06-16 15:43:07 +03:00
|
|
|
int rc;
|
2019-11-06 17:24:51 +03:00
|
|
|
size = (INT64)*pSize;
|
|
|
|
pDstData = (char*)malloc(size);
|
2014-10-18 02:23:07 +04:00
|
|
|
|
|
|
|
if (!pDstData)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
CopyMemory(pDstData, data, size);
|
2021-06-16 15:43:07 +03:00
|
|
|
rc = ConvertLineEndingToLF((char*)pDstData, (int)size);
|
|
|
|
if (rc < 0)
|
|
|
|
{
|
|
|
|
free(pDstData);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
*pSize = (UINT32)rc;
|
2014-10-18 02:23:07 +04:00
|
|
|
return pDstData;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* "CF_DIB":
|
|
|
|
*
|
|
|
|
* BITMAPINFO structure followed by the bitmap bits.
|
|
|
|
*/
|
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
static void* clipboard_synthesize_cf_dib(wClipboard* clipboard, UINT32 formatId, const void* data,
|
2019-11-06 17:24:51 +03:00
|
|
|
UINT32* pSize)
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
2014-10-18 03:14:54 +04:00
|
|
|
UINT32 SrcSize;
|
|
|
|
UINT32 DstSize;
|
|
|
|
BYTE* pDstData;
|
|
|
|
SrcSize = *pSize;
|
|
|
|
|
|
|
|
if (formatId == CF_DIBV5)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
else if (formatId == ClipboardGetFormatId(clipboard, "image/bmp"))
|
|
|
|
{
|
2021-08-26 12:23:32 +03:00
|
|
|
const BITMAPFILEHEADER* pFileHeader;
|
2014-10-18 03:14:54 +04:00
|
|
|
|
|
|
|
if (SrcSize < (sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER)))
|
|
|
|
return NULL;
|
|
|
|
|
2021-08-02 13:13:34 +03:00
|
|
|
pFileHeader = (const BITMAPFILEHEADER*)data;
|
2014-10-18 03:14:54 +04:00
|
|
|
|
|
|
|
if (pFileHeader->bfType != 0x4D42)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
DstSize = SrcSize - sizeof(BITMAPFILEHEADER);
|
2019-11-06 17:24:51 +03:00
|
|
|
pDstData = (BYTE*)malloc(DstSize);
|
2014-10-18 03:14:54 +04:00
|
|
|
|
|
|
|
if (!pDstData)
|
|
|
|
return NULL;
|
|
|
|
|
2021-08-02 13:13:34 +03:00
|
|
|
data = (const void*)&((const BYTE*)data)[sizeof(BITMAPFILEHEADER)];
|
2014-10-18 03:14:54 +04:00
|
|
|
CopyMemory(pDstData, data, DstSize);
|
|
|
|
*pSize = DstSize;
|
|
|
|
return pDstData;
|
|
|
|
}
|
2014-10-18 02:23:07 +04:00
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* "CF_DIBV5":
|
|
|
|
*
|
|
|
|
* BITMAPV5HEADER structure followed by the bitmap color space information and the bitmap bits.
|
|
|
|
*/
|
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
static void* clipboard_synthesize_cf_dibv5(wClipboard* clipboard, UINT32 formatId, const void* data,
|
2019-11-06 17:24:51 +03:00
|
|
|
UINT32* pSize)
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
2014-10-18 03:14:54 +04:00
|
|
|
if (formatId == CF_DIB)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
else if (formatId == ClipboardGetFormatId(clipboard, "image/bmp"))
|
|
|
|
{
|
|
|
|
}
|
2014-10-18 02:23:07 +04:00
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* "image/bmp":
|
|
|
|
*
|
|
|
|
* Bitmap file format.
|
|
|
|
*/
|
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
static void* clipboard_synthesize_image_bmp(wClipboard* clipboard, UINT32 formatId,
|
2019-11-06 17:24:51 +03:00
|
|
|
const void* data, UINT32* pSize)
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
2014-10-18 03:14:54 +04:00
|
|
|
UINT32 SrcSize;
|
|
|
|
UINT32 DstSize;
|
|
|
|
BYTE* pDstData;
|
|
|
|
SrcSize = *pSize;
|
|
|
|
|
|
|
|
if (formatId == CF_DIB)
|
|
|
|
{
|
|
|
|
BYTE* pDst;
|
2021-08-26 12:23:32 +03:00
|
|
|
const BITMAPINFOHEADER* pInfoHeader;
|
2014-10-18 03:14:54 +04:00
|
|
|
BITMAPFILEHEADER* pFileHeader;
|
|
|
|
|
2014-10-18 04:40:11 +04:00
|
|
|
if (SrcSize < sizeof(BITMAPINFOHEADER))
|
|
|
|
return NULL;
|
|
|
|
|
2021-08-02 13:13:34 +03:00
|
|
|
pInfoHeader = (const BITMAPINFOHEADER*)data;
|
2014-10-18 03:14:54 +04:00
|
|
|
|
|
|
|
if ((pInfoHeader->biBitCount < 1) || (pInfoHeader->biBitCount > 32))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
DstSize = sizeof(BITMAPFILEHEADER) + SrcSize;
|
2019-11-06 17:24:51 +03:00
|
|
|
pDstData = (BYTE*)malloc(DstSize);
|
2014-10-18 03:14:54 +04:00
|
|
|
|
|
|
|
if (!pDstData)
|
|
|
|
return NULL;
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
pFileHeader = (BITMAPFILEHEADER*)pDstData;
|
2014-10-18 03:14:54 +04:00
|
|
|
pFileHeader->bfType = 0x4D42;
|
|
|
|
pFileHeader->bfSize = DstSize;
|
|
|
|
pFileHeader->bfReserved1 = 0;
|
|
|
|
pFileHeader->bfReserved2 = 0;
|
|
|
|
pFileHeader->bfOffBits = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER);
|
|
|
|
pDst = &pDstData[sizeof(BITMAPFILEHEADER)];
|
|
|
|
CopyMemory(pDst, data, SrcSize);
|
|
|
|
*pSize = DstSize;
|
|
|
|
return pDstData;
|
|
|
|
}
|
|
|
|
else if (formatId == CF_DIBV5)
|
|
|
|
{
|
|
|
|
}
|
2014-10-18 02:23:07 +04:00
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* "HTML Format":
|
|
|
|
*
|
|
|
|
* HTML clipboard format: msdn.microsoft.com/en-us/library/windows/desktop/ms649015/
|
|
|
|
*/
|
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
static void* clipboard_synthesize_html_format(wClipboard* clipboard, UINT32 formatId,
|
2019-11-06 17:24:51 +03:00
|
|
|
const void* data, UINT32* pSize)
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
2014-10-18 03:14:54 +04:00
|
|
|
char* pSrcData = NULL;
|
|
|
|
char* pDstData = NULL;
|
2019-11-06 17:24:51 +03:00
|
|
|
INT64 SrcSize = (INT64)*pSize;
|
2014-10-18 02:23:07 +04:00
|
|
|
|
2014-10-18 03:14:54 +04:00
|
|
|
if (formatId == ClipboardGetFormatId(clipboard, "text/html"))
|
|
|
|
{
|
|
|
|
char* body;
|
|
|
|
BYTE bom[2];
|
2018-08-24 10:54:25 +03:00
|
|
|
char num[20];
|
2021-08-02 13:13:34 +03:00
|
|
|
const WCHAR* wstr;
|
2014-10-18 03:14:54 +04:00
|
|
|
|
|
|
|
if (SrcSize > 2)
|
|
|
|
{
|
2021-06-16 15:43:07 +03:00
|
|
|
if (SrcSize > INT_MAX)
|
|
|
|
return NULL;
|
2014-10-18 03:14:54 +04:00
|
|
|
CopyMemory(bom, data, 2);
|
|
|
|
|
|
|
|
if ((bom[0] == 0xFE) && (bom[1] == 0xFF))
|
|
|
|
{
|
2021-08-26 12:23:32 +03:00
|
|
|
ByteSwapUnicode((WCHAR*)data, (int)(SrcSize / 2));
|
2014-10-18 03:14:54 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if ((bom[0] == 0xFF) && (bom[1] == 0xFE))
|
|
|
|
{
|
2021-08-02 13:13:34 +03:00
|
|
|
wstr = (const WCHAR*)&((BYTE*)data)[2];
|
2021-06-16 15:43:07 +03:00
|
|
|
ConvertFromUnicode(CP_UTF8, 0, wstr, (int)(SrcSize - 2) / 2, &pSrcData, 0, NULL,
|
|
|
|
NULL);
|
2014-10-18 03:14:54 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!pSrcData)
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
pSrcData = (char*)calloc(1, SrcSize + 1);
|
2017-11-14 18:10:52 +03:00
|
|
|
|
2015-04-03 17:21:01 +03:00
|
|
|
if (!pSrcData)
|
|
|
|
return NULL;
|
2017-11-14 18:10:52 +03:00
|
|
|
|
2014-10-18 03:14:54 +04:00
|
|
|
CopyMemory(pSrcData, data, SrcSize);
|
|
|
|
}
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
pDstData = (char*)calloc(1, SrcSize + 200);
|
2017-11-14 18:10:52 +03:00
|
|
|
|
2015-04-03 17:21:01 +03:00
|
|
|
if (!pDstData)
|
|
|
|
{
|
|
|
|
free(pSrcData);
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-10-18 03:14:54 +04:00
|
|
|
|
2018-08-24 10:54:25 +03:00
|
|
|
sprintf_s(pDstData, SrcSize + 200,
|
|
|
|
"Version:0.9\r\n"
|
|
|
|
"StartHTML:0000000000\r\n"
|
|
|
|
"EndHTML:0000000000\r\n"
|
|
|
|
"StartFragment:0000000000\r\n"
|
|
|
|
"EndFragment:0000000000\r\n");
|
2014-10-18 03:14:54 +04:00
|
|
|
body = strstr(pSrcData, "<body");
|
|
|
|
|
|
|
|
if (!body)
|
|
|
|
body = strstr(pSrcData, "<BODY");
|
|
|
|
|
|
|
|
/* StartHTML */
|
2019-11-06 17:24:51 +03:00
|
|
|
sprintf_s(num, sizeof(num), "%010" PRIuz "", strnlen(pDstData, SrcSize + 200));
|
2014-10-18 03:14:54 +04:00
|
|
|
CopyMemory(&pDstData[23], num, 10);
|
|
|
|
|
|
|
|
if (!body)
|
|
|
|
strcat(pDstData, "<HTML><BODY>");
|
|
|
|
|
|
|
|
strcat(pDstData, "<!--StartFragment-->");
|
|
|
|
/* StartFragment */
|
2019-11-06 17:24:51 +03:00
|
|
|
sprintf_s(num, sizeof(num), "%010" PRIuz "", strnlen(pDstData, SrcSize + 200));
|
2014-10-18 03:14:54 +04:00
|
|
|
CopyMemory(&pDstData[69], num, 10);
|
|
|
|
strcat(pDstData, pSrcData);
|
|
|
|
/* EndFragment */
|
2019-11-06 17:24:51 +03:00
|
|
|
sprintf_s(num, sizeof(num), "%010" PRIuz "", strnlen(pDstData, SrcSize + 200));
|
2014-10-18 03:14:54 +04:00
|
|
|
CopyMemory(&pDstData[93], num, 10);
|
|
|
|
strcat(pDstData, "<!--EndFragment-->");
|
|
|
|
|
|
|
|
if (!body)
|
|
|
|
strcat(pDstData, "</BODY></HTML>");
|
|
|
|
|
|
|
|
/* EndHTML */
|
2019-11-06 17:24:51 +03:00
|
|
|
sprintf_s(num, sizeof(num), "%010" PRIuz "", strnlen(pDstData, SrcSize + 200));
|
2014-10-18 03:14:54 +04:00
|
|
|
CopyMemory(&pDstData[43], num, 10);
|
2019-11-06 17:24:51 +03:00
|
|
|
*pSize = (UINT32)strlen(pDstData) + 1;
|
2014-10-18 03:14:54 +04:00
|
|
|
free(pSrcData);
|
|
|
|
}
|
|
|
|
|
|
|
|
return pDstData;
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* "text/html":
|
|
|
|
*
|
|
|
|
* HTML text format.
|
|
|
|
*/
|
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
static void* clipboard_synthesize_text_html(wClipboard* clipboard, UINT32 formatId,
|
2019-11-06 17:24:51 +03:00
|
|
|
const void* data, UINT32* pSize)
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
2017-11-14 18:10:52 +03:00
|
|
|
long beg;
|
|
|
|
long end;
|
2021-08-02 13:13:34 +03:00
|
|
|
const char* str;
|
2014-10-18 03:14:54 +04:00
|
|
|
char* begStr;
|
|
|
|
char* endStr;
|
2017-11-14 18:10:52 +03:00
|
|
|
long DstSize = -1;
|
2014-10-18 03:14:54 +04:00
|
|
|
BYTE* pDstData = NULL;
|
|
|
|
|
|
|
|
if (formatId == ClipboardGetFormatId(clipboard, "HTML Format"))
|
|
|
|
{
|
2021-06-16 15:43:07 +03:00
|
|
|
INT64 SrcSize;
|
2021-08-02 13:13:34 +03:00
|
|
|
str = (const char*)data;
|
2019-11-06 17:24:51 +03:00
|
|
|
SrcSize = (INT64)*pSize;
|
2014-10-18 03:14:54 +04:00
|
|
|
begStr = strstr(str, "StartHTML:");
|
|
|
|
endStr = strstr(str, "EndHTML:");
|
|
|
|
|
|
|
|
if (!begStr || !endStr)
|
|
|
|
return NULL;
|
|
|
|
|
2017-11-14 18:10:52 +03:00
|
|
|
errno = 0;
|
2018-01-07 05:45:22 +03:00
|
|
|
beg = strtol(&begStr[10], NULL, 10);
|
2017-11-14 18:10:52 +03:00
|
|
|
|
|
|
|
if (errno != 0)
|
|
|
|
return NULL;
|
|
|
|
|
2018-01-07 05:45:22 +03:00
|
|
|
end = strtol(&endStr[8], NULL, 10);
|
2014-10-18 03:14:54 +04:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
if (beg < 0 || end < 0 || (beg > SrcSize) || (end > SrcSize) || (beg >= end) ||
|
|
|
|
(errno != 0))
|
2014-10-18 03:14:54 +04:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
DstSize = end - beg;
|
2021-06-16 15:43:07 +03:00
|
|
|
pDstData = (BYTE*)malloc((size_t)(SrcSize - beg + 1));
|
2014-10-18 03:14:54 +04:00
|
|
|
|
|
|
|
if (!pDstData)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
CopyMemory(pDstData, &str[beg], DstSize);
|
2019-11-06 17:24:51 +03:00
|
|
|
DstSize = ConvertLineEndingToLF((char*)pDstData, DstSize);
|
|
|
|
*pSize = (UINT32)DstSize;
|
2014-10-18 03:14:54 +04:00
|
|
|
}
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
return (void*)pDstData;
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
BOOL ClipboardInitSynthesizers(wClipboard* clipboard)
|
|
|
|
{
|
|
|
|
UINT32 formatId;
|
|
|
|
UINT32 altFormatId;
|
|
|
|
/**
|
|
|
|
* CF_TEXT
|
|
|
|
*/
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_TEXT, CF_OEMTEXT, clipboard_synthesize_cf_oemtext);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_TEXT, CF_UNICODETEXT,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_unicodetext);
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_TEXT, CF_LOCALE, clipboard_synthesize_cf_locale);
|
2014-10-18 02:23:07 +04:00
|
|
|
altFormatId = ClipboardRegisterFormat(clipboard, "UTF8_STRING");
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_TEXT, altFormatId, clipboard_synthesize_utf8_string);
|
2014-10-18 02:23:07 +04:00
|
|
|
/**
|
|
|
|
* CF_OEMTEXT
|
|
|
|
*/
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_OEMTEXT, CF_TEXT, clipboard_synthesize_cf_text);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_OEMTEXT, CF_UNICODETEXT,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_unicodetext);
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_OEMTEXT, CF_LOCALE, clipboard_synthesize_cf_locale);
|
2014-10-18 02:23:07 +04:00
|
|
|
altFormatId = ClipboardRegisterFormat(clipboard, "UTF8_STRING");
|
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_OEMTEXT, altFormatId,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_utf8_string);
|
2014-10-18 02:23:07 +04:00
|
|
|
/**
|
|
|
|
* CF_UNICODETEXT
|
|
|
|
*/
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_UNICODETEXT, CF_TEXT, clipboard_synthesize_cf_text);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_UNICODETEXT, CF_OEMTEXT,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_oemtext);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_UNICODETEXT, CF_LOCALE,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_locale);
|
2014-10-18 02:23:07 +04:00
|
|
|
altFormatId = ClipboardRegisterFormat(clipboard, "UTF8_STRING");
|
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_UNICODETEXT, altFormatId,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_utf8_string);
|
2014-10-18 02:23:07 +04:00
|
|
|
/**
|
|
|
|
* UTF8_STRING
|
|
|
|
*/
|
|
|
|
formatId = ClipboardRegisterFormat(clipboard, "UTF8_STRING");
|
|
|
|
|
|
|
|
if (formatId)
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_TEXT, clipboard_synthesize_cf_text);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_OEMTEXT,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_oemtext);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_UNICODETEXT,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_unicodetext);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_LOCALE,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_locale);
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* text/plain
|
|
|
|
*/
|
|
|
|
formatId = ClipboardRegisterFormat(clipboard, "text/plain");
|
|
|
|
|
|
|
|
if (formatId)
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_TEXT, clipboard_synthesize_cf_text);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_OEMTEXT,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_oemtext);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_UNICODETEXT,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_unicodetext);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_LOCALE,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_locale);
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* TEXT
|
|
|
|
*/
|
|
|
|
formatId = ClipboardRegisterFormat(clipboard, "TEXT");
|
|
|
|
|
|
|
|
if (formatId)
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_TEXT, clipboard_synthesize_cf_text);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_OEMTEXT,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_oemtext);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_UNICODETEXT,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_unicodetext);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_LOCALE,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_locale);
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* STRING
|
|
|
|
*/
|
|
|
|
formatId = ClipboardRegisterFormat(clipboard, "STRING");
|
|
|
|
|
|
|
|
if (formatId)
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_TEXT, clipboard_synthesize_cf_text);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_OEMTEXT,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_oemtext);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_UNICODETEXT,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_unicodetext);
|
2014-10-18 02:23:07 +04:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_LOCALE,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_cf_locale);
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* CF_DIB
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (formatId)
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_DIB, CF_DIBV5, clipboard_synthesize_cf_dibv5);
|
2014-10-18 02:23:07 +04:00
|
|
|
altFormatId = ClipboardRegisterFormat(clipboard, "image/bmp");
|
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_DIB, altFormatId,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_image_bmp);
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* CF_DIBV5
|
|
|
|
*/
|
|
|
|
|
2014-10-18 03:14:54 +04:00
|
|
|
if (formatId && 0)
|
2014-10-18 02:23:07 +04:00
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_DIBV5, CF_DIB, clipboard_synthesize_cf_dib);
|
2014-10-18 02:23:07 +04:00
|
|
|
altFormatId = ClipboardRegisterFormat(clipboard, "image/bmp");
|
|
|
|
ClipboardRegisterSynthesizer(clipboard, CF_DIBV5, altFormatId,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_image_bmp);
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* image/bmp
|
|
|
|
*/
|
|
|
|
formatId = ClipboardRegisterFormat(clipboard, "image/bmp");
|
|
|
|
|
|
|
|
if (formatId)
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_DIB, clipboard_synthesize_cf_dib);
|
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, CF_DIBV5, clipboard_synthesize_cf_dibv5);
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* HTML Format
|
|
|
|
*/
|
|
|
|
formatId = ClipboardRegisterFormat(clipboard, "HTML Format");
|
|
|
|
|
|
|
|
if (formatId)
|
|
|
|
{
|
|
|
|
altFormatId = ClipboardRegisterFormat(clipboard, "text/html");
|
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, altFormatId,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_text_html);
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* text/html
|
|
|
|
*/
|
|
|
|
formatId = ClipboardRegisterFormat(clipboard, "text/html");
|
|
|
|
|
|
|
|
if (formatId)
|
|
|
|
{
|
|
|
|
altFormatId = ClipboardRegisterFormat(clipboard, "HTML Format");
|
|
|
|
ClipboardRegisterSynthesizer(clipboard, formatId, altFormatId,
|
2017-11-14 18:10:52 +03:00
|
|
|
clipboard_synthesize_html_format);
|
2014-10-18 02:23:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|