mirror of
https://github.com/aseprite/aseprite.git
synced 2025-04-03 07:20:46 +00:00
Add support for animated webp files (fix #273)
This commit is contained in:
parent
b1823ab558
commit
6de103128b
@ -65,7 +65,6 @@ option(USE_SHARED_JPEGLIB "Use your installed copy of jpeglib" off)
|
|||||||
option(USE_SHARED_ZLIB "Use your installed copy of zlib" off)
|
option(USE_SHARED_ZLIB "Use your installed copy of zlib" off)
|
||||||
option(USE_SHARED_LIBPNG "Use your installed copy of libpng" off)
|
option(USE_SHARED_LIBPNG "Use your installed copy of libpng" off)
|
||||||
option(USE_SHARED_LIBLOADPNG "Use your installed copy of libloadpng" off)
|
option(USE_SHARED_LIBLOADPNG "Use your installed copy of libloadpng" off)
|
||||||
option(USE_SHARED_LIBWEBP "Use your installed copy of libwebp" off)
|
|
||||||
option(USE_SHARED_TINYXML "Use your installed copy of tinyxml" off)
|
option(USE_SHARED_TINYXML "Use your installed copy of tinyxml" off)
|
||||||
option(USE_SHARED_PIXMAN "Use your installed copy of pixman" off)
|
option(USE_SHARED_PIXMAN "Use your installed copy of pixman" off)
|
||||||
option(USE_SHARED_FREETYPE "Use shared FreeType library" off)
|
option(USE_SHARED_FREETYPE "Use shared FreeType library" off)
|
||||||
@ -221,21 +220,8 @@ add_definitions(-DPNG_NO_MMX_CODE) # Do not use MMX optimizations in PNG code
|
|||||||
|
|
||||||
# libwebp
|
# libwebp
|
||||||
if(WITH_WEBP_SUPPORT)
|
if(WITH_WEBP_SUPPORT)
|
||||||
if(USE_SHARED_LIBWEBP)
|
set(WEBP_LIBRARIES webp webpdemux webpmux)
|
||||||
find_package(PkgConfig)
|
set(WEBP_INCLUDE_DIR ${LIBWEBP_DIR}/src)
|
||||||
pkg_check_modules(WEBP libwebp)
|
|
||||||
if(NOT WEBP_FOUND)
|
|
||||||
message(FATAL_ERROR "libwebp not found")
|
|
||||||
endif()
|
|
||||||
else()
|
|
||||||
# Skia already includes webp library
|
|
||||||
if(NOT USE_SKIA_BACKEND)
|
|
||||||
set(WEBP_LIBRARIES webp)
|
|
||||||
else()
|
|
||||||
set(WEBP_LIBRARIES "")
|
|
||||||
endif()
|
|
||||||
set(WEBP_INCLUDE_DIR ${LIBWEBP_DIR}/src)
|
|
||||||
endif()
|
|
||||||
include_directories(${WEBP_INCLUDE_DIR})
|
include_directories(${WEBP_INCLUDE_DIR})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
@ -276,6 +276,15 @@
|
|||||||
<option id="show_alert" type="bool" default="true" />
|
<option id="show_alert" type="bool" default="true" />
|
||||||
<option id="quality" type="double" default="1.0" migrate="JPEG.Quality" />
|
<option id="quality" type="double" default="1.0" migrate="JPEG.Quality" />
|
||||||
</section>
|
</section>
|
||||||
|
<section id="webp">
|
||||||
|
<option id="show_alert" type="bool" default="true" />
|
||||||
|
<option id="loop" type="bool" default="true" />
|
||||||
|
<option id="type" type="int" default="0" />
|
||||||
|
<option id="quality" type="int" default="100" migrate="WEBP.Quality" />
|
||||||
|
<option id="compression" type="int" default="6" migrate="WEBP.Compression" />
|
||||||
|
<option id="image_hint" type="int" default="0" migrate="WEBP.ImageHint" />
|
||||||
|
<option id="image_preset" type="int" default="0" migrate="WEBP.ImagePreset" />
|
||||||
|
</section>
|
||||||
<section id="hue_saturation">
|
<section id="hue_saturation">
|
||||||
<option id="mode" type="HueSaturationMode" default="HueSaturationMode::HSL" />
|
<option id="mode" type="HueSaturationMode" default="HueSaturationMode::HSL" />
|
||||||
</section>
|
</section>
|
||||||
|
@ -1199,8 +1199,10 @@ color = Color:
|
|||||||
[webp_options]
|
[webp_options]
|
||||||
title = WebP Options
|
title = WebP Options
|
||||||
save_as = Save as:
|
save_as = Save as:
|
||||||
|
animation_loop = Animation &Loop
|
||||||
|
type = Type:
|
||||||
|
simple_webp = Simple: Good Lossless Compression
|
||||||
lossless_webp = Lossless WebP
|
lossless_webp = Lossless WebP
|
||||||
lossless_webp_tooltip = Save in simple WebP lossless format.
|
|
||||||
compression = Compression:
|
compression = Compression:
|
||||||
image_hint = Image Hint:
|
image_hint = Image Hint:
|
||||||
image_hint_default = Default
|
image_hint_default = Default
|
||||||
@ -1208,7 +1210,6 @@ image_hint_picture = Picture
|
|||||||
image_hint_photo = Photo
|
image_hint_photo = Photo
|
||||||
image_hint_graph = Graph
|
image_hint_graph = Graph
|
||||||
lossy_webp = Lossy WebP
|
lossy_webp = Lossy WebP
|
||||||
lossy_webp_tooltip = Save in simple WebP lossy format.
|
|
||||||
quality = Quality:
|
quality = Quality:
|
||||||
image_preset = Image Preset:
|
image_preset = Image Preset:
|
||||||
image_preset_default = Default
|
image_preset_default = Default
|
||||||
|
@ -1,41 +1,52 @@
|
|||||||
<!-- Aseprite -->
|
<!-- Aseprite -->
|
||||||
|
<!-- Copyright (C) 2016-2018 by David Capello -->
|
||||||
<!-- Copyright (C) 2015 by Gabriel Rauter -->
|
<!-- Copyright (C) 2015 by Gabriel Rauter -->
|
||||||
<!-- Copyright (C) 2016 by David Capello -->
|
|
||||||
<gui>
|
<gui>
|
||||||
<window id="webp_options" text="@.title">
|
<window id="webp_options" text="@.title">
|
||||||
<vbox>
|
<vbox>
|
||||||
<label text="@.save_as" />
|
<label text="@.save_as" />
|
||||||
<radio group="1" text="@.lossless_webp" id="lossless" tooltip="@.lossless_webp_tooltip" />
|
<check text="@.animation_loop" id="loop" />
|
||||||
<hbox>
|
<hbox>
|
||||||
<label width="55" text="@.compression" />
|
<label text="@.type" />
|
||||||
<slider min="0" max="9" id="compression" cell_align="horizontal" width="128" />
|
<combobox id="type" cell_align="horizontal" cell_hspan="2">
|
||||||
</hbox>
|
<listitem text="@.simple_webp" value="0" />
|
||||||
<hbox>
|
<listitem text="@.lossless_webp" value="1" />
|
||||||
<label width="55" text="@.image_hint" />
|
<listitem text="@.lossy_webp" value="2" />
|
||||||
<combobox width="128" id="image_hint">
|
|
||||||
<listitem text="@.image_hint_default" value="0" />
|
|
||||||
<listitem text="@.image_hint_picture" value="1" />
|
|
||||||
<listitem text="@.image_hint_photo" value="2" />
|
|
||||||
<listitem text="@.image_hint_graph" value="3" />
|
|
||||||
</combobox>
|
</combobox>
|
||||||
</hbox>
|
</hbox>
|
||||||
|
<vbox id="lossless_options">
|
||||||
|
<hbox>
|
||||||
|
<label width="55" text="@.compression" />
|
||||||
|
<slider min="0" max="9" id="compression" cell_align="horizontal" width="128" />
|
||||||
|
</hbox>
|
||||||
|
<hbox>
|
||||||
|
<label width="55" text="@.image_hint" />
|
||||||
|
<combobox width="128" id="image_hint">
|
||||||
|
<listitem text="@.image_hint_default" value="0" />
|
||||||
|
<listitem text="@.image_hint_picture" value="1" />
|
||||||
|
<listitem text="@.image_hint_photo" value="2" />
|
||||||
|
<listitem text="@.image_hint_graph" value="3" />
|
||||||
|
</combobox>
|
||||||
|
</hbox>
|
||||||
|
</vbox>
|
||||||
|
<vbox id="lossy_options">
|
||||||
|
<hbox>
|
||||||
|
<label width="55" text="@.quality" />
|
||||||
|
<slider min="0" max="100" id="quality" cell_align="horizontal" width="128" />
|
||||||
|
</hbox>
|
||||||
|
<hbox>
|
||||||
|
<label width="55" text="@.image_preset" />
|
||||||
|
<combobox width="128" id="image_preset">
|
||||||
|
<listitem text="@.image_preset_default" value="0" />
|
||||||
|
<listitem text="@.image_preset_picture" value="1" />
|
||||||
|
<listitem text="@.image_preset_photo" value="2" />
|
||||||
|
<listitem text="@.image_preset_drawing" value="3" />
|
||||||
|
<listitem text="@.image_preset_icon" value="4" />
|
||||||
|
<listitem text="@.image_preset_text" value="5" />
|
||||||
|
</combobox>
|
||||||
|
</hbox>
|
||||||
|
</vbox>
|
||||||
<separator horizontal="true" />
|
<separator horizontal="true" />
|
||||||
<radio group="1" text="@.lossy_webp" id="lossy" tooltip="@.lossy_webp_tooltip" />
|
|
||||||
<hbox>
|
|
||||||
<label width="55" text="@.quality" />
|
|
||||||
<slider min="0" max="100" id="quality" cell_align="horizontal" width="128" />
|
|
||||||
</hbox>
|
|
||||||
<hbox>
|
|
||||||
<label width="55" text="@.image_preset" />
|
|
||||||
<combobox width="128" id="image_preset">
|
|
||||||
<listitem text="@.image_preset_default" value="0" />
|
|
||||||
<listitem text="@.image_preset_picture" value="1" />
|
|
||||||
<listitem text="@.image_preset_photo" value="2" />
|
|
||||||
<listitem text="@.image_preset_drawing" value="3" />
|
|
||||||
<listitem text="@.image_preset_icon" value="4" />
|
|
||||||
<listitem text="@.image_preset_text" value="5" />
|
|
||||||
</combobox>
|
|
||||||
</hbox>
|
|
||||||
<hbox>
|
<hbox>
|
||||||
<boxfiller />
|
<boxfiller />
|
||||||
<hbox homogeneous="true">
|
<hbox homogeneous="true">
|
||||||
|
@ -18,9 +18,12 @@
|
|||||||
#include "app/file/format_options.h"
|
#include "app/file/format_options.h"
|
||||||
#include "app/file/webp_options.h"
|
#include "app/file/webp_options.h"
|
||||||
#include "app/ini_file.h"
|
#include "app/ini_file.h"
|
||||||
#include "base/file_handle.h"
|
#include "app/pref/preferences.h"
|
||||||
|
#include "base/bind.h"
|
||||||
#include "base/convert_to.h"
|
#include "base/convert_to.h"
|
||||||
|
#include "base/file_handle.h"
|
||||||
#include "doc/doc.h"
|
#include "doc/doc.h"
|
||||||
|
#include "render/render.h"
|
||||||
|
|
||||||
#include "webp_options.xml.h"
|
#include "webp_options.xml.h"
|
||||||
|
|
||||||
@ -29,9 +32,8 @@
|
|||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
// Include webp libraries
|
#include <webp/demux.h>
|
||||||
#include <webp/decode.h>
|
#include <webp/mux.h>
|
||||||
#include <webp/encode.h>
|
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
|
|
||||||
@ -57,7 +59,7 @@ class WebPFormat : public FileFormat {
|
|||||||
FILE_SUPPORT_SAVE |
|
FILE_SUPPORT_SAVE |
|
||||||
FILE_SUPPORT_RGB |
|
FILE_SUPPORT_RGB |
|
||||||
FILE_SUPPORT_RGBA |
|
FILE_SUPPORT_RGBA |
|
||||||
FILE_SUPPORT_SEQUENCES |
|
FILE_SUPPORT_FRAMES |
|
||||||
FILE_SUPPORT_GET_FORMAT_OPTIONS;
|
FILE_SUPPORT_GET_FORMAT_OPTIONS;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -65,8 +67,7 @@ class WebPFormat : public FileFormat {
|
|||||||
#ifdef ENABLE_SAVE
|
#ifdef ENABLE_SAVE
|
||||||
bool onSave(FileOp* fop) override;
|
bool onSave(FileOp* fop) override;
|
||||||
#endif
|
#endif
|
||||||
|
base::SharedPtr<FormatOptions> onGetFormatOptions(FileOp* fop) override;
|
||||||
base::SharedPtr<FormatOptions> onGetFormatOptions(FileOp* fop) override;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
FileFormat* CreateWebPFormat()
|
FileFormat* CreateWebPFormat()
|
||||||
@ -106,71 +107,118 @@ bool WebPFormat::onLoad(FileOp* fop)
|
|||||||
}
|
}
|
||||||
|
|
||||||
std::vector<uint8_t> buf(len);
|
std::vector<uint8_t> buf(len);
|
||||||
uint8_t* data = &buf.front();
|
if (fread(&buf[0], 1, buf.size(), fp) != buf.size()) {
|
||||||
|
|
||||||
if (!fread(data, sizeof(uint8_t), len, fp)) {
|
|
||||||
fop->setError("Error moving the whole WebP file to memory\n");
|
fop->setError("Error moving the whole WebP file to memory\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
WebPData webp_data;
|
||||||
|
WebPDataInit(&webp_data);
|
||||||
|
webp_data.bytes = &buf[0];
|
||||||
|
webp_data.size = buf.size();
|
||||||
|
|
||||||
|
WebPAnimDecoderOptions dec_options;
|
||||||
|
WebPAnimDecoderOptionsInit(&dec_options);
|
||||||
|
dec_options.color_mode = MODE_RGBA;
|
||||||
|
|
||||||
|
WebPAnimDecoder* dec = WebPAnimDecoderNew(&webp_data, &dec_options);
|
||||||
|
if (dec == nullptr) {
|
||||||
|
fop->setError("Error parsing WebP image\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
WebPAnimInfo anim_info;
|
||||||
|
if (!WebPAnimDecoderGetInfo(dec, &anim_info)) {
|
||||||
|
fop->setError("Error getting global info about the WebP animation\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
WebPDecoderConfig config;
|
WebPDecoderConfig config;
|
||||||
if (!WebPInitDecoderConfig(&config)) {
|
WebPInitDecoderConfig(&config);
|
||||||
fop->setError("WebP decoder cannot load this webp file version\n");
|
if (WebPGetFeatures(webp_data.bytes, webp_data.size, &config.input)) {
|
||||||
return false;
|
if (!fop->formatOptions()) {
|
||||||
}
|
base::SharedPtr<WebPOptions> opts(new WebPOptions());
|
||||||
|
WebPOptions::Type type = WebPOptions::Simple;
|
||||||
if (WebPGetFeatures(data, len, &config.input) != VP8_STATUS_OK) {
|
switch (config.input.format) {
|
||||||
fop->setError("Bad bitstream in WebP file\n");
|
case 0: type = WebPOptions::Simple; break;
|
||||||
return false;
|
case 1: type = WebPOptions::Lossy; break;
|
||||||
}
|
case 2: type = WebPOptions::Lossless; break;
|
||||||
|
}
|
||||||
fop->sequenceSetHasAlpha(config.input.has_alpha != 0);
|
opts->setType(type);
|
||||||
|
fop->setFormatOptions(opts);
|
||||||
Image* image = fop->sequenceImage(IMAGE_RGB, config.input.width, config.input.height);
|
|
||||||
|
|
||||||
config.output.colorspace = MODE_RGBA;
|
|
||||||
config.output.u.RGBA.rgba = (uint8_t*)image->getPixelAddress(0, 0);
|
|
||||||
config.output.u.RGBA.stride = config.input.width * sizeof(uint32_t);
|
|
||||||
config.output.u.RGBA.size = config.input.width * config.input.height * sizeof(uint32_t);
|
|
||||||
config.output.is_external_memory = 1;
|
|
||||||
|
|
||||||
WebPIDecoder* idec = WebPIDecode(NULL, 0, &config);
|
|
||||||
if (idec == NULL) {
|
|
||||||
fop->setError("Error creating WebP decoder\n");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
long bytes_remaining = len;
|
|
||||||
long bytes_read = std::max(4l, len/100l);
|
|
||||||
|
|
||||||
while (bytes_remaining > 0) {
|
|
||||||
VP8StatusCode status = WebPIAppend(idec, data, bytes_read);
|
|
||||||
if (status == VP8_STATUS_OK ||
|
|
||||||
status == VP8_STATUS_SUSPENDED) {
|
|
||||||
bytes_remaining -= bytes_read;
|
|
||||||
data += bytes_read;
|
|
||||||
if (bytes_remaining < bytes_read)
|
|
||||||
bytes_read = bytes_remaining;
|
|
||||||
fop->setProgress(1.0f - ((float)std::max(bytes_remaining, 0l)/(float)len));
|
|
||||||
}
|
}
|
||||||
else {
|
}
|
||||||
fop->setError("Error decoding WebP data: %s\n", getDecoderErrorMessage(status));
|
else {
|
||||||
WebPIDelete(idec);
|
config.input.has_alpha = false;
|
||||||
WebPFreeDecBuffer(&config.output);
|
}
|
||||||
|
|
||||||
|
const int w = anim_info.canvas_width;
|
||||||
|
const int h = anim_info.canvas_height;
|
||||||
|
|
||||||
|
Sprite* sprite = new Sprite(IMAGE_RGB, w, h, 256);
|
||||||
|
LayerImage* layer = new LayerImage(sprite);
|
||||||
|
sprite->root()->addLayer(layer);
|
||||||
|
sprite->setTotalFrames(anim_info.frame_count);
|
||||||
|
|
||||||
|
for (frame_t f=0; f<anim_info.frame_count; ++f) {
|
||||||
|
ImageRef image(Image::create(IMAGE_RGB, w, h));
|
||||||
|
Cel* cel = new Cel(f, image);
|
||||||
|
layer->addCel(cel);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool has_alpha = config.input.has_alpha;
|
||||||
|
frame_t f = 0;
|
||||||
|
int prev_timestamp = 0;
|
||||||
|
while (WebPAnimDecoderHasMoreFrames(dec)) {
|
||||||
|
uint8_t* frame_rgba;
|
||||||
|
int frame_timestamp = 0;
|
||||||
|
if (!WebPAnimDecoderGetNext(dec, &frame_rgba, &frame_timestamp)) {
|
||||||
|
fop->setError("Error loading WebP frame\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Cel* cel = layer->cel(f);
|
||||||
|
if (cel) {
|
||||||
|
memcpy(cel->image()->getPixelAddress(0, 0),
|
||||||
|
frame_rgba, h*w*sizeof(uint32_t));
|
||||||
|
|
||||||
|
if (!has_alpha) {
|
||||||
|
const uint32_t* src = (const uint32_t*)frame_rgba;
|
||||||
|
const uint32_t* src_end = src + w*h;
|
||||||
|
while (src < src_end) {
|
||||||
|
const uint8_t alpha = (*src >> 24) & 0xff;
|
||||||
|
if (alpha < 255) {
|
||||||
|
has_alpha = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
++src;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sprite->setFrameDuration(f, frame_timestamp - prev_timestamp);
|
||||||
|
|
||||||
|
prev_timestamp = frame_timestamp;
|
||||||
|
fop->setProgress(double(f) / double(anim_info.frame_count));
|
||||||
if (fop->isStop())
|
if (fop->isStop())
|
||||||
break;
|
break;
|
||||||
}
|
|
||||||
|
|
||||||
if (!fop->formatOptions()) {
|
++f;
|
||||||
base::SharedPtr<WebPOptions> webPOptions(new WebPOptions());
|
|
||||||
webPOptions->setLossless(std::min(config.input.format - 1, 1));
|
|
||||||
fop->setFormatOptions(webPOptions);
|
|
||||||
}
|
}
|
||||||
|
WebPAnimDecoderReset(dec);
|
||||||
|
|
||||||
WebPIDelete(idec);
|
if (!has_alpha)
|
||||||
WebPFreeDecBuffer(&config.output);
|
layer->configureAsBackground();
|
||||||
|
|
||||||
|
WebPAnimDecoderDelete(dec);
|
||||||
|
|
||||||
|
// Don't use WebPDataClear because webp_data use a std::vector<> data.
|
||||||
|
//WebPDataClear(&webp_data);
|
||||||
|
|
||||||
|
if (fop->isStop())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
fop->createDocument(sprite);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -179,145 +227,138 @@ bool WebPFormat::onLoad(FileOp* fop)
|
|||||||
struct WriterData {
|
struct WriterData {
|
||||||
FILE* fp;
|
FILE* fp;
|
||||||
FileOp* fop;
|
FileOp* fop;
|
||||||
|
frame_t f, n;
|
||||||
|
double progress;
|
||||||
|
|
||||||
|
WriterData(FILE* fp, FileOp* fop, frame_t f, frame_t n, double progress)
|
||||||
|
: fp(fp), fop(fop), f(f), n(n), progress(progress) { }
|
||||||
};
|
};
|
||||||
|
|
||||||
class ScopedWebPPicture {
|
static int progress_report(int percent, const WebPPicture* pic)
|
||||||
public:
|
|
||||||
ScopedWebPPicture(WebPPicture& pic) : m_pic(pic) {
|
|
||||||
}
|
|
||||||
~ScopedWebPPicture() {
|
|
||||||
WebPPictureFree(&m_pic);
|
|
||||||
}
|
|
||||||
private:
|
|
||||||
WebPPicture& m_pic;
|
|
||||||
};
|
|
||||||
|
|
||||||
const char* getEncoderErrorMessage(WebPEncodingError errorCode) {
|
|
||||||
switch (errorCode) {
|
|
||||||
case VP8_ENC_OK: return ""; break;
|
|
||||||
case VP8_ENC_ERROR_OUT_OF_MEMORY: return "memory error allocating objects"; break;
|
|
||||||
case VP8_ENC_ERROR_BITSTREAM_OUT_OF_MEMORY: return "memory error while flushing bits"; break;
|
|
||||||
case VP8_ENC_ERROR_NULL_PARAMETER: return "a pointer parameter is NULL"; break;
|
|
||||||
case VP8_ENC_ERROR_INVALID_CONFIGURATION: return "configuration is invalid"; break;
|
|
||||||
case VP8_ENC_ERROR_BAD_DIMENSION: return "picture has invalid width/height"; break;
|
|
||||||
case VP8_ENC_ERROR_PARTITION0_OVERFLOW: return "partition is bigger than 512k"; break;
|
|
||||||
case VP8_ENC_ERROR_PARTITION_OVERFLOW: return "partition is bigger than 16M"; break;
|
|
||||||
case VP8_ENC_ERROR_BAD_WRITE: return "error while flushing bytes"; break;
|
|
||||||
case VP8_ENC_ERROR_FILE_TOO_BIG: return "file is bigger than 4G"; break;
|
|
||||||
case VP8_ENC_ERROR_USER_ABORT: return "abort request by user"; break;
|
|
||||||
case VP8_ENC_ERROR_LAST: return "abort request by user"; break;
|
|
||||||
default: return ""; break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#if WEBP_ENCODER_ABI_VERSION < 0x0203
|
|
||||||
#define MAX_LEVEL 9
|
|
||||||
// Mapping between -z level and -m / -q parameter settings.
|
|
||||||
static const struct {
|
|
||||||
uint8_t method_;
|
|
||||||
uint8_t quality_;
|
|
||||||
} kLosslessPresets[MAX_LEVEL + 1] = {
|
|
||||||
{ 0, 0 }, { 1, 20 }, { 2, 25 }, { 3, 30 }, { 3, 50 },
|
|
||||||
{ 4, 50 }, { 4, 75 }, { 4, 90 }, { 5, 90 }, { 6, 100 }
|
|
||||||
};
|
|
||||||
int WebPConfigLosslessPreset(WebPConfig* config, int level) {
|
|
||||||
if (config == NULL || level < 0 || level > MAX_LEVEL) return 0;
|
|
||||||
config->lossless = 1;
|
|
||||||
config->method = kLosslessPresets[level].method_;
|
|
||||||
config->quality = kLosslessPresets[level].quality_;
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
static int ProgressReport(int percent, const WebPPicture* const pic)
|
|
||||||
{
|
{
|
||||||
FileOp* fop = ((WriterData*)pic->custom_ptr)->fop;
|
auto wd = (WriterData*)pic->user_data;
|
||||||
fop->setProgress((double)percent/(double)100);
|
FileOp* fop = wd->fop;
|
||||||
|
|
||||||
|
double newProgress = (double(wd->f) + double(percent)/100.0) / double(wd->n);
|
||||||
|
wd->progress = MAX(wd->progress, newProgress);
|
||||||
|
wd->progress = MID(0.0, wd->progress, 1.0);
|
||||||
|
|
||||||
|
fop->setProgress(wd->progress);
|
||||||
if (fop->isStop())
|
if (fop->isStop())
|
||||||
return false;
|
return false;
|
||||||
else
|
else
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int FileWriter(const uint8_t* data, size_t data_size, const WebPPicture* const pic)
|
|
||||||
{
|
|
||||||
return (data_size ? (fwrite(data, data_size, 1, ((WriterData*)pic->custom_ptr)->fp) == 1) : 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool WebPFormat::onSave(FileOp* fop)
|
bool WebPFormat::onSave(FileOp* fop)
|
||||||
{
|
{
|
||||||
FileHandle handle(open_file_with_exception_sync_on_close(fop->filename(), "wb"));
|
FileHandle handle(open_file_with_exception_sync_on_close(fop->filename(), "wb"));
|
||||||
FILE* fp = handle.get();
|
FILE* fp = handle.get();
|
||||||
|
|
||||||
WriterData wd = { fp, fop };
|
const Sprite* sprite = fop->document()->sprite();
|
||||||
|
const int w = sprite->width();
|
||||||
|
const int h = sprite->height();
|
||||||
|
|
||||||
const Image* image = fop->sequenceImage();
|
if (w > WEBP_MAX_DIMENSION ||
|
||||||
if (image->width() > WEBP_MAX_DIMENSION ||
|
h > WEBP_MAX_DIMENSION) {
|
||||||
image->height() > WEBP_MAX_DIMENSION) {
|
|
||||||
fop->setError("WebP format cannot store %dx%d images. The maximum allowed size is %dx%d\n",
|
fop->setError("WebP format cannot store %dx%d images. The maximum allowed size is %dx%d\n",
|
||||||
image->width(), image->height(),
|
w, h, WEBP_MAX_DIMENSION, WEBP_MAX_DIMENSION);
|
||||||
WEBP_MAX_DIMENSION, WEBP_MAX_DIMENSION);
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
base::SharedPtr<WebPOptions> webp_options = fop->formatOptions();
|
base::SharedPtr<WebPOptions> opts = fop->formatOptions();
|
||||||
|
|
||||||
WebPConfig config;
|
WebPConfig config;
|
||||||
|
WebPConfigInit(&config);
|
||||||
|
|
||||||
if (webp_options->lossless()) {
|
switch (opts->type()) {
|
||||||
if (!(WebPConfigInit(&config) &&
|
|
||||||
WebPConfigLosslessPreset(&config, webp_options->getMethod()))) {
|
case WebPOptions::Simple:
|
||||||
fop->setError("Error in WebP configuration\n");
|
case WebPOptions::Lossless:
|
||||||
return false;
|
if (!WebPConfigLosslessPreset(&config,
|
||||||
}
|
opts->compression())) {
|
||||||
config.image_hint = webp_options->getImageHint();
|
fop->setError("Error in WebP configuration\n");
|
||||||
}
|
return false;
|
||||||
else {
|
}
|
||||||
if (!WebPConfigPreset(&config, webp_options->getImagePreset(), static_cast<float>(webp_options->getQuality()))) {
|
config.image_hint = opts->imageHint();
|
||||||
fop->setError("Error in WebP configuration preset\n");
|
break;
|
||||||
return false;
|
|
||||||
}
|
case WebPOptions::Lossy:
|
||||||
|
if (!WebPConfigPreset(&config,
|
||||||
|
opts->imagePreset(),
|
||||||
|
static_cast<float>(opts->quality()))) {
|
||||||
|
fop->setError("Error in WebP configuration preset\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!WebPValidateConfig(&config)) {
|
WebPAnimEncoderOptions enc_options;
|
||||||
fop->setError("Error validating WebP encoder configuration\n");
|
WebPAnimEncoderOptionsInit(&enc_options);
|
||||||
return false;
|
enc_options.anim_params.loop_count =
|
||||||
}
|
(opts->loop() ? 0: // 0 = infinite
|
||||||
|
1); // 1 = loop once
|
||||||
|
|
||||||
|
ImageRef image(Image::create(IMAGE_RGB, w, h));
|
||||||
|
render::Render render;
|
||||||
|
|
||||||
|
WriterData wd(fp, fop, 0, sprite->totalFrames(), 0.0);
|
||||||
WebPPicture pic;
|
WebPPicture pic;
|
||||||
if (!WebPPictureInit(&pic)) {
|
WebPPictureInit(&pic);
|
||||||
fop->setError("Error encoding WebP picture, version mismatch\n");
|
pic.width = w;
|
||||||
return false;
|
pic.height = h;
|
||||||
}
|
pic.use_argb = true;
|
||||||
|
pic.argb = (uint32_t*)image->getPixelAddress(0, 0);
|
||||||
pic.width = image->width();
|
pic.argb_stride = w;
|
||||||
pic.height = image->height();
|
pic.user_data = &wd;
|
||||||
if (webp_options->lossless()) {
|
pic.progress_hook = progress_report;
|
||||||
pic.use_argb = true;
|
|
||||||
}
|
WebPAnimEncoder* enc = WebPAnimEncoderNew(sprite->width(),
|
||||||
|
sprite->height(),
|
||||||
if (!WebPPictureAlloc(&pic)) {
|
&enc_options);
|
||||||
fop->setError("Not enough memory to allocate a WebP picture\n");
|
int timestamp_ms = 0;
|
||||||
return false;
|
for (frame_t f=0; f<sprite->totalFrames(); ++f) {
|
||||||
}
|
// Render the frame in the bitmap
|
||||||
|
render.renderSprite(image.get(), sprite, f);
|
||||||
ScopedWebPPicture scopedPic(pic); // Calls WebPPictureFree automatically
|
|
||||||
|
// Switch R <-> B channels because WebPAnimEncoderAssemble()
|
||||||
if (!WebPPictureImportRGBA(&pic, (uint8_t*)image->getPixelAddress(0, 0), image->width() * sizeof(uint32_t))) {
|
// expects MODE_BGRA pictures.
|
||||||
fop->setError("Error converting RGBA data into a WebP picture\n");
|
{
|
||||||
return false;
|
LockImageBits<RgbTraits> bits(image.get(), Image::ReadWriteLock);
|
||||||
}
|
auto it = bits.begin(), end = bits.end();
|
||||||
|
for (; it != end; ++it) {
|
||||||
pic.writer = FileWriter;
|
auto c = *it;
|
||||||
pic.custom_ptr = &wd;
|
*it = rgba(rgba_getb(c), // Use blue in red channel
|
||||||
pic.progress_hook = ProgressReport;
|
rgba_getg(c),
|
||||||
|
rgba_getr(c), // Use red in blue channel
|
||||||
if (!WebPEncode(&config, &pic)) {
|
rgba_geta(c));
|
||||||
fop->setError("Error encoding image into WebP: %s\n",
|
}
|
||||||
getEncoderErrorMessage(pic.error_code));
|
}
|
||||||
|
|
||||||
|
if (!WebPAnimEncoderAdd(enc, &pic, timestamp_ms, &config)) {
|
||||||
|
if (!fop->isStop()) {
|
||||||
|
fop->setError("Error saving frame %d info\n", f);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
timestamp_ms += sprite->frameDuration(f);
|
||||||
|
|
||||||
|
wd.f = f;
|
||||||
|
}
|
||||||
|
WebPAnimEncoderAdd(enc, nullptr, timestamp_ms, nullptr);
|
||||||
|
|
||||||
|
WebPData webp_data;
|
||||||
|
WebPDataInit(&webp_data);
|
||||||
|
WebPAnimEncoderAssemble(enc, &webp_data);
|
||||||
|
WebPAnimEncoderDelete(enc);
|
||||||
|
|
||||||
|
if (fwrite(webp_data.bytes, 1, webp_data.size, fp) != webp_data.size) {
|
||||||
|
fop->setError("Error saving content into file\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
WebPDataClear(&webp_data);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -326,58 +367,101 @@ bool WebPFormat::onSave(FileOp* fop)
|
|||||||
// Shows the WebP configuration dialog.
|
// Shows the WebP configuration dialog.
|
||||||
base::SharedPtr<FormatOptions> WebPFormat::onGetFormatOptions(FileOp* fop)
|
base::SharedPtr<FormatOptions> WebPFormat::onGetFormatOptions(FileOp* fop)
|
||||||
{
|
{
|
||||||
base::SharedPtr<WebPOptions> webp_options;
|
base::SharedPtr<WebPOptions> opts;
|
||||||
if (fop->document()->getFormatOptions())
|
if (fop->document()->getFormatOptions())
|
||||||
webp_options = base::SharedPtr<WebPOptions>(fop->document()->getFormatOptions());
|
opts = base::SharedPtr<WebPOptions>(fop->document()->getFormatOptions());
|
||||||
|
|
||||||
if (!webp_options)
|
if (!opts)
|
||||||
webp_options.reset(new WebPOptions);
|
opts.reset(new WebPOptions);
|
||||||
|
|
||||||
// Non-interactive mode
|
// Non-interactive mode
|
||||||
if (!fop->context() ||
|
if (!fop->context() ||
|
||||||
!fop->context()->isUIAvailable())
|
!fop->context()->isUIAvailable())
|
||||||
return webp_options;
|
return opts;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// Configuration parameters
|
auto& pref = Preferences::instance();
|
||||||
webp_options->setQuality(get_config_int("WEBP", "Quality", webp_options->getQuality()));
|
|
||||||
webp_options->setMethod(get_config_int("WEBP", "Compression", webp_options->getMethod()));
|
|
||||||
webp_options->setImageHint(get_config_int("WEBP", "ImageHint", webp_options->getImageHint()));
|
|
||||||
webp_options->setImagePreset(get_config_int("WEBP", "ImagePreset", webp_options->getImagePreset()));
|
|
||||||
|
|
||||||
// Load the window to ask to the user the WebP options he wants.
|
if (pref.isSet(pref.webp.loop))
|
||||||
|
opts->setLoop(pref.webp.loop());
|
||||||
|
|
||||||
app::gen::WebpOptions win;
|
if (pref.isSet(pref.webp.type))
|
||||||
win.lossless()->setSelected(webp_options->lossless());
|
opts->setType(WebPOptions::Type(pref.webp.type()));
|
||||||
win.lossy()->setSelected(!webp_options->lossless());
|
|
||||||
win.quality()->setValue(static_cast<int>(webp_options->getQuality()));
|
|
||||||
win.compression()->setValue(webp_options->getMethod());
|
|
||||||
win.imageHint()->setSelectedItemIndex(webp_options->getImageHint());
|
|
||||||
win.imagePreset()->setSelectedItemIndex(webp_options->getImagePreset());
|
|
||||||
|
|
||||||
win.openWindowInForeground();
|
switch (opts->type()) {
|
||||||
|
case WebPOptions::Lossless:
|
||||||
if (win.closer() == win.ok()) {
|
if (pref.isSet(pref.webp.compression)) opts->setCompression(pref.webp.compression());
|
||||||
webp_options->setQuality(win.quality()->getValue());
|
if (pref.isSet(pref.webp.imageHint)) opts->setImageHint(WebPImageHint(pref.webp.imageHint()));
|
||||||
webp_options->setMethod(win.compression()->getValue());
|
break;
|
||||||
webp_options->setLossless(win.lossless()->isSelected());
|
case WebPOptions::Lossy:
|
||||||
webp_options->setImageHint(base::convert_to<int>(win.imageHint()->getValue()));
|
if (pref.isSet(pref.webp.quality)) opts->setQuality(pref.webp.quality());
|
||||||
webp_options->setImagePreset(base::convert_to<int>(win.imagePreset()->getValue()));
|
if (pref.isSet(pref.webp.imagePreset)) opts->setImagePreset(WebPPreset(pref.webp.imagePreset()));
|
||||||
|
break;
|
||||||
set_config_int("WEBP", "Quality", webp_options->getQuality());
|
|
||||||
set_config_int("WEBP", "Compression", webp_options->getMethod());
|
|
||||||
set_config_int("WEBP", "ImageHint", webp_options->getImageHint());
|
|
||||||
set_config_int("WEBP", "ImagePreset", webp_options->getImagePreset());
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
webp_options.reset(NULL);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return webp_options;
|
if (pref.webp.showAlert()) {
|
||||||
|
app::gen::WebpOptions win;
|
||||||
|
|
||||||
|
auto updatePanels = [&win, &opts]{
|
||||||
|
int o = base::convert_to<int>(win.type()->getValue());
|
||||||
|
opts->setType(WebPOptions::Type(o));
|
||||||
|
win.losslessOptions()->setVisible(o == int(WebPOptions::Lossless));
|
||||||
|
win.lossyOptions()->setVisible(o == int(WebPOptions::Lossy));
|
||||||
|
|
||||||
|
auto rc = win.bounds();
|
||||||
|
win.setBounds(
|
||||||
|
gfx::Rect(rc.origin(),
|
||||||
|
win.sizeHint()));
|
||||||
|
|
||||||
|
auto manager = win.manager();
|
||||||
|
if (manager)
|
||||||
|
manager->invalidateRect(rc); // TODO this should be automatic
|
||||||
|
// when a window bounds is modified
|
||||||
|
};
|
||||||
|
|
||||||
|
win.loop()->setSelected(opts->loop());
|
||||||
|
win.type()->setSelectedItemIndex(int(opts->type()));
|
||||||
|
win.compression()->setValue(opts->compression());
|
||||||
|
win.imageHint()->setSelectedItemIndex(opts->imageHint());
|
||||||
|
win.quality()->setValue(static_cast<int>(opts->quality()));
|
||||||
|
win.imagePreset()->setSelectedItemIndex(opts->imagePreset());
|
||||||
|
|
||||||
|
updatePanels();
|
||||||
|
win.type()->Change.connect(base::Bind<void>(updatePanels));
|
||||||
|
|
||||||
|
win.openWindowInForeground();
|
||||||
|
|
||||||
|
if (win.closer() == win.ok()) {
|
||||||
|
pref.webp.loop(win.loop()->isSelected());
|
||||||
|
pref.webp.type(base::convert_to<int>(win.type()->getValue()));
|
||||||
|
pref.webp.compression(win.compression()->getValue());
|
||||||
|
pref.webp.imageHint(base::convert_to<int>(win.imageHint()->getValue()));
|
||||||
|
pref.webp.quality(win.quality()->getValue());
|
||||||
|
pref.webp.imagePreset(base::convert_to<int>(win.imagePreset()->getValue()));
|
||||||
|
|
||||||
|
opts->setLoop(pref.webp.loop());
|
||||||
|
opts->setType(WebPOptions::Type(pref.webp.type()));
|
||||||
|
switch (opts->type()) {
|
||||||
|
case WebPOptions::Lossless:
|
||||||
|
opts->setCompression(pref.webp.compression());
|
||||||
|
opts->setImageHint(WebPImageHint(pref.webp.imageHint()));
|
||||||
|
break;
|
||||||
|
case WebPOptions::Lossy:
|
||||||
|
opts->setQuality(pref.webp.quality());
|
||||||
|
opts->setImagePreset(WebPPreset(pref.webp.imagePreset()));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
opts.reset(nullptr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return opts;
|
||||||
}
|
}
|
||||||
catch (std::exception& e) {
|
catch (const std::exception& e) {
|
||||||
Console::showException(e);
|
Console::showException(e);
|
||||||
return base::SharedPtr<WebPOptions>(0);
|
return base::SharedPtr<WebPOptions>(nullptr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
|
// Copyright (C) 2018 David Capello
|
||||||
// Copyright (C) 2015 Gabriel Rauter
|
// Copyright (C) 2015 Gabriel Rauter
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -14,32 +15,71 @@
|
|||||||
#include <webp/encode.h>
|
#include <webp/encode.h>
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
|
|
||||||
// Data for WebP files
|
// Data for WebP files
|
||||||
class WebPOptions : public FormatOptions {
|
class WebPOptions : public FormatOptions {
|
||||||
public:
|
public:
|
||||||
WebPOptions(): m_lossless(1), m_quality(75), m_method(6), m_image_hint(WEBP_HINT_DEFAULT), m_image_preset(WEBP_PRESET_DEFAULT) {};
|
enum Type { Simple, Lossless, Lossy };
|
||||||
|
|
||||||
bool lossless() { return m_lossless; }
|
// By default we use 6, because 9 is too slow
|
||||||
int getQuality() { return m_quality; }
|
const int kDefaultCompression = 6;
|
||||||
int getMethod() { return m_method; }
|
|
||||||
WebPImageHint getImageHint() { return m_image_hint; }
|
|
||||||
WebPPreset getImagePreset() { return m_image_preset; }
|
|
||||||
|
|
||||||
void setLossless(int lossless) { m_lossless = (lossless != 0); }
|
WebPOptions() : m_loop(true),
|
||||||
void setLossless(bool lossless) { m_lossless = lossless; }
|
m_type(Type::Simple),
|
||||||
void setQuality(int quality) { m_quality = quality; }
|
m_compression(kDefaultCompression),
|
||||||
void setMethod(int method) { m_method = method; }
|
m_imageHint(WEBP_HINT_DEFAULT),
|
||||||
void setImageHint(int imageHint) { m_image_hint = static_cast<WebPImageHint>(imageHint); }
|
m_quality(100),
|
||||||
void setImageHint(WebPImageHint imageHint) { m_image_hint = imageHint; }
|
m_imagePreset(WEBP_PRESET_DEFAULT) { }
|
||||||
void setImagePreset(int imagePreset) { m_image_preset = static_cast<WebPPreset>(imagePreset); };
|
|
||||||
void setImagePreset(WebPPreset imagePreset) { m_image_preset = imagePreset ; }
|
bool loop() const { return m_loop; }
|
||||||
|
Type type() const { return m_type; }
|
||||||
|
int compression() const { return m_compression; }
|
||||||
|
WebPImageHint imageHint() const { return m_imageHint; }
|
||||||
|
int quality() const { return m_quality; }
|
||||||
|
WebPPreset imagePreset() const { return m_imagePreset; }
|
||||||
|
|
||||||
|
void setLoop(const bool loop) {
|
||||||
|
m_loop = loop;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setType(const Type type) {
|
||||||
|
m_type = type;
|
||||||
|
|
||||||
|
if (m_type == Type::Simple) {
|
||||||
|
m_compression = kDefaultCompression;
|
||||||
|
m_imageHint = WEBP_HINT_DEFAULT;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void setCompression(const int compression) {
|
||||||
|
ASSERT(m_type == Type::Lossless);
|
||||||
|
m_compression = compression;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setImageHint(const WebPImageHint imageHint) {
|
||||||
|
ASSERT(m_type == Type::Lossless);
|
||||||
|
m_imageHint = imageHint;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setQuality(const int quality) {
|
||||||
|
ASSERT(m_type == Type::Lossy);
|
||||||
|
m_quality = quality;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setImagePreset(const WebPPreset imagePreset) {
|
||||||
|
ASSERT(m_type == Type::Lossy);
|
||||||
|
m_imagePreset = imagePreset;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool m_lossless; // Lossless encoding (0=lossy(default), 1=lossless).
|
bool m_loop;
|
||||||
int m_quality; // between 0 (smallest file) and 100 (biggest)
|
Type m_type;
|
||||||
int m_method; // quality/speed trade-off (0=fast, 9=slower-better)
|
// Lossless options
|
||||||
WebPImageHint m_image_hint; // Hint for image type (lossless only for now).
|
int m_compression; // Quality/speed trade-off (0=fast, 9=slower-better)
|
||||||
WebPPreset m_image_preset; // Image Preset for lossy webp.
|
WebPImageHint m_imageHint; // Hint for image type (lossless only for now).
|
||||||
|
// Lossy options
|
||||||
|
int m_quality; // Between 0 (smallest file) and 100 (biggest)
|
||||||
|
WebPPreset m_imagePreset; // Image Preset for lossy webp.
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace app
|
} // namespace app
|
||||||
|
17
third_party/CMakeLists.txt
vendored
17
third_party/CMakeLists.txt
vendored
@ -1,5 +1,5 @@
|
|||||||
# ASEPRITE
|
# ASEPRITE
|
||||||
# Copyright (C) 2001-2017 David Capello
|
# Copyright (C) 2001-2018 David Capello
|
||||||
|
|
||||||
include_directories(.)
|
include_directories(.)
|
||||||
|
|
||||||
@ -29,20 +29,17 @@ if(NOT USE_SHARED_LIBPNG)
|
|||||||
add_subdirectory(libpng)
|
add_subdirectory(libpng)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(WITH_WEBP_SUPPORT)
|
|
||||||
if(NOT USE_SHARED_LIBWEBP)
|
|
||||||
# Skia already includes webp library
|
|
||||||
if(NOT USE_SKIA_BACKEND)
|
|
||||||
add_subdirectory(libwebp-cmake)
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(NOT USE_SHARED_GIFLIB)
|
if(NOT USE_SHARED_GIFLIB)
|
||||||
set(GIFLIB_UTILS OFF CACHE BOOL "Build giflib utils")
|
set(GIFLIB_UTILS OFF CACHE BOOL "Build giflib utils")
|
||||||
add_subdirectory(giflib)
|
add_subdirectory(giflib)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
if(WITH_WEBP_SUPPORT)
|
||||||
|
# Enable img2webp so "webpmux" library is built
|
||||||
|
set(WEBP_BUILD_IMG2WEBP ON CACHE BOOL "Build the img2webp animation tool.")
|
||||||
|
add_subdirectory(libwebp)
|
||||||
|
endif()
|
||||||
|
|
||||||
if(NOT USE_SHARED_TINYXML)
|
if(NOT USE_SHARED_TINYXML)
|
||||||
add_subdirectory(tinyxml)
|
add_subdirectory(tinyxml)
|
||||||
endif()
|
endif()
|
||||||
|
45
third_party/libwebp-cmake/CMakeLists.txt
vendored
45
third_party/libwebp-cmake/CMakeLists.txt
vendored
@ -1,45 +0,0 @@
|
|||||||
project (webp C)
|
|
||||||
cmake_minimum_required(VERSION 2.6)
|
|
||||||
|
|
||||||
add_definitions(-DNDEBUG -DWEBP_USE_THREAD)
|
|
||||||
|
|
||||||
set(LIBWEBP_SOURCE_DIR "${CMAKE_CURRENT_SOURCE_DIR}/../libwebp")
|
|
||||||
|
|
||||||
file(GLOB WEBP_DEC_SRCS
|
|
||||||
RELATIVE ${CMAKE_CURRENT_SOURCE_DIR}
|
|
||||||
${LIBWEBP_SOURCE_DIR}/src/dec/*c
|
|
||||||
)
|
|
||||||
|
|
||||||
file(GLOB WEBP_DEMUX_SRCS
|
|
||||||
RELATIVE ${CMAKE_CURRENT_SOURCE_DIR}
|
|
||||||
${LIBWEBP_SOURCE_DIR}/src/demux/*c
|
|
||||||
)
|
|
||||||
|
|
||||||
file(GLOB WEBP_DSP_SRCS
|
|
||||||
RELATIVE ${CMAKE_CURRENT_SOURCE_DIR}
|
|
||||||
${LIBWEBP_SOURCE_DIR}/src/dsp/*c
|
|
||||||
)
|
|
||||||
|
|
||||||
file(GLOB WEBP_ENC_SRCS
|
|
||||||
RELATIVE ${CMAKE_CURRENT_SOURCE_DIR}
|
|
||||||
${LIBWEBP_SOURCE_DIR}/src/enc/*c
|
|
||||||
)
|
|
||||||
|
|
||||||
file(GLOB WEBP_UTILS_SRCS
|
|
||||||
RELATIVE ${CMAKE_CURRENT_SOURCE_DIR}
|
|
||||||
${LIBWEBP_SOURCE_DIR}/src/utils/*c
|
|
||||||
)
|
|
||||||
|
|
||||||
file(GLOB WEBP_MUX_SRCS
|
|
||||||
RELATIVE ${CMAKE_CURRENT_SOURCE_DIR}
|
|
||||||
${LIBWEBP_SOURCE_DIR}/src/mux/*c
|
|
||||||
)
|
|
||||||
|
|
||||||
file(GLOB WEBP_HEADERS
|
|
||||||
RELATIVE ${CMAKE_CURRENT_SOURCE_DIR}
|
|
||||||
${LIBWEBP_SOURCE_DIR}/src/webp/*.h
|
|
||||||
)
|
|
||||||
|
|
||||||
SET(CMAKE_DEBUG_POSTFIX "d")
|
|
||||||
set(WEBP_SOURCE ${WEBP_DEC_SRCS} ${WEBP_DEMUX_SRCS} ${WEBP_DSP_SRCS} ${WEBP_ENC_SRCS} ${WEBP_UTILS_SRCS} ${WEBP_MUX_SRC})
|
|
||||||
add_library(${PROJECT_NAME} STATIC ${WEBP_SOURCE} ${WEBP_HEADERS})
|
|
Loading…
x
Reference in New Issue
Block a user