#pragma once

#include <LibGfx/Rect.h>
#include <LibGfx/SharedImage.h>
#include <AK/Error.h>
#include <AK/MemoryStream.h>
#include <AK/OwnPtr.h>
#include <AK/Platform.h>
#include <AK/Result.h>
#include <AK/Utf8View.h>
#include <LibIPC/Attachment.h>
#include <LibIPC/Connection.h>
#include <LibIPC/Decoder.h>
#include <LibIPC/Encoder.h>
#include <LibIPC/File.h>
#include <LibIPC/Message.h>
#include <LibIPC/Stub.h>

#if defined(AK_COMPILER_CLANG)
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wdefaulted-function-deleted"
#endif

namespace Messages::CompositorClient {

enum class MessageID : i32 {
    DidAllocateBackingStores = 1,
    DidPaint = 2,
};

class DidAllocateBackingStores final : public IPC::Message {
public:
    DidAllocateBackingStores(u64 page_id, i32 front_bitmap_id, Gfx::SharedImage front_backing_store, i32 back_bitmap_id, Gfx::SharedImage back_backing_store)
        : m_page_id(move(page_id))
        , m_front_bitmap_id(move(front_bitmap_id))
        , m_front_backing_store(move(front_backing_store))
        , m_back_bitmap_id(move(back_bitmap_id))
        , m_back_backing_store(move(back_backing_store))
    {
    }

    DidAllocateBackingStores(DidAllocateBackingStores const&) = default;
    DidAllocateBackingStores(DidAllocateBackingStores&&) = default;
    DidAllocateBackingStores& operator=(DidAllocateBackingStores const&) = default;

    static constexpr u32 ENDPOINT_MAGIC = 2659200641;

    virtual u32 endpoint_magic() const override { return ENDPOINT_MAGIC; }
    virtual i32 message_id() const override { return (int)MessageID::DidAllocateBackingStores; }
    static i32 static_message_id() { return (int)MessageID::DidAllocateBackingStores; }
    virtual StringView message_name() const override { return "CompositorClient::DidAllocateBackingStores"sv; }

    static ErrorOr<NonnullOwnPtr<DidAllocateBackingStores>> decode(Stream& stream, Queue<IPC::Attachment>& attachments)
    {
        IPC::Decoder decoder { stream, attachments };
        auto page_id = TRY((decoder.decode<u64>()));
        auto front_bitmap_id = TRY((decoder.decode<i32>()));
        auto front_backing_store = TRY((decoder.decode<Gfx::SharedImage>()));
        auto back_bitmap_id = TRY((decoder.decode<i32>()));
        auto back_backing_store = TRY((decoder.decode<Gfx::SharedImage>()));
        return make<DidAllocateBackingStores>(move(page_id), move(front_bitmap_id), move(front_backing_store), move(back_bitmap_id), move(back_backing_store));
    }

    static ErrorOr<IPC::MessageBuffer> static_encode(u64 page_id, i32 front_bitmap_id, Gfx::SharedImage const& front_backing_store, i32 back_bitmap_id, Gfx::SharedImage const& back_backing_store)
    {
        IPC::MessageBuffer buffer;
        IPC::Encoder stream(buffer);
        TRY(stream.encode(ENDPOINT_MAGIC));
        TRY(stream.encode((int)MessageID::DidAllocateBackingStores));
        TRY(stream.encode(page_id));
        TRY(stream.encode(front_bitmap_id));
        TRY(stream.encode(front_backing_store));
        TRY(stream.encode(back_bitmap_id));
        TRY(stream.encode(back_backing_store));
        return buffer;
    }

    virtual ErrorOr<IPC::MessageBuffer> encode() const override
    {
        return static_encode(m_page_id, m_front_bitmap_id, m_front_backing_store, m_back_bitmap_id, m_back_backing_store);
    }

    u64 page_id() const { return m_page_id; }

