232 lines
6.4 KiB
C++
232 lines
6.4 KiB
C++
/*
|
|
* barrier -- mouse and keyboard sharing utility
|
|
* Copyright (C) 2012-2016 Symless Ltd.
|
|
* Copyright (C) 2002 Chris Schoeneman
|
|
*
|
|
* This package is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* found in the file LICENSE that should have accompanied this file.
|
|
*
|
|
* This package 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 program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "platform/MSWindowsClipboard.h"
|
|
|
|
#include "platform/MSWindowsClipboardTextConverter.h"
|
|
#include "platform/MSWindowsClipboardUTF16Converter.h"
|
|
#include "platform/MSWindowsClipboardBitmapConverter.h"
|
|
#include "platform/MSWindowsClipboardHTMLConverter.h"
|
|
#include "platform/MSWindowsClipboardFacade.h"
|
|
#include "arch/win32/ArchMiscWindows.h"
|
|
#include "base/Log.h"
|
|
|
|
//
|
|
// MSWindowsClipboard
|
|
//
|
|
|
|
UINT MSWindowsClipboard::s_ownershipFormat = 0;
|
|
|
|
MSWindowsClipboard::MSWindowsClipboard(HWND window) :
|
|
m_window(window),
|
|
m_time(0),
|
|
m_facade(new MSWindowsClipboardFacade()),
|
|
m_deleteFacade(true)
|
|
{
|
|
// add converters, most desired first
|
|
m_converters.push_back(new MSWindowsClipboardUTF16Converter);
|
|
m_converters.push_back(new MSWindowsClipboardBitmapConverter);
|
|
m_converters.push_back(new MSWindowsClipboardHTMLConverter);
|
|
}
|
|
|
|
MSWindowsClipboard::~MSWindowsClipboard()
|
|
{
|
|
clearConverters();
|
|
|
|
// dependency injection causes confusion over ownership, so we need
|
|
// logic to decide whether or not we delete the facade. there must
|
|
// be a more elegant way of doing this.
|
|
if (m_deleteFacade)
|
|
delete m_facade;
|
|
}
|
|
|
|
void
|
|
MSWindowsClipboard::setFacade(IMSWindowsClipboardFacade& facade)
|
|
{
|
|
delete m_facade;
|
|
m_facade = &facade;
|
|
m_deleteFacade = false;
|
|
}
|
|
|
|
bool
|
|
MSWindowsClipboard::emptyUnowned()
|
|
{
|
|
LOG((CLOG_DEBUG "empty clipboard"));
|
|
|
|
// empty the clipboard (and take ownership)
|
|
if (!EmptyClipboard()) {
|
|
// unable to cause this in integ tests, but this error has never
|
|
// actually been reported by users.
|
|
LOG((CLOG_DEBUG "failed to grab clipboard"));
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
MSWindowsClipboard::empty()
|
|
{
|
|
if (!emptyUnowned()) {
|
|
return false;
|
|
}
|
|
|
|
// mark clipboard as being owned by barrier
|
|
HGLOBAL data = GlobalAlloc(GMEM_MOVEABLE | GMEM_DDESHARE, 1);
|
|
if (NULL == SetClipboardData(getOwnershipFormat(), data)) {
|
|
LOG((CLOG_DEBUG "failed to set clipboard data"));
|
|
GlobalFree(data);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void
|
|
MSWindowsClipboard::add(EFormat format, const std::string& data)
|
|
{
|
|
LOG((CLOG_DEBUG "add %d bytes to clipboard format: %d", data.size(), format));
|
|
|
|
// convert data to win32 form
|
|
for (ConverterList::const_iterator index = m_converters.begin();
|
|
index != m_converters.end(); ++index) {
|
|
IMSWindowsClipboardConverter* converter = *index;
|
|
|
|
// skip converters for other formats
|
|
if (converter->getFormat() == format) {
|
|
HANDLE win32Data = converter->fromIClipboard(data);
|
|
if (win32Data != NULL) {
|
|
UINT win32Format = converter->getWin32Format();
|
|
m_facade->write(win32Data, win32Format);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool
|
|
MSWindowsClipboard::open(Time time) const
|
|
{
|
|
LOG((CLOG_DEBUG "open clipboard"));
|
|
|
|
if (!OpenClipboard(m_window)) {
|
|
// unable to cause this in integ tests; but this can happen!
|
|
// * http://symless.com/pm/issues/86
|
|
// * http://symless.com/pm/issues/1256
|
|
// logging improved to see if we can catch more info next time.
|
|
LOG((CLOG_WARN "failed to open clipboard: %d", GetLastError()));
|
|
return false;
|
|
}
|
|
|
|
m_time = time;
|
|
|
|
return true;
|
|
}
|
|
|
|
void
|
|
MSWindowsClipboard::close() const
|
|
{
|
|
LOG((CLOG_DEBUG "close clipboard"));
|
|
CloseClipboard();
|
|
}
|
|
|
|
IClipboard::Time
|
|
MSWindowsClipboard::getTime() const
|
|
{
|
|
return m_time;
|
|
}
|
|
|
|
bool
|
|
MSWindowsClipboard::has(EFormat format) const
|
|
{
|
|
for (ConverterList::const_iterator index = m_converters.begin();
|
|
index != m_converters.end(); ++index) {
|
|
IMSWindowsClipboardConverter* converter = *index;
|
|
if (converter->getFormat() == format) {
|
|
if (IsClipboardFormatAvailable(converter->getWin32Format())) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
std::string MSWindowsClipboard::get(EFormat format) const
|
|
{
|
|
// find the converter for the first clipboard format we can handle
|
|
IMSWindowsClipboardConverter* converter = NULL;
|
|
for (ConverterList::const_iterator index = m_converters.begin();
|
|
index != m_converters.end(); ++index) {
|
|
|
|
converter = *index;
|
|
if (converter->getFormat() == format) {
|
|
break;
|
|
}
|
|
converter = NULL;
|
|
}
|
|
|
|
// if no converter then we don't recognize any formats
|
|
if (converter == NULL) {
|
|
LOG((CLOG_WARN "no converter for format %d", format));
|
|
return {};
|
|
}
|
|
|
|
// get a handle to the clipboard data
|
|
HANDLE win32Data = GetClipboardData(converter->getWin32Format());
|
|
if (win32Data == NULL) {
|
|
// nb: can't cause this using integ tests; this is only caused when
|
|
// the selected converter returns an invalid format -- which you
|
|
// cannot cause using public functions.
|
|
return {};
|
|
}
|
|
|
|
// convert
|
|
return converter->toIClipboard(win32Data);
|
|
}
|
|
|
|
void
|
|
MSWindowsClipboard::clearConverters()
|
|
{
|
|
for (ConverterList::iterator index = m_converters.begin();
|
|
index != m_converters.end(); ++index) {
|
|
delete *index;
|
|
}
|
|
m_converters.clear();
|
|
}
|
|
|
|
bool
|
|
MSWindowsClipboard::isOwnedByBarrier()
|
|
{
|
|
// create ownership format if we haven't yet
|
|
if (s_ownershipFormat == 0) {
|
|
s_ownershipFormat = RegisterClipboardFormat(TEXT("BarrierOwnership"));
|
|
}
|
|
return (IsClipboardFormatAvailable(getOwnershipFormat()) != 0);
|
|
}
|
|
|
|
UINT
|
|
MSWindowsClipboard::getOwnershipFormat()
|
|
{
|
|
// create ownership format if we haven't yet
|
|
if (s_ownershipFormat == 0) {
|
|
s_ownershipFormat = RegisterClipboardFormat(TEXT("BarrierOwnership"));
|
|
}
|
|
|
|
// return the format
|
|
return s_ownershipFormat;
|
|
}
|