lib_ui/ui/effects/gradient.h

244 lines
5.7 KiB
C++

// 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 "base/flat_map.h"
#include "ui/effects/animation_value.h"
#include <QtGui/QLinearGradient>
#include <QtGui/QRadialGradient>
namespace anim {
struct gradient_colors {
explicit gradient_colors(QColor color) {
stops.push_back({ 0., color });
stops.push_back({ 1., color });
}
explicit gradient_colors(std::vector<QColor> colors) {
if (colors.size() == 1) {
gradient_colors(colors.front());
return;
}
const auto last = float(colors.size() - 1);
for (auto i = 0; i < colors.size(); i++) {
stops.push_back({ i / last, std::move(colors[i]) });
}
}
explicit gradient_colors(QGradientStops colors)
: stops(std::move(colors)) {
}
QGradientStops stops;
};
namespace details {
template <typename T, typename Derived>
class gradients {
public:
gradients(base::flat_map<T, std::vector<QColor>> colors) {
Expects(colors.size() > 0);
for (const auto &[key, value] : colors) {
auto c = gradient_colors(std::move(value));
_gradients.emplace(key, gradient_with_stops(std::move(c.stops)));
}
}
gradients(base::flat_map<T, gradient_colors> colors) {
Expects(colors.size() > 0);
for (const auto &[key, c] : colors) {
_gradients.emplace(key, gradient_with_stops(std::move(c.stops)));
}
}
QGradient gradient(T state1, T state2, float64 b_ratio) const {
if (b_ratio == 0.) {
return _gradients.find(state1)->second;
} else if (b_ratio == 1.) {
return _gradients.find(state2)->second;
}
auto gradient = empty_gradient();
const auto gradient1 = _gradients.find(state1);
const auto gradient2 = _gradients.find(state2);
Assert(gradient1 != end(_gradients));
Assert(gradient2 != end(_gradients));
const auto stopsFrom = gradient1->second.stops();
const auto stopsTo = gradient2->second.stops();
if ((stopsFrom.size() == stopsTo.size())
&& ranges::equal(
stopsFrom,
stopsTo,
ranges::equal_to(),
&QGradientStop::first,
&QGradientStop::first)) {
const auto size = stopsFrom.size();
const auto &p = b_ratio;
for (auto i = 0; i < size; i++) {
auto c = color(stopsFrom[i].second, stopsTo[i].second, p);
gradient.setColorAt(stopsTo[i].first, std::move(c));
}
return gradient;
}
const auto invert = (stopsFrom.size() > stopsTo.size());
if (invert) {
b_ratio = 1. - b_ratio;
}
const auto &stops1 = invert ? stopsTo : stopsFrom;
const auto &stops2 = invert ? stopsFrom : stopsTo;
const auto size1 = stops1.size();
const auto size2 = stops2.size();
for (auto i = 0; i < size1; i++) {
const auto point1 = stops1[i].first;
const auto previousPoint1 = i ? stops1[i - 1].first : -1.;
for (auto n = 0; n < size2; n++) {
const auto point2 = stops2[n].first;
if ((point2 <= previousPoint1) || (point2 > point1)) {
continue;
}
const auto color2 = stops2[n].second;
QColor result;
if (point2 < point1) {
const auto pointRatio2 = (point2 - previousPoint1)
/ (point1 - previousPoint1);
const auto color1 = color(
stops1[i - 1].second,
stops1[i].second,
pointRatio2);
result = color(color1, color2, b_ratio);
} else {
// point2 == point1
result = color(stops1[i].second, color2, b_ratio);
}
gradient.setColorAt(point2, std::move(result));
}
}
return gradient;
}
protected:
void cache_gradients() {
auto copy = std::move(_gradients);
for (const auto &[key, value] : copy) {
_gradients.emplace(key, gradient_with_stops(value.stops()));
}
}
private:
QGradient empty_gradient() const {
return static_cast<const Derived*>(this)->empty_gradient();
}
QGradient gradient_with_stops(QGradientStops stops) const {
auto gradient = empty_gradient();
gradient.setStops(std::move(stops));
return gradient;
}
base::flat_map<T, QGradient> _gradients;
};
} // namespace details
template <typename T>
class linear_gradients final
: public details::gradients<T, linear_gradients<T>> {
using parent = details::gradients<T, linear_gradients<T>>;
public:
linear_gradients(
base::flat_map<T, std::vector<QColor>> colors,
QPointF point1,
QPointF point2)
: parent(std::move(colors)) {
set_points(point1, point2);
}
linear_gradients(
base::flat_map<T, gradient_colors> colors,
QPointF point1,
QPointF point2)
: parent(std::move(colors)) {
set_points(point1, point2);
}
void set_points(QPointF point1, QPointF point2) {
if (_point1 == point1 && _point2 == point2) {
return;
}
_point1 = point1;
_point2 = point2;
parent::cache_gradients();
}
private:
friend class details::gradients<T, linear_gradients<T>>;
QGradient empty_gradient() const {
return QLinearGradient(_point1, _point2);
}
QPointF _point1;
QPointF _point2;
};
template <typename T>
class radial_gradients final
: public details::gradients<T, radial_gradients<T>> {
using parent = details::gradients<T, radial_gradients<T>>;
public:
radial_gradients(
base::flat_map<T, std::vector<QColor>> colors,
QPointF center,
float radius)
: parent(std::move(colors)) {
set_points(center, radius);
}
radial_gradients(
base::flat_map<T, gradient_colors> colors,
QPointF center,
float radius)
: parent(std::move(colors)) {
set_points(center, radius);
}
void set_points(QPointF center, float radius) {
if (_center == center && _radius == radius) {
return;
}
_center = center;
_radius = radius;
parent::cache_gradients();
}
private:
friend class details::gradients<T, radial_gradients<T>>;
QGradient empty_gradient() const {
return QRadialGradient(_center, _radius);
}
QPointF _center;
float _radius = 0.;
};
} // namespace anim