/* * Copyright (c) 2020-2021, Andreas Kling * Copyright (c) 2021, Mustafa Quraish * Copyright (c) 2021, Tobias Christiansen * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace PixelPaint { class Layer; class Selection; class ImageClient { public: virtual void image_did_add_layer(size_t) { } virtual void image_did_remove_layer(size_t) { } virtual void image_did_modify_layer_properties(size_t) { } virtual void image_did_modify_layer_bitmap(size_t) { } virtual void image_did_modify_layer_stack() { } virtual void image_did_change(Gfx::IntRect const&) { } virtual void image_did_change_rect(Gfx::IntRect const&) { } virtual void image_select_layer(Layer*) { } virtual void image_did_change_title(String const&) { } protected: virtual ~ImageClient() = default; }; class Image : public RefCounted { public: static RefPtr try_create_with_size(Gfx::IntSize const&); static Result, String> try_create_from_pixel_paint_json(JsonObject const&); static RefPtr try_create_from_bitmap(NonnullRefPtr); static RefPtr try_decode_bitmap(const ReadonlyBytes& bitmap_data); // This generates a new Bitmap with the final image (all layers composed according to their attributes.) RefPtr try_compose_bitmap(Gfx::BitmapFormat format) const; RefPtr try_copy_bitmap(Selection const&) const; size_t layer_count() const { return m_layers.size(); } Layer const& layer(size_t index) const { return m_layers.at(index); } Layer& layer(size_t index) { return m_layers.at(index); } Gfx::IntSize const& size() const { return m_size; } Gfx::IntRect rect() const { return { {}, m_size }; } void add_layer(NonnullRefPtr); RefPtr take_snapshot() const; void restore_snapshot(Image const&); void paint_into(GUI::Painter&, Gfx::IntRect const& dest_rect) const; void serialize_as_json(JsonObjectSerializer& json) const; Result write_to_file(String const& file_path) const; Result export_bmp_to_fd_and_close(int fd, bool preserve_alpha_channel); Result export_png_to_fd_and_close(int fd, bool preserve_alpha_channel); void move_layer_to_front(Layer&); void move_layer_to_back(Layer&); void move_layer_up(Layer&); void move_layer_down(Layer&); void change_layer_index(size_t old_index, size_t new_index); void remove_layer(Layer&); void select_layer(Layer*); void flatten_all_layers(); void merge_visible_layers(); void merge_active_layer_down(Layer& layer); void add_client(ImageClient&); void remove_client(ImageClient&); void layer_did_modify_bitmap(Badge, Layer const&, Gfx::IntRect const& modified_layer_rect); void layer_did_modify_properties(Badge, Layer const&); size_t index_of(Layer const&) const; String const& path() const { return m_path; } void set_path(String); String const& title() const { return m_title; } void set_title(String); void flip(Gfx::Orientation orientation); void rotate(Gfx::RotationDirection direction); void crop(Gfx::IntRect const& rect); Color color_at(Gfx::IntPoint const& point) const; private: explicit Image(Gfx::IntSize const&); void did_change(Gfx::IntRect const& modified_rect = {}); void did_change_rect(Gfx::IntRect const& modified_rect = {}); void did_modify_layer_stack(); String m_path; String m_title; Gfx::IntSize m_size; NonnullRefPtrVector m_layers; HashTable m_clients; }; class ImageUndoCommand : public GUI::Command { public: ImageUndoCommand(Image& image); virtual void undo() override; virtual void redo() override; private: RefPtr m_snapshot; Image& m_image; }; }