зеркало из https://github.com/mozilla/pjs.git
449 строки
14 KiB
C++
449 строки
14 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-*
|
|
*
|
|
* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (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.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is the Mozilla OS/2 libraries.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* John Fairhurst, <john_fairhurst@iname.com>.
|
|
* Portions created by the Initial Developer are Copyright (C) 1999
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* 2000/08/04 Henry Sobotka <sobotka@axess.com> Update from M7
|
|
* 2000/10/02 IBM Corp. Sync-up to M18 level
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
#include "nsXPCOM.h"
|
|
#include "nsISupportsPrimitives.h"
|
|
#include "nsCOMPtr.h"
|
|
#include "nsPrimitiveHelpers.h"
|
|
#include "nsXPIDLString.h"
|
|
#include "prmem.h"
|
|
#include "nsOS2Uni.h"
|
|
#include "nsClipboard.h"
|
|
|
|
#define INCL_DOSERRORS
|
|
#define INCL_WIN
|
|
#include <os2.h>
|
|
|
|
inline PRUint32 RegisterClipboardFormat(PCSZ pcszFormat)
|
|
{
|
|
ATOM atom = WinFindAtom(WinQuerySystemAtomTable(), pcszFormat);
|
|
if (!atom) {
|
|
atom = WinAddAtom(WinQuerySystemAtomTable(), pcszFormat);
|
|
}
|
|
return atom;
|
|
}
|
|
|
|
nsClipboard::nsClipboard() : nsBaseClipboard()
|
|
{
|
|
RegisterClipboardFormat(kTextMime);
|
|
RegisterClipboardFormat(kUnicodeMime);
|
|
RegisterClipboardFormat(kHTMLMime);
|
|
RegisterClipboardFormat(kAOLMailMime);
|
|
RegisterClipboardFormat(kPNGImageMime);
|
|
RegisterClipboardFormat(kJPEGImageMime);
|
|
RegisterClipboardFormat(kGIFImageMime);
|
|
RegisterClipboardFormat(kFileMime);
|
|
RegisterClipboardFormat(kURLMime);
|
|
RegisterClipboardFormat(kNativeImageMime);
|
|
RegisterClipboardFormat(kNativeHTMLMime);
|
|
}
|
|
|
|
nsClipboard::~nsClipboard()
|
|
{}
|
|
|
|
nsresult nsClipboard::SetNativeClipboardData(PRInt32 aWhichClipboard)
|
|
{
|
|
if (aWhichClipboard != kGlobalClipboard)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
return DoClipboardAction(Write);
|
|
}
|
|
|
|
nsresult nsClipboard::GetNativeClipboardData(nsITransferable *aTransferable, PRInt32 aWhichClipboard)
|
|
{
|
|
// make sure we have a good transferable
|
|
if (!aTransferable || aWhichClipboard != kGlobalClipboard)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
nsITransferable *tmp = mTransferable;
|
|
mTransferable = aTransferable;
|
|
nsresult rc = DoClipboardAction(Read);
|
|
mTransferable = tmp;
|
|
return rc;
|
|
}
|
|
|
|
// Get some data from the clipboard
|
|
bool nsClipboard::GetClipboardData(const char *aFlavor)
|
|
{
|
|
PRUint32 ulFormatID = GetFormatID(aFlavor);
|
|
|
|
bool found = GetClipboardDataByID( ulFormatID, aFlavor );
|
|
|
|
if (!found)
|
|
{
|
|
if (!strcmp( aFlavor, kUnicodeMime ))
|
|
{
|
|
found = GetClipboardDataByID( CF_TEXT, aFlavor );
|
|
}
|
|
else if (strstr( aFlavor, "image/" ))
|
|
{
|
|
found = GetClipboardDataByID( CF_BITMAP, aFlavor );
|
|
}
|
|
}
|
|
|
|
return found;
|
|
}
|
|
|
|
bool nsClipboard::GetClipboardDataByID(PRUint32 aFormatID, const char *aFlavor)
|
|
{
|
|
PVOID pDataMem;
|
|
PRUint32 NumOfBytes;
|
|
bool TempBufAllocated = false;
|
|
|
|
PVOID pClipboardData = reinterpret_cast<PVOID>(WinQueryClipbrdData(0, aFormatID));
|
|
|
|
if (!pClipboardData)
|
|
return false;
|
|
|
|
if (strstr( aFlavor, "text/" )) // All text/.. flavors are null-terminated
|
|
{
|
|
pDataMem = pClipboardData;
|
|
|
|
if (aFormatID == CF_TEXT) // CF_TEXT is one byte character set
|
|
{
|
|
PRUint32 NumOfChars = strlen( static_cast<char*>(pDataMem) );
|
|
NumOfBytes = NumOfChars;
|
|
|
|
if (!strcmp( aFlavor, kUnicodeMime )) // Asked for unicode, but only plain text available. Convert it!
|
|
{
|
|
nsAutoChar16Buffer buffer;
|
|
PRInt32 bufLength;
|
|
MultiByteToWideChar(0, static_cast<char*>(pDataMem), NumOfChars,
|
|
buffer, bufLength);
|
|
pDataMem = ToNewUnicode(nsDependentString(buffer.Elements()));
|
|
TempBufAllocated = true;
|
|
NumOfBytes = bufLength * sizeof(UniChar);
|
|
}
|
|
|
|
}
|
|
else // All other text/.. flavors are in unicode
|
|
{
|
|
PRUint32 NumOfChars = UniStrlen( static_cast<UniChar*>(pDataMem) );
|
|
NumOfBytes = NumOfChars * sizeof(UniChar);
|
|
PVOID pTempBuf = nsMemory::Alloc(NumOfBytes);
|
|
memcpy(pTempBuf, pDataMem, NumOfBytes);
|
|
pDataMem = pTempBuf;
|
|
TempBufAllocated = true;
|
|
}
|
|
|
|
// DOM wants LF only, so convert from CRLF
|
|
nsLinebreakHelpers::ConvertPlatformToDOMLinebreaks( aFlavor, &pDataMem, // pDataMem could be reallocated !!
|
|
reinterpret_cast<PRInt32*>(&NumOfBytes) ); // yuck
|
|
|
|
}
|
|
else // Assume rest of flavors are binary data
|
|
{
|
|
if (aFormatID == CF_BITMAP)
|
|
{
|
|
if (!strcmp( aFlavor, kJPEGImageMime ))
|
|
{
|
|
// OS2TODO Convert bitmap to jpg
|
|
#ifdef DEBUG
|
|
printf( "nsClipboard:: No JPG found on clipboard; need to convert BMP\n");
|
|
#endif
|
|
}
|
|
else if (!strcmp( aFlavor, kGIFImageMime ))
|
|
{
|
|
// OS2TODO Convert bitmap to gif
|
|
#ifdef DEBUG
|
|
printf( "nsClipboard:: No GIF found on clipboard; need to convert BMP\n");
|
|
#endif
|
|
}
|
|
else if (!strcmp( aFlavor, kPNGImageMime ))
|
|
{
|
|
// OS2TODO Convert bitmap to png
|
|
#ifdef DEBUG
|
|
printf( "nsClipboard:: No PNG found on clipboard; need to convert BMP\n");
|
|
#endif
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pDataMem = static_cast<PBYTE>(pClipboardData) + sizeof(PRUint32);
|
|
NumOfBytes = *(static_cast<PRUint32*>(pClipboardData));
|
|
}
|
|
}
|
|
|
|
nsCOMPtr<nsISupports> genericDataWrapper;
|
|
nsPrimitiveHelpers::CreatePrimitiveForData( aFlavor, pDataMem, NumOfBytes, getter_AddRefs(genericDataWrapper) );
|
|
#ifdef DEBUG
|
|
nsresult errCode =
|
|
#endif
|
|
mTransferable->SetTransferData( aFlavor, genericDataWrapper, NumOfBytes );
|
|
#ifdef DEBUG
|
|
if (errCode != NS_OK)
|
|
printf( "nsClipboard:: Error setting data into transferable\n" );
|
|
#endif
|
|
|
|
if (TempBufAllocated)
|
|
nsMemory::Free(pDataMem);
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
// Set some data onto the clipboard
|
|
void nsClipboard::SetClipboardData(const char *aFlavor)
|
|
{
|
|
void *pMozData = nsnull;
|
|
PRUint32 NumOfBytes = 0;
|
|
|
|
// Get the data from the transferable
|
|
nsCOMPtr<nsISupports> genericDataWrapper;
|
|
#ifdef DEBUG
|
|
nsresult errCode =
|
|
#endif
|
|
mTransferable->GetTransferData( aFlavor, getter_AddRefs(genericDataWrapper), &NumOfBytes );
|
|
#ifdef DEBUG
|
|
if (NS_FAILED(errCode)) printf( "nsClipboard:: Error getting data from transferable\n" );
|
|
#endif
|
|
if (NumOfBytes == 0) return;
|
|
nsPrimitiveHelpers::CreateDataFromPrimitive( aFlavor, genericDataWrapper, &pMozData, NumOfBytes );
|
|
|
|
/* If creating the data failed, just return */
|
|
if (!pMozData) {
|
|
return;
|
|
}
|
|
|
|
PRUint32 ulFormatID = GetFormatID(aFlavor);
|
|
|
|
if (strstr( aFlavor, "text/" )) // All text/.. flavors are null-terminated
|
|
{
|
|
if (ulFormatID == CF_TEXT) // CF_TEXT is one byte character set
|
|
{
|
|
char* pByteMem = nsnull;
|
|
|
|
if (DosAllocSharedMem( reinterpret_cast<PPVOID>(&pByteMem), nsnull, NumOfBytes + sizeof(char),
|
|
PAG_WRITE | PAG_COMMIT | OBJ_GIVEABLE ) == NO_ERROR)
|
|
{
|
|
memcpy( pByteMem, pMozData, NumOfBytes ); // Copy text string
|
|
pByteMem[NumOfBytes] = '\0'; // Append terminator
|
|
|
|
// With Warp4 copying more than 64K to the clipboard works well, but
|
|
// legacy apps cannot always handle it. So output an alarm to alert the
|
|
// user that there might be a problem.
|
|
if (strlen(pByteMem) > 0xFFFF) {
|
|
WinAlarm(HWND_DESKTOP, WA_ERROR);
|
|
}
|
|
WinSetClipbrdData(0, reinterpret_cast<ULONG>(pByteMem), ulFormatID, CFI_POINTER);
|
|
}
|
|
}
|
|
else // All other text/.. flavors are in unicode
|
|
{
|
|
UniChar* pUnicodeMem = nsnull;
|
|
PRUint32 NumOfChars = NumOfBytes / sizeof(UniChar);
|
|
|
|
if (DosAllocSharedMem( reinterpret_cast<PPVOID>(&pUnicodeMem), nsnull, NumOfBytes + sizeof(UniChar),
|
|
PAG_WRITE | PAG_COMMIT | OBJ_GIVEABLE ) == NO_ERROR)
|
|
{
|
|
memcpy( pUnicodeMem, pMozData, NumOfBytes ); // Copy text string
|
|
pUnicodeMem[NumOfChars] = L'\0'; // Append terminator
|
|
|
|
WinSetClipbrdData( 0, reinterpret_cast<ULONG>(pUnicodeMem), ulFormatID, CFI_POINTER );
|
|
}
|
|
|
|
// If the flavor is unicode, we also put it on the clipboard as CF_TEXT
|
|
// after conversion to locale charset.
|
|
|
|
if (!strcmp( aFlavor, kUnicodeMime ))
|
|
{
|
|
char* pByteMem = nsnull;
|
|
|
|
if (DosAllocSharedMem(reinterpret_cast<PPVOID>(&pByteMem), nsnull,
|
|
NumOfBytes + 1,
|
|
PAG_WRITE | PAG_COMMIT | OBJ_GIVEABLE ) == NO_ERROR)
|
|
{
|
|
PRUnichar* uchtemp = (PRUnichar*)pMozData;
|
|
for (PRUint32 i=0;i<NumOfChars;i++) {
|
|
switch (uchtemp[i]) {
|
|
case 0x2018:
|
|
case 0x2019:
|
|
uchtemp[i] = 0x0027;
|
|
break;
|
|
case 0x201C:
|
|
case 0x201D:
|
|
uchtemp[i] = 0x0022;
|
|
break;
|
|
case 0x2014:
|
|
uchtemp[i] = 0x002D;
|
|
break;
|
|
}
|
|
}
|
|
|
|
nsAutoCharBuffer buffer;
|
|
PRInt32 bufLength;
|
|
WideCharToMultiByte(0, static_cast<PRUnichar*>(pMozData),
|
|
NumOfBytes, buffer, bufLength);
|
|
memcpy(pByteMem, buffer.Elements(), NumOfBytes);
|
|
// With Warp4 copying more than 64K to the clipboard works well, but
|
|
// legacy apps cannot always handle it. So output an alarm to alert the
|
|
// user that there might be a problem.
|
|
if (strlen(pByteMem) > 0xFFFF) {
|
|
WinAlarm(HWND_DESKTOP, WA_ERROR);
|
|
}
|
|
WinSetClipbrdData(0, reinterpret_cast<ULONG>(pByteMem), CF_TEXT, CFI_POINTER);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else // Assume rest of flavors are binary data
|
|
{
|
|
PBYTE pBinaryMem = nsnull;
|
|
|
|
if (DosAllocSharedMem( reinterpret_cast<PPVOID>(&pBinaryMem), nsnull, NumOfBytes + sizeof(PRUint32),
|
|
PAG_WRITE | PAG_COMMIT | OBJ_GIVEABLE ) == NO_ERROR)
|
|
{
|
|
*(reinterpret_cast<PRUint32*>(pBinaryMem)) = NumOfBytes; // First DWORD contains data length
|
|
memcpy( pBinaryMem + sizeof(PRUint32), pMozData, NumOfBytes ); // Copy binary data
|
|
|
|
WinSetClipbrdData( 0, reinterpret_cast<ULONG>(pBinaryMem), ulFormatID, CFI_POINTER );
|
|
}
|
|
|
|
// If the flavor is image, we also put it on clipboard as CF_BITMAP
|
|
// after conversion to OS2 bitmap
|
|
|
|
if (strstr (aFlavor, "image/"))
|
|
{
|
|
// XXX OS2TODO Convert jpg, gif, png to bitmap
|
|
#ifdef DEBUG
|
|
printf( "nsClipboard:: Putting image on clipboard; should also convert to BMP\n" );
|
|
#endif
|
|
}
|
|
}
|
|
nsMemory::Free(pMozData);
|
|
}
|
|
|
|
// Go through the flavors in the transferable and either get or set them
|
|
nsresult nsClipboard::DoClipboardAction(ClipboardAction aAction)
|
|
{
|
|
nsresult rc = NS_ERROR_FAILURE;
|
|
|
|
if (WinOpenClipbrd(0/*hab*/)) {
|
|
|
|
if (aAction == Write)
|
|
WinEmptyClipbrd(0/*hab*/);
|
|
|
|
// Get the list of formats the transferable can handle
|
|
nsCOMPtr<nsISupportsArray> pFormats;
|
|
if(aAction == Read)
|
|
rc = mTransferable->FlavorsTransferableCanImport(getter_AddRefs(pFormats));
|
|
else
|
|
rc = mTransferable->FlavorsTransferableCanExport(getter_AddRefs(pFormats));
|
|
|
|
if (NS_FAILED(rc))
|
|
return NS_ERROR_FAILURE;
|
|
|
|
PRUint32 cFormats = 0;
|
|
pFormats->Count(&cFormats);
|
|
|
|
for (PRUint32 i = 0; i < cFormats; i++) {
|
|
|
|
nsCOMPtr<nsISupports> genericFlavor;
|
|
pFormats->GetElementAt(i, getter_AddRefs(genericFlavor));
|
|
nsCOMPtr<nsISupportsCString> currentFlavor(do_QueryInterface(genericFlavor));
|
|
|
|
if (currentFlavor) {
|
|
nsXPIDLCString flavorStr;
|
|
currentFlavor->ToString(getter_Copies(flavorStr));
|
|
|
|
if (aAction == Read) {
|
|
if (GetClipboardData(flavorStr))
|
|
break;
|
|
}
|
|
else
|
|
SetClipboardData(flavorStr);
|
|
}
|
|
}
|
|
WinCloseClipbrd(0/*hab*/);
|
|
rc = NS_OK;
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
// get the format ID for a given mimetype
|
|
PRUint32 nsClipboard::GetFormatID(const char *aMimeStr)
|
|
{
|
|
if (strcmp(aMimeStr, kTextMime) == 0)
|
|
return CF_TEXT;
|
|
|
|
return RegisterClipboardFormat(aMimeStr);
|
|
}
|
|
|
|
NS_IMETHODIMP nsClipboard::HasDataMatchingFlavors(const char** aFlavorList,
|
|
PRUint32 aLength,
|
|
PRInt32 aWhichClipboard,
|
|
bool *_retval)
|
|
{
|
|
*_retval = false;
|
|
if (aWhichClipboard != kGlobalClipboard || !aFlavorList)
|
|
return NS_OK;
|
|
|
|
for (PRUint32 i = 0; i < aLength; ++i) {
|
|
ULONG fmtInfo = 0;
|
|
PRUint32 format = GetFormatID(aFlavorList[i]);
|
|
|
|
if (WinQueryClipbrdFmtInfo(0/*hab*/, format, &fmtInfo)) {
|
|
*_retval = true;
|
|
break;
|
|
}
|
|
|
|
// if the client asked for unicode and it wasn't present, check if we have CF_TEXT.
|
|
if (!strcmp(aFlavorList[i], kUnicodeMime)) {
|
|
if (WinQueryClipbrdFmtInfo(0/*hab*/, CF_TEXT, &fmtInfo)) {
|
|
*_retval = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// OS2TODO - Support for Images
|
|
// if the client asked for image/.. and it wasn't present, check if we have CF_BITMAP.
|
|
if (strstr(aFlavorList[i], "image/")) {
|
|
if (WinQueryClipbrdFmtInfo (0, CF_BITMAP, &fmtInfo)) {
|
|
#ifdef DEBUG
|
|
printf("nsClipboard:: Image present on clipboard; need to add BMP conversion!\n");
|
|
#endif
|
|
// *_retval = true;
|
|
// break;
|
|
}
|
|
}
|
|
}
|
|
return NS_OK;
|
|
}
|