2021-11-01 04:40:58 +01:00
|
|
|
//
|
2021-12-18 06:38:30 +01:00
|
|
|
// fiv-io.h: image operations
|
2021-11-01 04:40:58 +01:00
|
|
|
//
|
2021-12-31 02:19:17 +01:00
|
|
|
// Copyright (c) 2021 - 2022, Přemysl Eric Janouch <p@janouch.name>
|
2021-11-01 04:40:58 +01:00
|
|
|
//
|
|
|
|
// Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
// purpose with or without fee is hereby granted.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
// WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
// MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
|
|
|
|
// SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
// WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
|
|
|
|
// OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
|
|
|
// CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
//
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <cairo.h>
|
2021-11-21 16:03:54 +01:00
|
|
|
#include <gio/gio.h>
|
2021-12-18 06:38:30 +01:00
|
|
|
#include <glib.h>
|
2022-01-13 23:42:17 +01:00
|
|
|
#include <webp/encode.h> // WebPConfig
|
2021-11-01 04:40:58 +01:00
|
|
|
|
2021-12-22 22:07:49 +01:00
|
|
|
// --- Colour management -------------------------------------------------------
|
|
|
|
|
|
|
|
// TODO(p): Make it possible to use Skia's skcms,
|
|
|
|
// which also supports premultiplied alpha.
|
2022-01-22 22:51:46 +01:00
|
|
|
// NOTE: Little CMS 2.13 will support premultiplied alpha in 2022.
|
2021-12-22 22:07:49 +01:00
|
|
|
typedef void *FivIoProfile;
|
|
|
|
FivIoProfile fiv_io_profile_new(const void *data, size_t len);
|
|
|
|
FivIoProfile fiv_io_profile_new_sRGB(void);
|
|
|
|
void fiv_io_profile_free(FivIoProfile self);
|
|
|
|
|
2021-12-28 19:58:14 +01:00
|
|
|
// From libwebp, verified to exactly match [x * a / 255].
|
|
|
|
#define PREMULTIPLY8(a, x) (((uint32_t) (x) * (uint32_t) (a) * 32897U) >> 23)
|
|
|
|
|
2021-12-22 22:07:49 +01:00
|
|
|
// --- Loading -----------------------------------------------------------------
|
|
|
|
|
2021-12-18 06:38:30 +01:00
|
|
|
extern const char *fiv_io_supported_media_types[];
|
2021-11-01 04:40:58 +01:00
|
|
|
|
2022-06-05 13:29:38 +02:00
|
|
|
gchar **fiv_io_all_supported_media_types(void);
|
2021-11-18 12:44:25 +01:00
|
|
|
|
2021-11-25 02:46:36 +01:00
|
|
|
// Userdata are typically attached to all Cairo surfaces in an animation.
|
|
|
|
|
2021-11-28 20:05:36 +01:00
|
|
|
/// GBytes with plain Exif/TIFF data.
|
2021-12-18 06:38:30 +01:00
|
|
|
extern cairo_user_data_key_t fiv_io_key_exif;
|
|
|
|
/// FivIoOrientation, as a uintptr_t.
|
|
|
|
extern cairo_user_data_key_t fiv_io_key_orientation;
|
2021-11-25 02:46:36 +01:00
|
|
|
/// GBytes with plain ICC profile data.
|
2021-12-18 06:38:30 +01:00
|
|
|
extern cairo_user_data_key_t fiv_io_key_icc;
|
2021-12-16 00:28:37 +01:00
|
|
|
/// GBytes with plain XMP data.
|
2021-12-18 06:38:30 +01:00
|
|
|
extern cairo_user_data_key_t fiv_io_key_xmp;
|
2021-12-28 19:37:15 +01:00
|
|
|
/// GBytes with a WebP's THUM chunk, used for our thumbnails.
|
|
|
|
extern cairo_user_data_key_t fiv_io_key_thum;
|
2022-06-04 22:47:44 +02:00
|
|
|
/// GHashTable with key-value pairs from PNG's tEXt, zTXt, iTXt chunks.
|
|
|
|
/// Currently only read by fiv_io_open_png_thumbnail().
|
|
|
|
extern cairo_user_data_key_t fiv_io_key_text;
|
2021-11-25 02:46:36 +01:00
|
|
|
|
|
|
|
/// The next frame in a sequence, as a surface, in a chain, pre-composited.
|
|
|
|
/// There is no wrap-around.
|
2021-12-18 06:38:30 +01:00
|
|
|
extern cairo_user_data_key_t fiv_io_key_frame_next;
|
2021-11-26 20:54:08 +01:00
|
|
|
/// The previous frame in a sequence, as a surface, in a chain, pre-composited.
|
|
|
|
/// This is a weak pointer that wraps around, and needn't be present
|
|
|
|
/// for static images.
|
2021-12-18 06:38:30 +01:00
|
|
|
extern cairo_user_data_key_t fiv_io_key_frame_previous;
|
2021-11-25 02:46:36 +01:00
|
|
|
/// Frame duration in milliseconds as an intptr_t.
|
2021-12-18 06:38:30 +01:00
|
|
|
extern cairo_user_data_key_t fiv_io_key_frame_duration;
|
2021-11-26 00:46:00 +01:00
|
|
|
/// How many times to repeat the animation, or zero for +inf, as a uintptr_t.
|
2021-12-18 06:38:30 +01:00
|
|
|
extern cairo_user_data_key_t fiv_io_key_loops;
|
2021-11-25 02:46:36 +01:00
|
|
|
|
2021-11-27 02:49:26 +01:00
|
|
|
/// The first frame of the next page, as a surface, in a chain.
|
|
|
|
/// There is no wrap-around.
|
2021-12-18 06:38:30 +01:00
|
|
|
extern cairo_user_data_key_t fiv_io_key_page_next;
|
2021-11-27 02:49:26 +01:00
|
|
|
/// The first frame of the previous page, as a surface, in a chain.
|
|
|
|
/// There is no wrap-around. This is a weak pointer.
|
2021-12-18 06:38:30 +01:00
|
|
|
extern cairo_user_data_key_t fiv_io_key_page_previous;
|
2021-11-27 02:49:26 +01:00
|
|
|
|
2022-01-22 23:17:17 +01:00
|
|
|
typedef struct _FivIoRenderClosure {
|
2022-01-24 04:03:19 +01:00
|
|
|
/// The rendering is allowed to fail, returning NULL.
|
2022-01-22 23:17:17 +01:00
|
|
|
cairo_surface_t *(*render)(struct _FivIoRenderClosure *, double scale);
|
|
|
|
} FivIoRenderClosure;
|
|
|
|
|
|
|
|
/// A FivIoRenderClosure for parametrized re-rendering of vector formats.
|
|
|
|
/// This is attached at the page level.
|
|
|
|
/// The rendered image will not have this key.
|
|
|
|
extern cairo_user_data_key_t fiv_io_key_render;
|
|
|
|
|
2022-01-24 04:03:19 +01:00
|
|
|
typedef struct {
|
|
|
|
const char *uri; ///< Source URI
|
|
|
|
FivIoProfile screen_profile; ///< Target colour space or NULL
|
|
|
|
int screen_dpi; ///< Target DPI
|
|
|
|
gboolean enhance; ///< Enhance JPEG (currently)
|
|
|
|
gboolean first_frame_only; ///< Only interested in the 1st frame
|
|
|
|
GPtrArray *warnings; ///< String vector for non-fatal errors
|
|
|
|
} FivIoOpenContext;
|
|
|
|
|
|
|
|
cairo_surface_t *fiv_io_open(const FivIoOpenContext *ctx, GError **error);
|
|
|
|
cairo_surface_t *fiv_io_open_from_data(
|
|
|
|
const char *data, size_t len, const FivIoOpenContext *ctx, GError **error);
|
2022-06-04 22:47:44 +02:00
|
|
|
cairo_surface_t *fiv_io_open_png_thumbnail(const char *path, GError **error);
|
2021-11-25 02:46:36 +01:00
|
|
|
|
2022-02-20 19:43:21 +01:00
|
|
|
// --- Thumbnail passing utilities ---------------------------------------------
|
|
|
|
|
2022-06-06 15:22:23 +02:00
|
|
|
enum { FIV_IO_SERIALIZE_LOW_QUALITY = 1 << 0 };
|
|
|
|
|
|
|
|
void fiv_io_serialize_to_stdout(cairo_surface_t *surface, guint64 user_data);
|
|
|
|
cairo_surface_t *fiv_io_deserialize(GBytes *bytes, guint64 *user_data);
|
2022-02-20 19:43:21 +01:00
|
|
|
|
2021-12-31 02:19:17 +01:00
|
|
|
// --- Filesystem --------------------------------------------------------------
|
|
|
|
|
2022-01-05 08:26:28 +01:00
|
|
|
typedef enum _FivIoModelSort {
|
|
|
|
FIV_IO_MODEL_SORT_NAME,
|
|
|
|
FIV_IO_MODEL_SORT_MTIME,
|
|
|
|
FIV_IO_MODEL_SORT_COUNT,
|
|
|
|
|
|
|
|
FIV_IO_MODEL_SORT_MIN = 0,
|
|
|
|
FIV_IO_MODEL_SORT_MAX = FIV_IO_MODEL_SORT_COUNT - 1
|
|
|
|
} FivIoModelSort;
|
|
|
|
|
2021-12-31 02:19:17 +01:00
|
|
|
#define FIV_TYPE_IO_MODEL (fiv_io_model_get_type())
|
|
|
|
G_DECLARE_FINAL_TYPE(FivIoModel, fiv_io_model, FIV, IO_MODEL, GObject)
|
|
|
|
|
|
|
|
/// Loads a directory. Clears itself even on failure.
|
|
|
|
gboolean fiv_io_model_open(FivIoModel *self, GFile *directory, GError **error);
|
|
|
|
|
|
|
|
/// Returns the current location as a GFile.
|
|
|
|
/// There is no ownership transfer, and the object may be NULL.
|
|
|
|
GFile *fiv_io_model_get_location(FivIoModel *self);
|
|
|
|
|
2022-06-04 01:19:56 +02:00
|
|
|
typedef struct {
|
2022-06-05 13:29:38 +02:00
|
|
|
gchar *uri; ///< GIO URI
|
Support opening collections of files
Implement a process-local VFS to enable grouping together arbitrary
URIs passed via program arguments, DnD, or the file open dialog.
This VFS contains FivCollectionFile objects, which act as "simple"
proxies over arbitrary GFiles. Their true URIs may be retrieved
through the "standard::target-uri" attribute, in a similar way to
GVfs's "recent" and "trash" backends.
(The main reason we proxy rather than just hackishly return foreign
GFiles from the VFS is that loading them would switch the current
directory, and break iteration as a result.
We could also keep the collection outside of GVfs, but that would
result in considerable special-casing, and the author wouldn't gain
intimate knowledge of GIO.)
There is no perceived need to keep old collections when opening
new ones, so we simply change and reload the contents when needed.
Similarly, there is no intention to make the VFS writeable.
The process-locality of this and other URI schemes has proven to be
rather annoying when passing files to other applications,
however most of the resulting complexity appears to be essential
rather than accidental.
Note that the GTK+ file chooser widget is retarded, and doesn't
recognize URIs that lack the authority part in the location bar.
2022-07-28 00:37:36 +02:00
|
|
|
gchar *target_uri; ///< GIO URI for any target
|
2022-06-05 13:29:38 +02:00
|
|
|
gchar *collate_key; ///< Collate key for the filename
|
2022-06-04 01:19:56 +02:00
|
|
|
gint64 mtime_msec; ///< Modification time in milliseconds
|
|
|
|
} FivIoModelEntry;
|
|
|
|
|
|
|
|
const FivIoModelEntry *fiv_io_model_get_files(FivIoModel *self, gsize *len);
|
|
|
|
const FivIoModelEntry *fiv_io_model_get_subdirs(FivIoModel *self, gsize *len);
|
2021-11-25 02:46:36 +01:00
|
|
|
|
2021-12-14 05:03:55 +01:00
|
|
|
// --- Export ------------------------------------------------------------------
|
|
|
|
|
2021-12-28 18:51:00 +01:00
|
|
|
/// Encodes a Cairo surface as a WebP bitstream, following the configuration.
|
|
|
|
/// The result needs to be freed using WebPFree/WebPDataClear().
|
|
|
|
unsigned char *fiv_io_encode_webp(
|
|
|
|
cairo_surface_t *surface, const WebPConfig *config, size_t *len);
|
|
|
|
|
2021-12-26 19:41:42 +01:00
|
|
|
/// Saves the page as a lossless WebP still picture or animation.
|
2021-12-14 05:03:55 +01:00
|
|
|
/// If no exact frame is specified, this potentially creates an animation.
|
2021-12-18 06:38:30 +01:00
|
|
|
gboolean fiv_io_save(cairo_surface_t *page, cairo_surface_t *frame,
|
2022-06-05 13:29:38 +02:00
|
|
|
FivIoProfile target, const char *path, GError **error);
|
2021-12-14 05:03:55 +01:00
|
|
|
|
2021-11-26 00:16:12 +01:00
|
|
|
// --- Metadata ----------------------------------------------------------------
|
|
|
|
|
|
|
|
// https://www.cipa.jp/std/documents/e/DC-008-2012_E.pdf Table 6
|
2021-12-18 06:38:30 +01:00
|
|
|
typedef enum _FivIoOrientation {
|
|
|
|
FivIoOrientationUnknown = 0,
|
|
|
|
FivIoOrientation0 = 1,
|
|
|
|
FivIoOrientationMirror0 = 2,
|
|
|
|
FivIoOrientation180 = 3,
|
|
|
|
FivIoOrientationMirror180 = 4,
|
|
|
|
FivIoOrientationMirror270 = 5,
|
|
|
|
FivIoOrientation90 = 6,
|
|
|
|
FivIoOrientationMirror90 = 7,
|
|
|
|
FivIoOrientation270 = 8
|
|
|
|
} FivIoOrientation;
|
|
|
|
|
2022-01-07 08:50:07 +01:00
|
|
|
/// Returns a rendering matrix for a surface, and its target dimensions.
|
|
|
|
cairo_matrix_t fiv_io_orientation_apply(cairo_surface_t *surface,
|
|
|
|
FivIoOrientation orientation, double *width, double *height);
|
|
|
|
void fiv_io_orientation_dimensions(cairo_surface_t *surface,
|
|
|
|
FivIoOrientation orientation, double *width, double *height);
|
2021-12-28 23:10:45 +01:00
|
|
|
|
|
|
|
/// Extracts the orientation field from Exif, if there's any.
|
2021-12-18 06:38:30 +01:00
|
|
|
FivIoOrientation fiv_io_exif_orientation(const guint8 *exif, gsize len);
|
2021-11-26 00:16:12 +01:00
|
|
|
|
2021-12-14 05:03:55 +01:00
|
|
|
/// Save metadata attached by this module in Exiv2 format.
|
2021-12-18 06:38:30 +01:00
|
|
|
gboolean fiv_io_save_metadata(
|
2022-06-05 13:29:38 +02:00
|
|
|
cairo_surface_t *page, const char *path, GError **error);
|