mirror of
https://github.com/RGBCube/serenity
synced 2025-07-26 07:07:45 +00:00
SpiceAgent: Support copying and pasting images
This commit is contained in:
parent
d4bb6a1a1e
commit
c8b13bd053
4 changed files with 143 additions and 8 deletions
|
@ -5,4 +5,64 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "ClipboardServerConnection.h"
|
#include "ClipboardServerConnection.h"
|
||||||
|
#include <AK/ByteBuffer.h>
|
||||||
|
#include <AK/Function.h>
|
||||||
|
#include <LibGfx/Bitmap.h>
|
||||||
|
|
||||||
|
// Copied from LibGUI/Clipboard.cpp
|
||||||
|
RefPtr<Gfx::Bitmap> ClipboardServerConnection::get_bitmap()
|
||||||
|
{
|
||||||
|
auto clipping = get_clipboard_data();
|
||||||
|
|
||||||
|
if (clipping.mime_type() != "image/x-serenityos")
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
HashMap<String, String> const& metadata = clipping.metadata().entries();
|
||||||
|
auto width = metadata.get("width").value_or("0").to_uint();
|
||||||
|
if (!width.has_value() || width.value() == 0)
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
auto height = metadata.get("height").value_or("0").to_uint();
|
||||||
|
if (!height.has_value() || height.value() == 0)
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
auto scale = metadata.get("scale").value_or("0").to_uint();
|
||||||
|
if (!scale.has_value() || scale.value() == 0)
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
auto pitch = metadata.get("pitch").value_or("0").to_uint();
|
||||||
|
if (!pitch.has_value() || pitch.value() == 0)
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
auto format = metadata.get("format").value_or("0").to_uint();
|
||||||
|
if (!format.has_value() || format.value() == 0)
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
auto data = ByteBuffer::copy(clipping.data().data<void>(), clipping.data().size());
|
||||||
|
auto clipping_bitmap = Gfx::Bitmap::create_wrapper((Gfx::BitmapFormat)format.value(), { (int)width.value(), (int)height.value() }, scale.value(), pitch.value(), data.data());
|
||||||
|
auto bitmap = Gfx::Bitmap::create(Gfx::BitmapFormat::BGRA8888, { (int)width.value(), (int)height.value() }, scale.value());
|
||||||
|
|
||||||
|
for (int y = 0; y < clipping_bitmap->physical_height(); ++y) {
|
||||||
|
for (int x = 0; x < clipping_bitmap->physical_width(); ++x) {
|
||||||
|
auto pixel = clipping_bitmap->get_pixel(x, y);
|
||||||
|
bitmap->set_pixel(x, y, pixel);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return bitmap;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copied from LibGUI/Clipboard.cpp
|
||||||
|
void ClipboardServerConnection::set_bitmap(Gfx::Bitmap const& bitmap)
|
||||||
|
{
|
||||||
|
HashMap<String, String> metadata;
|
||||||
|
metadata.set("width", String::number(bitmap.width()));
|
||||||
|
metadata.set("height", String::number(bitmap.height()));
|
||||||
|
metadata.set("scale", String::number(bitmap.scale()));
|
||||||
|
metadata.set("format", String::number((int)bitmap.format()));
|
||||||
|
metadata.set("pitch", String::number(bitmap.pitch()));
|
||||||
|
ReadonlyBytes data { bitmap.scanline(0), bitmap.size_in_bytes() };
|
||||||
|
auto buffer = Core::AnonymousBuffer::create_with_size(bitmap.size_in_bytes());
|
||||||
|
memcpy(buffer.data<u8>(), data.data(), data.size());
|
||||||
|
this->async_set_clipboard_data(buffer, "image/x-serenityos", metadata);
|
||||||
|
}
|
||||||
|
|
|
@ -18,6 +18,8 @@ class ClipboardServerConnection final
|
||||||
C_OBJECT(ClipboardServerConnection);
|
C_OBJECT(ClipboardServerConnection);
|
||||||
|
|
||||||
Function<void()> on_data_changed;
|
Function<void()> on_data_changed;
|
||||||
|
RefPtr<Gfx::Bitmap> get_bitmap();
|
||||||
|
void set_bitmap(Gfx::Bitmap const& bitmap);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
ClipboardServerConnection()
|
ClipboardServerConnection()
|
||||||
|
|
|
@ -9,6 +9,12 @@
|
||||||
#include <AK/String.h>
|
#include <AK/String.h>
|
||||||
#include <LibC/memory.h>
|
#include <LibC/memory.h>
|
||||||
#include <LibC/unistd.h>
|
#include <LibC/unistd.h>
|
||||||
|
#include <LibGfx/BMPLoader.h>
|
||||||
|
#include <LibGfx/BMPWriter.h>
|
||||||
|
#include <LibGfx/Bitmap.h>
|
||||||
|
#include <LibGfx/JPGLoader.h>
|
||||||
|
#include <LibGfx/PNGLoader.h>
|
||||||
|
#include <LibGfx/PNGWriter.h>
|
||||||
|
|
||||||
SpiceAgent::SpiceAgent(int fd, ClipboardServerConnection& connection)
|
SpiceAgent::SpiceAgent(int fd, ClipboardServerConnection& connection)
|
||||||
: m_fd(fd)
|
: m_fd(fd)
|
||||||
|
@ -23,13 +29,32 @@ SpiceAgent::SpiceAgent(int fd, ClipboardServerConnection& connection)
|
||||||
m_just_set_clip = false;
|
m_just_set_clip = false;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
auto grab_buffer = ClipboardGrab::make_buffer({ ClipboardType::Text });
|
auto mime = m_clipboard_connection.get_clipboard_data().mime_type();
|
||||||
|
Optional<ClipboardType> type = mime_type_to_clipboard_type(mime);
|
||||||
|
if (!type.has_value())
|
||||||
|
return;
|
||||||
|
|
||||||
|
auto grab_buffer = ClipboardGrab::make_buffer({ *type });
|
||||||
send_message(grab_buffer);
|
send_message(grab_buffer);
|
||||||
};
|
};
|
||||||
auto buffer = AnnounceCapabilities::make_buffer(true, { Capability::ClipboardByDemand });
|
auto buffer = AnnounceCapabilities::make_buffer(true, { Capability::ClipboardByDemand });
|
||||||
send_message(buffer);
|
send_message(buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Optional<SpiceAgent::ClipboardType> SpiceAgent::mime_type_to_clipboard_type(const String& mime)
|
||||||
|
{
|
||||||
|
if (mime == "text/plain")
|
||||||
|
return ClipboardType::Text;
|
||||||
|
else if (mime == "image/jpeg")
|
||||||
|
return ClipboardType::JPG;
|
||||||
|
else if (mime == "image/bmp")
|
||||||
|
return ClipboardType::BMP;
|
||||||
|
else if (mime == "image/png" || mime == "image/x-serenityos")
|
||||||
|
return ClipboardType::PNG;
|
||||||
|
else
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
void SpiceAgent::on_message_received()
|
void SpiceAgent::on_message_received()
|
||||||
{
|
{
|
||||||
ChunkHeader header {};
|
ChunkHeader header {};
|
||||||
|
@ -47,19 +72,50 @@ void SpiceAgent::on_message_received()
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case (u32)MessageType::ClipboardRequest: {
|
case (u32)MessageType::ClipboardRequest: {
|
||||||
auto clip_data = m_clipboard_connection.get_clipboard_data().data();
|
auto* request_message = reinterpret_cast<ClipboardRequest*>(message->data);
|
||||||
ByteBuffer byte_buffer = ByteBuffer::copy(clip_data.data<void>(), clip_data.size());
|
auto clipboard = m_clipboard_connection.get_clipboard_data();
|
||||||
auto clipboard_buffer = Clipboard::make_buffer(ClipboardType::Text, byte_buffer);
|
auto& mime = clipboard.mime_type();
|
||||||
|
ByteBuffer byte_buffer;
|
||||||
|
if (mime == "image/x-serenityos") {
|
||||||
|
auto bitmap = m_clipboard_connection.get_bitmap();
|
||||||
|
byte_buffer = Gfx::PNGWriter::encode(*bitmap);
|
||||||
|
} else {
|
||||||
|
auto clip_data = clipboard.data();
|
||||||
|
byte_buffer = ByteBuffer::copy(clip_data.data<void>(), clip_data.size());
|
||||||
|
}
|
||||||
|
auto clipboard_buffer = Clipboard::make_buffer((ClipboardType)request_message->type, byte_buffer);
|
||||||
send_message(clipboard_buffer);
|
send_message(clipboard_buffer);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case (u32)MessageType::ClipboardGrab: {
|
case (u32)MessageType::ClipboardGrab: {
|
||||||
auto request_buffer = ClipboardRequest::make_buffer(ClipboardType::Text);
|
auto* grab_message = reinterpret_cast<ClipboardGrab*>(message->data);
|
||||||
|
auto found_type = ClipboardType::None;
|
||||||
|
for (size_t i = 0; i < (message->size / 4); i++) {
|
||||||
|
auto type = (ClipboardType)grab_message->types[i];
|
||||||
|
if (found_type == ClipboardType::None) {
|
||||||
|
found_type = static_cast<ClipboardType>(type);
|
||||||
|
} else if (found_type == ClipboardType::Text) {
|
||||||
|
switch (type) {
|
||||||
|
case ClipboardType::PNG:
|
||||||
|
case ClipboardType::BMP:
|
||||||
|
case ClipboardType::JPG:
|
||||||
|
found_type = type;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (found_type == ClipboardType::None)
|
||||||
|
return;
|
||||||
|
|
||||||
|
auto request_buffer = ClipboardRequest::make_buffer(found_type);
|
||||||
send_message(request_buffer);
|
send_message(request_buffer);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case (u32)MessageType::Clipboard: {
|
case (u32)MessageType::Clipboard: {
|
||||||
auto* clipboard_message = reinterpret_cast<Clipboard*>(message->data);
|
auto* clipboard_message = reinterpret_cast<Clipboard*>(message->data);
|
||||||
|
auto type = (ClipboardType)clipboard_message->type;
|
||||||
auto data_buffer = ByteBuffer::create_uninitialized(message->size - sizeof(u32));
|
auto data_buffer = ByteBuffer::create_uninitialized(message->size - sizeof(u32));
|
||||||
|
|
||||||
const auto total_bytes = message->size - sizeof(Clipboard);
|
const auto total_bytes = message->size - sizeof(Clipboard);
|
||||||
|
@ -74,9 +130,25 @@ void SpiceAgent::on_message_received()
|
||||||
}
|
}
|
||||||
|
|
||||||
m_just_set_clip = true;
|
m_just_set_clip = true;
|
||||||
auto anon_buffer = Core::AnonymousBuffer::create_with_size(data_buffer.size());
|
if (type == ClipboardType::Text) {
|
||||||
memcpy(anon_buffer.data<void>(), data_buffer.data(), data_buffer.size());
|
auto anon_buffer = Core::AnonymousBuffer::create_with_size(data_buffer.size());
|
||||||
m_clipboard_connection.async_set_clipboard_data(anon_buffer, "text/plain", {});
|
memcpy(anon_buffer.data<void>(), data_buffer.data(), data_buffer.size());
|
||||||
|
m_clipboard_connection.async_set_clipboard_data(anon_buffer, "text/plain", {});
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
RefPtr<Gfx::Bitmap> bitmap;
|
||||||
|
if (type == ClipboardType::PNG) {
|
||||||
|
bitmap = Gfx::load_png_from_memory(data_buffer.data(), data_buffer.size());
|
||||||
|
} else if (type == ClipboardType::BMP) {
|
||||||
|
bitmap = Gfx::load_bmp_from_memory(data_buffer.data(), data_buffer.size());
|
||||||
|
} else if (type == ClipboardType::JPG) {
|
||||||
|
bitmap = Gfx::load_jpg_from_memory(data_buffer.data(), data_buffer.size());
|
||||||
|
} else {
|
||||||
|
dbgln("Unknown clipboard type: {}", (u32)type);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
m_clipboard_connection.set_bitmap(*bitmap);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
|
|
|
@ -124,4 +124,5 @@ private:
|
||||||
bool m_just_set_clip { false };
|
bool m_just_set_clip { false };
|
||||||
void read_n(void* dest, size_t n);
|
void read_n(void* dest, size_t n);
|
||||||
static Message* initialize_headers(u8* data, size_t additional_data_size, MessageType type);
|
static Message* initialize_headers(u8* data, size_t additional_data_size, MessageType type);
|
||||||
|
static Optional<ClipboardType> mime_type_to_clipboard_type(const String& mime);
|
||||||
};
|
};
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue