2019-06-22 05:26:42 -05:00
|
|
|
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */
|
|
|
|
/*
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Clipboard abstraction.
|
2020-04-18 03:39:50 -05:00
|
|
|
|
|
|
|
#pragma once
|
2019-06-22 05:26:42 -05:00
|
|
|
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2019-07-04 04:50:33 -05:00
|
|
|
#include <unordered_map>
|
|
|
|
#include <mutex>
|
|
|
|
|
2019-06-22 05:26:42 -05:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <Log.hpp>
|
|
|
|
#include <Exceptions.hpp>
|
2019-07-04 04:50:33 -05:00
|
|
|
#include <Poco/MemoryStream.h>
|
2019-06-22 05:26:42 -05:00
|
|
|
|
|
|
|
struct ClipboardData
|
|
|
|
{
|
|
|
|
std::vector<std::string> _mimeTypes;
|
|
|
|
std::vector<std::string> _content;
|
|
|
|
ClipboardData()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void read(std::istream& inStream)
|
|
|
|
{
|
|
|
|
while (!inStream.eof())
|
|
|
|
{
|
|
|
|
std::string mime, hexLen, newline;
|
|
|
|
std::getline(inStream, mime, '\n');
|
|
|
|
std::getline(inStream, hexLen, '\n');
|
2019-06-24 07:36:46 -05:00
|
|
|
if (mime.length() && hexLen.length() && !inStream.fail())
|
2019-06-22 05:26:42 -05:00
|
|
|
{
|
2019-06-24 07:36:46 -05:00
|
|
|
uint64_t len = strtoll( hexLen.c_str(), nullptr, 16 );
|
|
|
|
std::string content(len, ' ');
|
|
|
|
inStream.read(&content[0], len);
|
|
|
|
if (inStream.fail())
|
|
|
|
throw ParseError("error during reading the stream");
|
|
|
|
std::getline(inStream, newline, '\n');
|
|
|
|
if (mime.length() > 0)
|
|
|
|
{
|
|
|
|
_mimeTypes.push_back(mime);
|
|
|
|
_content.push_back(content);
|
|
|
|
}
|
2019-06-22 05:26:42 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-15 11:03:45 -06:00
|
|
|
std::size_t size() const
|
2019-06-22 05:26:42 -05:00
|
|
|
{
|
|
|
|
assert(_mimeTypes.size() == _content.size());
|
|
|
|
return _mimeTypes.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
void dumpState(std::ostream& os)
|
|
|
|
{
|
|
|
|
os << "Clipboard with " << size() << " entries\n";
|
|
|
|
for (size_t i = 0; i < size(); ++i)
|
|
|
|
os << "\t[" << i << "] - size " << _content[i].size() <<
|
|
|
|
" type: '" << _mimeTypes[i] << "'\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
bool findType(const std::string &mime, std::string &value)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < _mimeTypes.size(); ++i)
|
|
|
|
{
|
|
|
|
if (_mimeTypes[i] == mime)
|
|
|
|
{
|
|
|
|
value = _content[i];
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
value = "";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-07-04 04:50:33 -05:00
|
|
|
/// Used to store expired view's clipboards
|
|
|
|
class ClipboardCache
|
|
|
|
{
|
|
|
|
std::mutex _mutex;
|
|
|
|
struct Entry {
|
|
|
|
std::chrono::steady_clock::time_point _inserted;
|
|
|
|
std::shared_ptr<std::string> _rawData; // big.
|
|
|
|
};
|
|
|
|
// clipboard key -> data
|
|
|
|
std::unordered_map<std::string, Entry> _cache;
|
|
|
|
public:
|
|
|
|
ClipboardCache()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void insertClipboard(const std::string key[2],
|
2020-11-15 11:03:45 -06:00
|
|
|
const char *data, std::size_t size)
|
2019-07-04 04:50:33 -05:00
|
|
|
{
|
|
|
|
if (size == 0)
|
|
|
|
{
|
|
|
|
LOG_TRC("clipboard cache - ignores empty clipboard data");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Entry ent;
|
|
|
|
ent._inserted = std::chrono::steady_clock::now();
|
|
|
|
ent._rawData = std::make_shared<std::string>(data, size);
|
|
|
|
LOG_TRC("insert cached clipboard: " + key[0] + " and " + key[1]);
|
|
|
|
std::lock_guard<std::mutex> lock(_mutex);
|
|
|
|
_cache[key[0]] = ent;
|
|
|
|
_cache[key[1]] = ent;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::shared_ptr<std::string> getClipboard(const std::string &key)
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> lock(_mutex);
|
|
|
|
std::shared_ptr<std::string> data;
|
|
|
|
auto it = _cache.find(key);
|
|
|
|
if (it != _cache.end())
|
|
|
|
data = it->second._rawData;
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
void checkexpiry()
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> lock(_mutex);
|
|
|
|
auto now = std::chrono::steady_clock::now();
|
|
|
|
LOG_TRC("check expiry of cached clipboards");
|
|
|
|
for (auto it = _cache.begin(); it != _cache.end();)
|
|
|
|
{
|
|
|
|
if (std::chrono::duration_cast<std::chrono::minutes>(now - it->second._inserted).count() >= 10)
|
|
|
|
{
|
|
|
|
LOG_TRC("expiring expiry of cached clipboard: " + it->first);
|
|
|
|
it = _cache.erase(it);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-06-22 05:26:42 -05:00
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|