/* * Copyright (c) 2018-2020, Andreas Kling * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include namespace GUI { class ClipboardServerConnection : public IPC::ServerConnection , public ClipboardClientEndpoint { C_OBJECT(ClipboardServerConnection); public: virtual void handshake() override { auto response = send_sync(); set_my_client_id(response->client_id()); } private: ClipboardServerConnection() : IPC::ServerConnection(*this, "/tmp/portal/clipboard") { } virtual void handle(const Messages::ClipboardClient::ClipboardDataChanged&) override; }; Clipboard& Clipboard::the() { static Clipboard* s_the; if (!s_the) s_the = new Clipboard; return *s_the; } ClipboardServerConnection* s_connection; static ClipboardServerConnection& connection() { return *s_connection; } void Clipboard::initialize(Badge) { s_connection = &ClipboardServerConnection::construct().leak_ref(); } Clipboard::Clipboard() { } Clipboard::DataAndType Clipboard::data_and_type() const { auto response = connection().send_sync(); if (response->shbuf_id() < 0) return {}; auto shared_buffer = SharedBuffer::create_from_shbuf_id(response->shbuf_id()); if (!shared_buffer) { dbgprintf("GUI::Clipboard::data() failed to attach to the shared buffer\n"); return {}; } if (response->data_size() > shared_buffer->size()) { dbgprintf("GUI::Clipboard::data() clipping contents size is greater than shared buffer size\n"); return {}; } auto data = String((const char*)shared_buffer->data(), response->data_size()); auto type = response->mime_type(); return { data, type }; } void Clipboard::set_data(const StringView& data, const String& type) { auto shared_buffer = SharedBuffer::create_with_size(data.length() + 1); if (!shared_buffer) { dbgprintf("GUI::Clipboard::set_data() failed to create a shared buffer\n"); return; } if (!data.is_empty()) memcpy(shared_buffer->data(), data.characters_without_null_termination(), data.length() + 1); else ((u8*)shared_buffer->data())[0] = '\0'; shared_buffer->seal(); shared_buffer->share_with(connection().server_pid()); connection().send_sync(shared_buffer->shbuf_id(), data.length(), type); } void ClipboardServerConnection::handle(const Messages::ClipboardClient::ClipboardDataChanged& message) { auto& clipboard = Clipboard::the(); if (clipboard.on_change) clipboard.on_change(message.mime_type()); } }