mirror of https://github.com/procxx/kepka.git
Started Lottie::CacheState class.
This commit is contained in:
parent
35bc2cc2a5
commit
10772f4ac5
|
@ -1100,11 +1100,8 @@ std::unique_ptr<Lottie::Animation> LottieFromDocument(
|
|||
data,
|
||||
filepath,
|
||||
box);
|
||||
} else if (!data.isEmpty()) {
|
||||
return Lottie::FromData(data);
|
||||
} else {
|
||||
return Lottie::FromFile(filepath);
|
||||
}
|
||||
return Lottie::FromContent(data, filepath);
|
||||
}
|
||||
|
||||
} // namespace Stickers
|
||||
|
|
|
@ -1368,9 +1368,9 @@ void StickersListWidget::setupLottie(Set &set, int section, int index) {
|
|||
auto &sticker = set.stickers[index];
|
||||
const auto document = sticker.document;
|
||||
|
||||
sticker.animated = document->data().isEmpty()
|
||||
? Lottie::FromFile(document->filepath())
|
||||
: Lottie::FromData(document->data());
|
||||
sticker.animated = Lottie::FromContent(
|
||||
document->data(),
|
||||
document->filepath());
|
||||
const auto animation = sticker.animated.get();
|
||||
|
||||
animation->updates(
|
||||
|
|
|
@ -97,9 +97,7 @@ QSize HistorySticker::countCurrentSize(int newWidth) {
|
|||
}
|
||||
|
||||
void HistorySticker::setupLottie() {
|
||||
_lottie = _data->data().isEmpty()
|
||||
? Lottie::FromFile(_data->filepath())
|
||||
: Lottie::FromData(_data->data());
|
||||
_lottie = Lottie::FromContent(_data->data(), _data->filepath());
|
||||
_parent->data()->history()->owner().registerHeavyViewPart(_parent);
|
||||
|
||||
_lottie->updates(
|
||||
|
|
|
@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "lottie/lottie_animation.h"
|
||||
|
||||
#include "lottie/lottie_frame_renderer.h"
|
||||
#include "lottie/lottie_cache.h"
|
||||
#include "storage/cache/storage_cache_database.h"
|
||||
#include "base/algorithm.h"
|
||||
#include "zlib.h"
|
||||
|
@ -56,19 +57,85 @@ std::string UnpackGzip(const QByteArray &bytes) {
|
|||
return result;
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
std::unique_ptr<Animation> FromFile(const QString &path) {
|
||||
return FromData([&] {
|
||||
auto f = QFile(path);
|
||||
return (f.size() <= kMaxFileSize && f.open(QIODevice::ReadOnly))
|
||||
? f.readAll()
|
||||
: QByteArray();
|
||||
}());
|
||||
QByteArray ReadFile(const QString &filepath) {
|
||||
auto f = QFile(filepath);
|
||||
return (f.size() <= kMaxFileSize && f.open(QIODevice::ReadOnly))
|
||||
? f.readAll()
|
||||
: QByteArray();
|
||||
}
|
||||
|
||||
std::unique_ptr<Animation> FromData(const QByteArray &data) {
|
||||
return std::make_unique<Animation>(base::duplicate(data));
|
||||
QByteArray ReadContent(const QByteArray &data, const QString &filepath) {
|
||||
return data.isEmpty() ? ReadFile(filepath) : base::duplicate(data);
|
||||
}
|
||||
|
||||
std::optional<Error> ContentError(const QByteArray &content) {
|
||||
if (content.size() > kMaxFileSize) {
|
||||
qWarning() << "Lottie Error: Too large file: " << content.size();
|
||||
return Error::ParseFailed;
|
||||
}
|
||||
return std::nullopt;
|
||||
}
|
||||
|
||||
std::unique_ptr<rlottie::Animation> CreateImplementation(
|
||||
const QByteArray &content) {
|
||||
const auto string = UnpackGzip(content);
|
||||
Assert(string.size() <= kMaxFileSize);
|
||||
|
||||
auto result = rlottie::Animation::loadFromData(string, std::string());
|
||||
if (!result) {
|
||||
qWarning() << "Lottie Error: Parse failed.";
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
details::InitData CheckSharedState(std::unique_ptr<SharedState> state) {
|
||||
Expects(state != nullptr);
|
||||
|
||||
auto information = state->information();
|
||||
if (!information.frameRate
|
||||
|| information.framesCount <= 0
|
||||
|| information.size.isEmpty()) {
|
||||
return Error::NotSupported;
|
||||
}
|
||||
return state;
|
||||
}
|
||||
|
||||
details::InitData Init(const QByteArray &content) {
|
||||
if (const auto error = ContentError(content)) {
|
||||
return *error;
|
||||
}
|
||||
auto animation = CreateImplementation(content);
|
||||
return animation
|
||||
? CheckSharedState(std::make_unique<SharedState>(
|
||||
std::move(animation)))
|
||||
: Error::ParseFailed;
|
||||
}
|
||||
|
||||
details::InitData Init(
|
||||
const QByteArray &content,
|
||||
not_null<Storage::Cache::Database*> cache,
|
||||
Storage::Cache::Key key,
|
||||
const QByteArray &cached,
|
||||
QSize box) {
|
||||
if (const auto error = ContentError(content)) {
|
||||
return *error;
|
||||
}
|
||||
auto state = CacheState(cached, box);
|
||||
const auto prepare = !state.framesCount()
|
||||
|| (state.framesReady() < state.framesCount());
|
||||
auto animation = prepare ? CreateImplementation(content) : nullptr;
|
||||
return (!prepare || animation)
|
||||
? CheckSharedState(std::make_unique<SharedState>(
|
||||
std::move(animation)))
|
||||
: Error::ParseFailed;
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
std::unique_ptr<Animation> FromContent(
|
||||
const QByteArray &data,
|
||||
const QString &filepath) {
|
||||
return std::make_unique<Animation>(ReadContent(data, filepath));
|
||||
}
|
||||
|
||||
std::unique_ptr<Animation> FromCached(
|
||||
|
@ -77,40 +144,14 @@ std::unique_ptr<Animation> FromCached(
|
|||
const QByteArray &data,
|
||||
const QString &filepath,
|
||||
QSize box) {
|
||||
return data.isEmpty()
|
||||
? Lottie::FromFile(filepath)
|
||||
: Lottie::FromData(data);
|
||||
return std::make_unique<Animation>(
|
||||
cache,
|
||||
key,
|
||||
ReadContent(data, filepath),
|
||||
box);
|
||||
}
|
||||
|
||||
auto Init(QByteArray &&content)
|
||||
-> base::variant<std::unique_ptr<SharedState>, Error> {
|
||||
if (content.size() > kMaxFileSize) {
|
||||
qWarning()
|
||||
<< "Lottie Error: Too large file: "
|
||||
<< content.size();
|
||||
return Error::ParseFailed;
|
||||
}
|
||||
const auto string = UnpackGzip(content);
|
||||
Assert(string.size() <= kMaxFileSize);
|
||||
|
||||
auto animation = rlottie::Animation::loadFromData(string, std::string());
|
||||
if (!animation) {
|
||||
qWarning()
|
||||
<< "Lottie Error: Parse failed.";
|
||||
return Error::ParseFailed;
|
||||
}
|
||||
|
||||
auto result = std::make_unique<SharedState>(std::move(animation));
|
||||
auto information = result->information();
|
||||
if (!information.frameRate
|
||||
|| information.framesCount <= 0
|
||||
|| information.size.isEmpty()) {
|
||||
return Error::NotSupported;
|
||||
}
|
||||
return std::move(result);
|
||||
}
|
||||
|
||||
QImage ReadThumbnail(QByteArray &&content) {
|
||||
QImage ReadThumbnail(const QByteArray &content) {
|
||||
return Init(std::move(content)).match([](
|
||||
const std::unique_ptr<SharedState> &state) {
|
||||
return state->frameForPaint()->original;
|
||||
|
@ -119,15 +160,28 @@ QImage ReadThumbnail(QByteArray &&content) {
|
|||
});
|
||||
}
|
||||
|
||||
Animation::Animation(QByteArray &&content)
|
||||
Animation::Animation(const QByteArray &content)
|
||||
: _timer([=] { checkNextFrameRender(); }) {
|
||||
const auto weak = base::make_weak(this);
|
||||
crl::async([=, content = base::take(content)]() mutable {
|
||||
crl::on_main(weak, [this, result = Init(std::move(content))]() mutable {
|
||||
result.match([&](std::unique_ptr<SharedState> &state) {
|
||||
parseDone(std::move(state));
|
||||
}, [&](Error error) {
|
||||
parseFailed(error);
|
||||
crl::async([=] {
|
||||
crl::on_main(weak, [=, data = Init(content)]() mutable {
|
||||
initDone(std::move(data));
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
Animation::Animation(
|
||||
not_null<Storage::Cache::Database*> cache,
|
||||
Storage::Cache::Key key,
|
||||
const QByteArray &content,
|
||||
QSize box)
|
||||
: _timer([=] { checkNextFrameRender(); }) {
|
||||
const auto weak = base::make_weak(this);
|
||||
cache->get(key, [=](QByteArray &&cached) mutable {
|
||||
crl::async([=] {
|
||||
auto result = Init(content, cache, key, cached, box);
|
||||
crl::on_main(weak, [=, data = std::move(result)]() mutable {
|
||||
initDone(std::move(data));
|
||||
});
|
||||
});
|
||||
});
|
||||
|
@ -140,6 +194,14 @@ Animation::~Animation() {
|
|||
}
|
||||
}
|
||||
|
||||
void Animation::initDone(details::InitData &&data) {
|
||||
data.match([&](std::unique_ptr<SharedState> &state) {
|
||||
parseDone(std::move(state));
|
||||
}, [&](Error error) {
|
||||
parseFailed(error);
|
||||
});
|
||||
}
|
||||
|
||||
void Animation::parseDone(std::unique_ptr<SharedState> state) {
|
||||
Expects(state != nullptr);
|
||||
|
||||
|
|
|
@ -32,14 +32,15 @@ struct Key;
|
|||
|
||||
namespace Lottie {
|
||||
|
||||
constexpr auto kMaxFileSize = 1024 * 1024;
|
||||
inline constexpr auto kMaxFileSize = 1024 * 1024;
|
||||
|
||||
class Animation;
|
||||
class SharedState;
|
||||
class Animation;
|
||||
class FrameRenderer;
|
||||
|
||||
std::unique_ptr<Animation> FromFile(const QString &path);
|
||||
std::unique_ptr<Animation> FromData(const QByteArray &data);
|
||||
std::unique_ptr<Animation> FromContent(
|
||||
const QByteArray &data,
|
||||
const QString &filepath);
|
||||
std::unique_ptr<Animation> FromCached(
|
||||
not_null<Storage::Cache::Database*> cache,
|
||||
Storage::Cache::Key key,
|
||||
|
@ -47,11 +48,22 @@ std::unique_ptr<Animation> FromCached(
|
|||
const QString &filepath,
|
||||
QSize box);
|
||||
|
||||
QImage ReadThumbnail(QByteArray &&content);
|
||||
QImage ReadThumbnail(const QByteArray &content);
|
||||
|
||||
namespace details {
|
||||
|
||||
using InitData = base::variant<std::unique_ptr<SharedState>, Error>;
|
||||
|
||||
} // namespace details
|
||||
|
||||
class Animation final : public base::has_weak_ptr {
|
||||
public:
|
||||
explicit Animation(QByteArray &&content);
|
||||
explicit Animation(const QByteArray &content);
|
||||
Animation(
|
||||
not_null<Storage::Cache::Database*> cache,
|
||||
Storage::Cache::Key key,
|
||||
const QByteArray &content,
|
||||
QSize box);
|
||||
~Animation();
|
||||
|
||||
//void play(const PlaybackOptions &options);
|
||||
|
@ -69,6 +81,7 @@ public:
|
|||
void checkStep();
|
||||
|
||||
private:
|
||||
void initDone(details::InitData &&data);
|
||||
void parseDone(std::unique_ptr<SharedState> state);
|
||||
void parseFailed(Error error);
|
||||
|
||||
|
|
|
@ -0,0 +1,225 @@
|
|||
/*
|
||||
This file is part of Telegram Desktop,
|
||||
the official desktop application for the Telegram messaging service.
|
||||
|
||||
For license and copyright information please follow this link:
|
||||
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||
*/
|
||||
#include "lottie/lottie_cache.h"
|
||||
|
||||
#include "lottie/lottie_frame_renderer.h"
|
||||
#include "base/bytes.h"
|
||||
|
||||
#include <QDataStream>
|
||||
|
||||
namespace Lottie {
|
||||
namespace {
|
||||
|
||||
constexpr auto kAlignStorage = 16;
|
||||
|
||||
bool UncompressToRaw(AlignedStorage &to, bytes::const_span from) {
|
||||
if (from.empty() || from.size() > to.rawSize()) {
|
||||
return false;
|
||||
} else if (from.size() == to.rawSize()) {
|
||||
memcpy(to.raw(), from.data(), from.size());
|
||||
return true;
|
||||
} else {
|
||||
// #TODO stickers
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
void Decode(QImage &to, const AlignedStorage &from, const QSize &fromSize) {
|
||||
auto fromBytes = static_cast<const char*>(from.aligned());
|
||||
auto toBytes = to.bits();
|
||||
const auto fromPerLine = from.bytesPerLine();
|
||||
const auto toPerLine = to.bytesPerLine();
|
||||
for (auto i = 0; i != to.height(); ++i) {
|
||||
memcpy(toBytes, fromBytes, to.width() * 4);
|
||||
fromBytes += fromPerLine;
|
||||
toBytes += toPerLine;
|
||||
}
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
void AlignedStorage::allocate(int packedBytesPerLine, int lines) {
|
||||
Expects(packedBytesPerLine >= 0);
|
||||
Expects(lines >= 0);
|
||||
|
||||
_packedBytesPerLine = packedBytesPerLine;
|
||||
_lines = lines;
|
||||
reallocate();
|
||||
}
|
||||
|
||||
void AlignedStorage::reallocate() {
|
||||
const auto perLine = bytesPerLine();
|
||||
const auto total = perLine * _lines;
|
||||
_buffer = QByteArray(total + kAlignStorage - 1, Qt::Uninitialized);
|
||||
_raw = (perLine != _packedBytesPerLine)
|
||||
? QByteArray(_packedBytesPerLine * _lines, Qt::Uninitialized)
|
||||
: QByteArray();
|
||||
}
|
||||
|
||||
int AlignedStorage::lines() const {
|
||||
return _lines;
|
||||
}
|
||||
|
||||
int AlignedStorage::rawSize() const {
|
||||
return _lines * _packedBytesPerLine;
|
||||
}
|
||||
|
||||
void *AlignedStorage::raw() {
|
||||
return (bytesPerLine() == _packedBytesPerLine) ? aligned() : _raw.data();
|
||||
}
|
||||
|
||||
const void *AlignedStorage::raw() const {
|
||||
return (bytesPerLine() == _packedBytesPerLine) ? aligned() : _raw.data();
|
||||
}
|
||||
|
||||
int AlignedStorage::bytesPerLine() const {
|
||||
return kAlignStorage
|
||||
* ((_packedBytesPerLine + kAlignStorage - 1) / kAlignStorage);
|
||||
}
|
||||
|
||||
void *AlignedStorage::aligned() {
|
||||
const auto result = reinterpret_cast<quintptr>(_buffer.data());
|
||||
return reinterpret_cast<void*>(kAlignStorage
|
||||
* ((result + kAlignStorage - 1) / kAlignStorage));
|
||||
}
|
||||
|
||||
const void *AlignedStorage::aligned() const {
|
||||
const auto result = reinterpret_cast<quintptr>(_buffer.data());
|
||||
return reinterpret_cast<void*>(kAlignStorage
|
||||
* ((result + kAlignStorage - 1) / kAlignStorage));
|
||||
}
|
||||
|
||||
void AlignedStorage::copyRawToAligned() {
|
||||
const auto fromPerLine = _packedBytesPerLine;
|
||||
const auto toPerLine = bytesPerLine();
|
||||
if (fromPerLine == toPerLine) {
|
||||
return;
|
||||
}
|
||||
auto from = static_cast<char*>(raw());
|
||||
auto to = static_cast<char*>(aligned());
|
||||
for (auto i = 0; i != _lines; ++i) {
|
||||
memcpy(from, to, fromPerLine);
|
||||
from += fromPerLine;
|
||||
to += toPerLine;
|
||||
}
|
||||
}
|
||||
|
||||
void AlignedStorage::copyAlignedToRaw() {
|
||||
const auto fromPerLine = bytesPerLine();
|
||||
const auto toPerLine = _packedBytesPerLine;
|
||||
if (fromPerLine == toPerLine) {
|
||||
return;
|
||||
}
|
||||
auto from = static_cast<char*>(aligned());
|
||||
auto to = static_cast<char*>(raw());
|
||||
for (auto i = 0; i != _lines; ++i) {
|
||||
memcpy(from, to, toPerLine);
|
||||
from += fromPerLine;
|
||||
to += toPerLine;
|
||||
}
|
||||
}
|
||||
|
||||
CacheState::CacheState(const QByteArray &data, QSize box)
|
||||
: _data(data) {
|
||||
if (!readHeader(box)) {
|
||||
_framesReady = 0;
|
||||
}
|
||||
}
|
||||
|
||||
int CacheState::frameRate() const {
|
||||
return _frameRate;
|
||||
}
|
||||
|
||||
int CacheState::framesReady() const {
|
||||
return _framesReady;
|
||||
}
|
||||
|
||||
int CacheState::framesCount() const {
|
||||
return _framesCount;
|
||||
}
|
||||
|
||||
bool CacheState::readHeader(QSize box) {
|
||||
if (_data.isEmpty()) {
|
||||
return false;
|
||||
|
||||
}
|
||||
QDataStream stream(&_data, QIODevice::ReadOnly);
|
||||
|
||||
auto encoder = uchar(0);
|
||||
stream >> encoder;
|
||||
if (static_cast<Encoder>(encoder) != Encoder::YUV420A4_LZ4) {
|
||||
return false;
|
||||
}
|
||||
auto size = QSize();
|
||||
auto original = QSize();
|
||||
auto frameRate = qint32(0);
|
||||
auto framesCount = qint32(0);
|
||||
auto framesReady = qint32(0);
|
||||
stream
|
||||
>> size
|
||||
>> original
|
||||
>> frameRate
|
||||
>> framesCount
|
||||
>> framesReady;
|
||||
if (stream.status() != QDataStream::Ok
|
||||
|| original.isEmpty()
|
||||
|| (original.width() > kMaxSize)
|
||||
|| (original.height() > kMaxSize)
|
||||
|| (frameRate <= 0)
|
||||
|| (frameRate > kMaxFrameRate)
|
||||
|| (framesCount <= 0)
|
||||
|| (framesCount > kMaxFramesCount)
|
||||
|| (framesReady <= 0)
|
||||
|| (framesReady > framesCount)
|
||||
|| FrameRequest{ box }.size(original) != size) {
|
||||
return false;
|
||||
}
|
||||
_size = size;
|
||||
_original = original;
|
||||
_frameRate = frameRate;
|
||||
_framesCount = framesCount;
|
||||
_framesReady = framesReady;
|
||||
prepareBuffers();
|
||||
if (!readCompressedDelta(stream.device()->pos())) {
|
||||
return false;
|
||||
}
|
||||
_uncompressed.copyRawToAligned();
|
||||
std::swap(_uncompressed, _previous);
|
||||
Decode(_firstFrame, _previous, _size);
|
||||
return true;
|
||||
}
|
||||
|
||||
QImage CacheState::takeFirstFrame() {
|
||||
return std::move(_firstFrame);
|
||||
}
|
||||
|
||||
void CacheState::prepareBuffers() {
|
||||
_uncompressed.allocate(_size.width() * 4, _size.height());
|
||||
}
|
||||
|
||||
int CacheState::uncompressedDeltaSize() const {
|
||||
return _size.width() * _size.height() * 4; // #TODO stickers
|
||||
}
|
||||
|
||||
bool CacheState::readCompressedDelta(int offset) {
|
||||
auto length = qint32(0);
|
||||
const auto part = bytes::make_span(_data).subspan(offset);
|
||||
if (part.size() < sizeof(length)) {
|
||||
return false;
|
||||
}
|
||||
bytes::copy(bytes::object_as_span(&length), part);
|
||||
const auto bytes = part.subspan(sizeof(length));
|
||||
const auto uncompressedSize = uncompressedDeltaSize();
|
||||
|
||||
_offset = offset + length;
|
||||
return (length <= bytes.size())
|
||||
? UncompressToRaw(_uncompressed, bytes.subspan(0, length))
|
||||
: false;
|
||||
}
|
||||
|
||||
} // namespace Lottie
|
|
@ -0,0 +1,80 @@
|
|||
/*
|
||||
This file is part of Telegram Desktop,
|
||||
the official desktop application for the Telegram messaging service.
|
||||
|
||||
For license and copyright information please follow this link:
|
||||
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||
*/
|
||||
#pragma once
|
||||
|
||||
#include <QImage>
|
||||
#include <QSize>
|
||||
#include <QByteArray>
|
||||
|
||||
namespace Lottie {
|
||||
|
||||
class AlignedStorage {
|
||||
public:
|
||||
void allocate(int packedBytesPerLine, int lines);
|
||||
|
||||
int lines() const;
|
||||
int rawSize() const;
|
||||
|
||||
// Gives a pointer to packedBytesPerLine * lines bytes of memory.
|
||||
void *raw();
|
||||
const void *raw() const;
|
||||
|
||||
// Gives a stride value in the aligned storage (% 16 == 0).
|
||||
int bytesPerLine() const;
|
||||
|
||||
// Gives a pointer to the aligned memory (% 16 == 0).
|
||||
void *aligned();
|
||||
const void *aligned() const;
|
||||
|
||||
void copyRawToAligned();
|
||||
void copyAlignedToRaw();
|
||||
|
||||
private:
|
||||
void reallocate();
|
||||
|
||||
int _packedBytesPerLine = 0;
|
||||
int _lines = 0;
|
||||
QByteArray _raw;
|
||||
QByteArray _buffer;
|
||||
|
||||
};
|
||||
|
||||
class CacheState {
|
||||
public:
|
||||
enum class Encoder : uchar {
|
||||
YUV420A4_LZ4,
|
||||
};
|
||||
|
||||
CacheState(const QByteArray &data, QSize box);
|
||||
|
||||
[[nodiscard]] int frameRate() const;
|
||||
[[nodiscard]] int framesReady() const;
|
||||
[[nodiscard]] int framesCount() const;
|
||||
[[nodiscard]] QImage takeFirstFrame();
|
||||
|
||||
private:
|
||||
[[nodiscard]] bool readHeader(QSize box);
|
||||
void prepareBuffers();
|
||||
[[nodiscard]] bool readCompressedDelta(int offset);
|
||||
[[nodiscard]] int uncompressedDeltaSize() const;
|
||||
|
||||
QByteArray _data;
|
||||
QSize _size;
|
||||
QSize _original;
|
||||
AlignedStorage _uncompressed;
|
||||
AlignedStorage _previous;
|
||||
QImage _firstFrame;
|
||||
int _frameRate = 0;
|
||||
int _framesCount = 0;
|
||||
int _framesReady = 0;
|
||||
int _offset = 0;
|
||||
Encoder _encoder = Encoder::YUV420A4_LZ4;
|
||||
|
||||
};
|
||||
|
||||
} // namespace Lottie
|
|
@ -23,8 +23,6 @@ namespace Lottie {
|
|||
namespace {
|
||||
|
||||
constexpr auto kDisplaySkipped = crl::time(-1);
|
||||
constexpr auto kMaxFrameRate = 120;
|
||||
constexpr auto kMaxSize = 3096;
|
||||
|
||||
std::weak_ptr<FrameRenderer> GlobalInstance;
|
||||
|
||||
|
@ -199,7 +197,7 @@ void SharedState::calculateProperties() {
|
|||
(width > 0 && width < kMaxSize) ? int(width) : 0,
|
||||
(height > 0 && height < kMaxSize) ? int(height) : 0);
|
||||
_frameRate = (rate >= 1. && rate <= kMaxFrameRate) ? int(rate) : 0;
|
||||
_framesCount = (count > 0) ? int(count) : 0;
|
||||
_framesCount = (count > 0 && count <= kMaxFramesCount) ? int(count) : 0;
|
||||
}
|
||||
|
||||
bool SharedState::isValid() const {
|
||||
|
|
|
@ -23,6 +23,10 @@ class Animation;
|
|||
|
||||
namespace Lottie {
|
||||
|
||||
inline constexpr auto kMaxFrameRate = 120;
|
||||
inline constexpr auto kMaxSize = 3096;
|
||||
inline constexpr auto kMaxFramesCount = 600;
|
||||
|
||||
class Animation;
|
||||
class JsonObject;
|
||||
|
||||
|
|
|
@ -1049,9 +1049,7 @@ QSize MediaPreviewWidget::currentDimensions() const {
|
|||
void MediaPreviewWidget::setupLottie() {
|
||||
Expects(_document != nullptr);
|
||||
|
||||
_lottie = _document->data().isEmpty()
|
||||
? Lottie::FromFile(_document->filepath())
|
||||
: Lottie::FromData(_document->data());
|
||||
_lottie = Lottie::FromContent(_document->data(), _document->filepath());
|
||||
|
||||
_lottie->updates(
|
||||
) | rpl::start_with_next_error([=](Lottie::Update update) {
|
||||
|
|
|
@ -53,6 +53,8 @@
|
|||
'sources': [
|
||||
'<(src_loc)/lottie/lottie_animation.cpp',
|
||||
'<(src_loc)/lottie/lottie_animation.h',
|
||||
'<(src_loc)/lottie/lottie_cache.cpp',
|
||||
'<(src_loc)/lottie/lottie_cache.h',
|
||||
'<(src_loc)/lottie/lottie_common.h',
|
||||
'<(src_loc)/lottie/lottie_frame_renderer.cpp',
|
||||
'<(src_loc)/lottie/lottie_frame_renderer.h',
|
||||
|
|
Loading…
Reference in New Issue