| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420 |
- // This file is part of Desktop App Toolkit,
- // a set of libraries for developing nice desktop applications.
- //
- // For license and copyright information please follow this link:
- // https://github.com/desktop-app/legal/blob/master/LEGAL
- //
- #pragma once
- #include "ui/style/style_core.h"
- #include "base/basic_types.h"
- #include <QtGui/QPainterPath>
- #include <crl/crl_time.h>
- namespace anim {
- enum class type : uchar {
- normal,
- instant,
- };
- enum class activation : uchar {
- normal,
- background,
- };
- enum class repeat : uchar {
- loop,
- once,
- };
- using transition = Fn<float64(float64 delta, float64 dt)>;
- extern transition linear;
- extern transition sineInOut;
- extern transition halfSine;
- extern transition easeOutBack;
- extern transition easeInCirc;
- extern transition easeOutCirc;
- extern transition easeInCubic;
- extern transition easeOutCubic;
- extern transition easeInQuint;
- extern transition easeOutQuint;
- inline transition bumpy(float64 bump) {
- auto dt0 = (bump - sqrt(bump * (bump - 1.)));
- auto k = (1 / (2 * dt0 - 1));
- return [bump, dt0, k](float64 delta, float64 dt) {
- return delta * (bump - k * (dt - dt0) * (dt - dt0));
- };
- }
- // Basic animated value.
- class value {
- public:
- using ValueType = float64;
- value() = default;
- value(float64 from) : _cur(from), _from(from) {
- }
- value(float64 from, float64 to) : _cur(from), _from(from), _delta(to - from) {
- }
- void start(float64 to) {
- _from = _cur;
- _delta = to - _from;
- }
- void restart() {
- _delta = _from + _delta - _cur;
- _from = _cur;
- }
- float64 from() const {
- return _from;
- }
- float64 current() const {
- return _cur;
- }
- float64 to() const {
- return _from + _delta;
- }
- void add(float64 delta) {
- _from += delta;
- _cur += delta;
- }
- value &update(float64 dt, transition func) {
- _cur = _from + func(_delta, dt);
- return *this;
- }
- void finish() {
- _cur = _from + _delta;
- _from = _cur;
- _delta = 0;
- }
- private:
- float64 _cur = 0.;
- float64 _from = 0.;
- float64 _delta = 0.;
- };
- TG_FORCE_INLINE float64 interpolateToF(int a, int b, float64 b_ratio) {
- return a + float64(b - a) * b_ratio;
- }
- TG_FORCE_INLINE int interpolate(int a, int b, float64 b_ratio) {
- return base::SafeRound(interpolateToF(a, b, b_ratio));
- }
- #ifdef ARCH_CPU_32_BITS
- #define SHIFTED_USE_32BIT
- #endif // ARCH_CPU_32_BITS
- #ifdef SHIFTED_USE_32BIT
- using ShiftedMultiplier = uint32;
- struct Shifted {
- Shifted() = default;
- Shifted(uint32 low, uint32 high) : low(low), high(high) {
- }
- uint32 low = 0;
- uint32 high = 0;
- };
- TG_FORCE_INLINE Shifted operator+(Shifted a, Shifted b) {
- return Shifted(a.low + b.low, a.high + b.high);
- }
- TG_FORCE_INLINE Shifted operator*(Shifted shifted, ShiftedMultiplier multiplier) {
- return Shifted(shifted.low * multiplier, shifted.high * multiplier);
- }
- TG_FORCE_INLINE Shifted operator*(ShiftedMultiplier multiplier, Shifted shifted) {
- return Shifted(shifted.low * multiplier, shifted.high * multiplier);
- }
- TG_FORCE_INLINE Shifted shifted(uint32 components) {
- return Shifted(
- (components & 0x000000FFU) | ((components & 0x0000FF00U) << 8),
- ((components & 0x00FF0000U) >> 16) | ((components & 0xFF000000U) >> 8));
- }
- TG_FORCE_INLINE uint32 unshifted(Shifted components) {
- return ((components.low & 0x0000FF00U) >> 8)
- | ((components.low & 0xFF000000U) >> 16)
- | ((components.high & 0x0000FF00U) << 8)
- | (components.high & 0xFF000000U);
- }
- TG_FORCE_INLINE Shifted reshifted(Shifted components) {
- return Shifted((components.low >> 8) & 0x00FF00FFU, (components.high >> 8) & 0x00FF00FFU);
- }
- TG_FORCE_INLINE Shifted shifted(QColor color) {
- // Make it premultiplied.
- auto alpha = static_cast<uint32>((color.alpha() & 0xFF) + 1);
- auto components = Shifted(static_cast<uint32>(color.blue() & 0xFF) | (static_cast<uint32>(color.green() & 0xFF) << 16),
- static_cast<uint32>(color.red() & 0xFF) | (static_cast<uint32>(255) << 16));
- return reshifted(components * alpha);
- }
- TG_FORCE_INLINE uint32 getPremultiplied(QColor color) {
- // Make it premultiplied.
- auto alpha = static_cast<uint32>((color.alpha() & 0xFF) + 1);
- auto components = Shifted(static_cast<uint32>(color.blue() & 0xFF) | (static_cast<uint32>(color.green() & 0xFF) << 16),
- static_cast<uint32>(color.red() & 0xFF) | (static_cast<uint32>(255) << 16));
- return unshifted(components * alpha);
- }
- TG_FORCE_INLINE uint32 getAlpha(Shifted components) {
- return (components.high & 0x00FF0000U) >> 16;
- }
- TG_FORCE_INLINE Shifted non_premultiplied(QColor color) {
- return Shifted(static_cast<uint32>(color.blue() & 0xFF) | (static_cast<uint32>(color.green() & 0xFF) << 16),
- static_cast<uint32>(color.red() & 0xFF) | (static_cast<uint32>(color.alpha() & 0xFF) << 16));
- }
- TG_FORCE_INLINE QColor color(QColor a, QColor b, float64 b_ratio) {
- auto bOpacity = std::clamp(interpolate(0, 255, b_ratio), 0, 255) + 1;
- auto aOpacity = (256 - bOpacity);
- auto components = (non_premultiplied(a) * aOpacity + non_premultiplied(b) * bOpacity);
- return {
- static_cast<int>((components.high >> 8) & 0xFF),
- static_cast<int>((components.low >> 24) & 0xFF),
- static_cast<int>((components.low >> 8) & 0xFF),
- static_cast<int>((components.high >> 24) & 0xFF),
- };
- }
- #else // SHIFTED_USE_32BIT
- using ShiftedMultiplier = uint64;
- struct Shifted {
- Shifted() = default;
- Shifted(uint32 value) : value(value) {
- }
- Shifted(uint64 value) : value(value) {
- }
- uint64 value = 0;
- };
- TG_FORCE_INLINE Shifted operator+(Shifted a, Shifted b) {
- return Shifted(a.value + b.value);
- }
- TG_FORCE_INLINE Shifted operator*(Shifted shifted, ShiftedMultiplier multiplier) {
- return Shifted(shifted.value * multiplier);
- }
- TG_FORCE_INLINE Shifted operator*(ShiftedMultiplier multiplier, Shifted shifted) {
- return Shifted(shifted.value * multiplier);
- }
- TG_FORCE_INLINE Shifted shifted(uint32 components) {
- auto wide = static_cast<uint64>(components);
- return (wide & 0x00000000000000FFULL)
- | ((wide & 0x000000000000FF00ULL) << 8)
- | ((wide & 0x0000000000FF0000ULL) << 16)
- | ((wide & 0x00000000FF000000ULL) << 24);
- }
- TG_FORCE_INLINE uint32 unshifted(Shifted components) {
- return static_cast<uint32>((components.value & 0x000000000000FF00ULL) >> 8)
- | static_cast<uint32>((components.value & 0x00000000FF000000ULL) >> 16)
- | static_cast<uint32>((components.value & 0x0000FF0000000000ULL) >> 24)
- | static_cast<uint32>((components.value & 0xFF00000000000000ULL) >> 32);
- }
- TG_FORCE_INLINE Shifted reshifted(Shifted components) {
- return (components.value >> 8) & 0x00FF00FF00FF00FFULL;
- }
- TG_FORCE_INLINE Shifted shifted(QColor color) {
- // Make it premultiplied.
- auto alpha = static_cast<uint64>((color.alpha() & 0xFF) + 1);
- auto components = static_cast<uint64>(color.blue() & 0xFF)
- | (static_cast<uint64>(color.green() & 0xFF) << 16)
- | (static_cast<uint64>(color.red() & 0xFF) << 32)
- | (static_cast<uint64>(255) << 48);
- return reshifted(components * alpha);
- }
- TG_FORCE_INLINE uint32 getPremultiplied(QColor color) {
- // Make it premultiplied.
- auto alpha = static_cast<uint64>((color.alpha() & 0xFF) + 1);
- auto components = static_cast<uint64>(color.blue() & 0xFF)
- | (static_cast<uint64>(color.green() & 0xFF) << 16)
- | (static_cast<uint64>(color.red() & 0xFF) << 32)
- | (static_cast<uint64>(255) << 48);
- return unshifted(components * alpha);
- }
- TG_FORCE_INLINE uint32 getAlpha(Shifted components) {
- return (components.value & 0x00FF000000000000ULL) >> 48;
- }
- TG_FORCE_INLINE Shifted non_premultiplied(QColor color) {
- return static_cast<uint64>(color.blue() & 0xFF)
- | (static_cast<uint64>(color.green() & 0xFF) << 16)
- | (static_cast<uint64>(color.red() & 0xFF) << 32)
- | (static_cast<uint64>(color.alpha() & 0xFF) << 48);
- }
- TG_FORCE_INLINE QColor color(QColor a, QColor b, float64 b_ratio) {
- auto bOpacity = std::clamp(interpolate(0, 255, b_ratio), 0, 255) + 1;
- auto aOpacity = (256 - bOpacity);
- auto components = (non_premultiplied(a) * aOpacity + non_premultiplied(b) * bOpacity);
- return {
- static_cast<int>((components.value >> 40) & 0xFF),
- static_cast<int>((components.value >> 24) & 0xFF),
- static_cast<int>((components.value >> 8) & 0xFF),
- static_cast<int>((components.value >> 56) & 0xFF),
- };
- }
- #endif // SHIFTED_USE_32BIT
- TG_FORCE_INLINE QColor color(style::color a, QColor b, float64 b_ratio) {
- return color(a->c, b, b_ratio);
- }
- TG_FORCE_INLINE QColor color(QColor a, style::color b, float64 b_ratio) {
- return color(a, b->c, b_ratio);
- }
- TG_FORCE_INLINE QColor color(style::color a, style::color b, float64 b_ratio) {
- return color(a->c, b->c, b_ratio);
- }
- TG_FORCE_INLINE QPen pen(QColor a, QColor b, float64 b_ratio) {
- return color(a, b, b_ratio);
- }
- TG_FORCE_INLINE QPen pen(style::color a, QColor b, float64 b_ratio) {
- return (b_ratio > 0) ? pen(a->c, b, b_ratio) : a;
- }
- TG_FORCE_INLINE QPen pen(QColor a, style::color b, float64 b_ratio) {
- return (b_ratio < 1) ? pen(a, b->c, b_ratio) : b;
- }
- TG_FORCE_INLINE QPen pen(style::color a, style::color b, float64 b_ratio) {
- return (b_ratio > 0) ? ((b_ratio < 1) ? pen(a->c, b->c, b_ratio) : b) : a;
- }
- TG_FORCE_INLINE QBrush brush(QColor a, QColor b, float64 b_ratio) {
- return color(a, b, b_ratio);
- }
- TG_FORCE_INLINE QBrush brush(style::color a, QColor b, float64 b_ratio) {
- return (b_ratio > 0) ? brush(a->c, b, b_ratio) : a;
- }
- TG_FORCE_INLINE QBrush brush(QColor a, style::color b, float64 b_ratio) {
- return (b_ratio < 1) ? brush(a, b->c, b_ratio) : b;
- }
- TG_FORCE_INLINE QBrush brush(style::color a, style::color b, float64 b_ratio) {
- return (b_ratio > 0) ? ((b_ratio < 1) ? brush(a->c, b->c, b_ratio) : b) : a;
- }
- TG_FORCE_INLINE QColor with_alpha(QColor color, float64 alpha) {
- color.setAlphaF(color.alphaF() * alpha);
- return color;
- }
- template <int N>
- QPainterPath interpolate(QPointF (&from)[N], QPointF (&to)[N], float64 k) {
- static_assert(N > 1, "Wrong points count in path!");
- auto from_coef = 1. - k, to_coef = k;
- QPainterPath result;
- auto x = from[0].x() * from_coef + to[0].x() * to_coef;
- auto y = from[0].y() * from_coef + to[0].y() * to_coef;
- result.moveTo(x, y);
- for (int i = 1; i != N; ++i) {
- result.lineTo(from[i].x() * from_coef + to[i].x() * to_coef, from[i].y() * from_coef + to[i].y() * to_coef);
- }
- result.lineTo(x, y);
- return result;
- }
- template <int N>
- QPainterPath path(QPointF (&from)[N]) {
- static_assert(N > 1, "Wrong points count in path!");
- QPainterPath result;
- auto x = from[0].x();
- auto y = from[0].y();
- result.moveTo(x, y);
- for (int i = 1; i != N; ++i) {
- result.lineTo(from[i].x(), from[i].y());
- }
- result.lineTo(x, y);
- return result;
- }
- rpl::producer<bool> Disables();
- bool Disabled();
- void SetDisabled(bool disabled);
- int SlowMultiplier();
- void SetSlowMultiplier(int multiplier); // 1 - default, 10 - slow x10.
- void DrawStaticLoading(
- QPainter &p,
- QRectF rect,
- float64 stroke,
- QPen pen,
- QBrush brush = Qt::NoBrush);
- class continuous_value {
- public:
- continuous_value() = default;
- continuous_value(float64 duration) : _duration(duration) {
- }
- void start(float64 to, float64 duration) {
- _to = to;
- _delta = (_to - _cur) / duration;
- }
- void start(float64 to) {
- start(to, _duration);
- }
- void reset() {
- _to = _cur = _delta = 0.;
- }
- float64 current() const {
- return _cur;
- }
- float64 to() const {
- return _to;
- }
- float64 delta() const {
- return _delta;
- }
- void update(crl::time dt, Fn<void(float64 &)> &&callback = nullptr) {
- if (_to != _cur) {
- _cur += _delta * dt;
- if ((_to != _cur) && ((_delta > 0) == (_cur > _to))) {
- _cur = _to;
- }
- if (callback) {
- callback(_cur);
- }
- }
- }
- private:
- float64 _duration = 0.;
- float64 _to = 0.;
- float64 _cur = 0.;
- float64 _delta = 0.;
- };
- } // namespace anim
|