    i32 front_bitmap_id() const { return m_front_bitmap_id; }

    Gfx::SharedImage const& front_backing_store() const { return m_front_backing_store; }
    Gfx::SharedImage take_front_backing_store() { return move(m_front_backing_store); }

    i32 back_bitmap_id() const { return m_back_bitmap_id; }

    Gfx::SharedImage const& back_backing_store() const { return m_back_backing_store; }
    Gfx::SharedImage take_back_backing_store() { return move(m_back_backing_store); }

private:
    u64 m_page_id;
    i32 m_front_bitmap_id;
    Gfx::SharedImage m_front_backing_store;
    i32 m_back_bitmap_id;
    Gfx::SharedImage m_back_backing_store;
};

class DidPaint final : public IPC::Message {
public:
    DidPaint(u64 page_id, Gfx::IntRect content_rect, i32 bitmap_id)
        : m_page_id(move(page_id))
        , m_content_rect(move(content_rect))
        , m_bitmap_id(move(bitmap_id))
    {
    }

    DidPaint(DidPaint const&) = default;
    DidPaint(DidPaint&&) = default;
    DidPaint& operator=(DidPaint const&) = default;

    static constexpr u32 ENDPOINT_MAGIC = 2659200641;

    virtual u32 endpoint_magic() const override { return ENDPOINT_MAGIC; }
    virtual i32 message_id() const override { return (int)MessageID::DidPaint; }
    static i32 static_message_id() { return (int)MessageID::DidPaint; }
    virtual StringView message_name() const override { return "CompositorClient::DidPaint"sv; }

    static ErrorOr<NonnullOwnPtr<DidPaint>> decode(Stream& stream, Queue<IPC::Attachment>& attachments)
    {
        IPC::Decoder decoder { stream, attachments };
        auto page_id = TRY((decoder.decode<u64>()));
        auto content_rect = TRY((decoder.decode<Gfx::IntRect>()));
        auto bitmap_id = TRY((decoder.decode<i32>()));
        return make<DidPaint>(move(page_id), move(content_rect), move(bitmap_id));
    }

    static ErrorOr<IPC::MessageBuffer> static_encode(u64 page_id, Gfx::IntRect const& content_rect, i32 bitmap_id)
    {
        IPC::MessageBuffer buffer;
        IPC::Encoder stream(buffer);
        TRY(stream.encode(ENDPOINT_MAGIC));
        TRY(stream.encode((int)MessageID::DidPaint));
        TRY(stream.encode(page_id));
        TRY(stream.encode(content_rect));
        TRY(stream.encode(bitmap_id));
        return buffer;
    }

    virtual ErrorOr<IPC::MessageBuffer> encode() const override
    {
        return static_encode(m_page_id, m_content_rect, m_bitmap_id);
    }

    u64 page_id() const { return m_page_id; }

    Gfx::IntRect const& content_rect() const { return m_content_rect; }
    Gfx::IntRect take_content_rect() { return move(m_content_rect); }

    i32 bitmap_id() const { return m_bitmap_id; }

private:
    u64 m_page_id;
    Gfx::IntRect m_content_rect;
    i32 m_bitmap_id;
};

} // namespace Messages::CompositorClient

template<typename LocalEndpoint, typename PeerEndpoint>
class CompositorClientProxy {
public:
    // Used to disambiguate the constructor call.
    struct Tag { };

    CompositorClientProxy(IPC::Connection<LocalEndpoint, PeerEndpoint>& connection, Tag)
        : m_connection(connection)
    {
    }

    void async_did_allocate_backing_stores(u64 page_id, i32 front_bitmap_id, Gfx::SharedImage const& front_backing_store, i32 back_bitmap_id, Gfx::SharedImage const& back_backing_store)
    {
        auto message_buffer = MUST(Messages::CompositorClient::DidAllocateBackingStores::static_encode(page_id, front_bitmap_id, move(front_backing_store), back_bitmap_id, move(back_backing_store)));
        (void)m_connection.post_message(message_buffer);
    }

    void async_did_paint(u64 page_id, Gfx::IntRect const& content_rect, i32 bitmap_id)
    {
        auto message_buffer = MUST(Messages::CompositorClient::DidPaint::static_encode(page_id, move(content_rect), bitmap_id));
        (void)m_connection.post_message(message_buffer);
    }

private:
    IPC::Connection<LocalEndpoint, PeerEndpoint>& m_connection;
};

template<typename LocalEndpoint, typename PeerEndpoint>
class CompositorClientProxy;
class CompositorClientStub;

class CompositorClientEndpoint {
public:
    template<typename LocalEndpoint>
    using Proxy = CompositorClientProxy<LocalEndpoint, CompositorClientEndpoint>;
    using Stub = CompositorClientStub;

    static u32 static_magic() { return 2659200641; }

    static ErrorOr<NonnullOwnPtr<IPC::Message>> decode_message(ReadonlyBytes buffer, [[maybe_unused]] Queue<IPC::Attachment>& attachments)
    {
        FixedMemoryStream stream { buffer };
        auto message_endpoint_magic = TRY(stream.read_value<u32>());

        if (message_endpoint_magic != static_magic())
            return Error::from_string_literal("Endpoint magic number mismatch, not my message!");

        auto message_id = TRY(stream.read_value<i32>());

        switch (message_id) {
        case (int)Messages::CompositorClient::MessageID::DidAllocateBackingStores:
            return Messages::CompositorClient::DidAllocateBackingStores::decode(stream, attachments);
        case (int)Messages::CompositorClient::MessageID::DidPaint:
            return Messages::CompositorClient::DidPaint::decode(stream, attachments);
        default:
            return Error::from_string_literal("Failed to decode CompositorClient message");
        }

        VERIFY_NOT_REACHED();
    }
};

class CompositorClientStub : public IPC::Stub {
public:
    CompositorClientStub() { }
    virtual ~CompositorClientStub() override { }

    virtual u32 magic() const override { return 2659200641; }
    virtual ByteString name() const override { return "CompositorClient"; }

    virtual ErrorOr<OwnPtr<IPC::MessageBuffer>> handle(NonnullOwnPtr<IPC::Message> message) override
    {
        switch (message->message_id()) {
        case (int)Messages::CompositorClient::MessageID::DidAllocateBackingStores:
            return handle_did_allocate_backing_stores(*message);
        case (int)Messages::CompositorClient::MessageID::DidPaint:
            return handle_did_paint(*message);
        default:
            return Error::from_string_literal("Unknown message ID for CompositorClient endpoint");
        }
    }

    NEVER_INLINE ErrorOr<OwnPtr<IPC::MessageBuffer>> handle_did_allocate_backing_stores(IPC::Message& message)
    {
        auto& request = static_cast<Messages::CompositorClient::DidAllocateBackingStores&>(message);
        did_allocate_backing_stores(request.page_id(), request.front_bitmap_id(), request.take_front_backing_store(), request.back_bitmap_id(), request.take_back_backing_store());
        return nullptr;
    }

    NEVER_INLINE ErrorOr<OwnPtr<IPC::MessageBuffer>> handle_did_paint(IPC::Message& message)
    {
        auto& request = static_cast<Messages::CompositorClient::DidPaint&>(message);
        did_paint(request.page_id(), request.take_content_rect(), request.bitmap_id());
        return nullptr;
    }

    virtual void did_allocate_backing_stores(u64 page_id, i32 front_bitmap_id, Gfx::SharedImage front_backing_store, i32 back_bitmap_id, Gfx::SharedImage back_backing_store) = 0;
    virtual void did_paint(u64 page_id, Gfx::IntRect content_rect, i32 bitmap_id) = 0;
};

#if defined(AK_COMPILER_CLANG)
#pragma clang diagnostic pop
#endif
