mirror of https://github.com/procxx/kepka.git
Finish animations disabling.
Replace some infinite animations with static layouts.
This commit is contained in:
parent
f68466b072
commit
1ffbec0215
|
@ -253,7 +253,7 @@ void ProxyRow::updateFields(View &&view) {
|
|||
}
|
||||
|
||||
void ProxyRow::step_radial(TimeMs ms, bool timer) {
|
||||
if (timer) {
|
||||
if (timer && !anim::Disabled()) {
|
||||
update();
|
||||
}
|
||||
}
|
||||
|
@ -375,13 +375,20 @@ void ProxyRow::paintCheck(Painter &p, TimeMs ms) {
|
|||
p.setPen(pen);
|
||||
p.setBrush(_st->bg);
|
||||
const auto rect = rtlrect(QRectF(left, top, _st->diameter, _st->diameter).marginsRemoved(QMarginsF(_st->thickness / 2., _st->thickness / 2., _st->thickness / 2., _st->thickness / 2.)), outerWidth);
|
||||
if (loading.arcLength < FullArcLength) {
|
||||
if (_progress && loading.shown > 0 && anim::Disabled()) {
|
||||
anim::DrawStaticLoading(
|
||||
p,
|
||||
rect,
|
||||
_st->thickness,
|
||||
pen.color(),
|
||||
_st->bg);
|
||||
} else if (loading.arcLength < FullArcLength) {
|
||||
p.drawArc(rect, loading.arcFrom, loading.arcLength);
|
||||
} else {
|
||||
p.drawEllipse(rect);
|
||||
}
|
||||
|
||||
if (toggled > 0) {
|
||||
if (toggled > 0 && (!_progress || !anim::Disabled())) {
|
||||
p.setPen(Qt::NoPen);
|
||||
p.setBrush(anim::brush(_st->untoggledFg, _st->toggledFg, toggled * set));
|
||||
|
||||
|
|
|
@ -186,7 +186,7 @@ void LocalStorageBox::Row::toggleProgress(bool shown) {
|
|||
}
|
||||
|
||||
void LocalStorageBox::Row::step_radial(TimeMs ms, bool timer) {
|
||||
if (timer) {
|
||||
if (timer && !anim::Disabled()) {
|
||||
RpWidget::update();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1204,6 +1204,9 @@ void StickersBox::Inner::leaveToChildEvent(QEvent *e, QWidget *child) {
|
|||
}
|
||||
|
||||
void StickersBox::Inner::step_shifting(TimeMs ms, bool timer) {
|
||||
if (anim::Disabled()) {
|
||||
ms += st::stickersRowDuration;
|
||||
}
|
||||
auto animating = false;
|
||||
auto updateMin = -1;
|
||||
auto updateMax = 0;
|
||||
|
|
|
@ -653,6 +653,9 @@ void StickersListWidget::Footer::paintSetIcon(
|
|||
}
|
||||
|
||||
void StickersListWidget::Footer::step_icons(TimeMs ms, bool timer) {
|
||||
if (anim::Disabled()) {
|
||||
ms += st::stickerIconMove;
|
||||
}
|
||||
if (_iconsStartAnim) {
|
||||
auto dt = (ms - _iconsStartAnim) / float64(st::stickerIconMove);
|
||||
if (dt >= 1) {
|
||||
|
|
|
@ -1000,6 +1000,10 @@ bool DialogsInner::updateReorderPinned(QPoint localPosition) {
|
|||
}
|
||||
|
||||
void DialogsInner::step_pinnedShifting(TimeMs ms, bool timer) {
|
||||
if (anim::Disabled()) {
|
||||
ms += st::stickersRowDuration;
|
||||
}
|
||||
|
||||
auto animating = false;
|
||||
auto updateMin = -1;
|
||||
auto updateMax = 0;
|
||||
|
|
|
@ -632,7 +632,7 @@ bool History::updateSendActionNeedsAnimating(TimeMs ms, bool force) {
|
|||
}
|
||||
}
|
||||
auto result = (!_typing.isEmpty() || !_sendActions.isEmpty());
|
||||
if (changed || result) {
|
||||
if (changed || (result && !anim::Disabled())) {
|
||||
App::histories().sendActionAnimationUpdated().notify({
|
||||
this,
|
||||
_sendActionAnimation.width(),
|
||||
|
|
|
@ -616,6 +616,9 @@ void ReplyKeyboard::startAnimation(int i, int j, int direction) {
|
|||
}
|
||||
|
||||
void ReplyKeyboard::step_selected(TimeMs ms, bool timer) {
|
||||
if (anim::Disabled()) {
|
||||
ms += st::botKbDuration;
|
||||
}
|
||||
for (auto i = _animations.begin(); i != _animations.end();) {
|
||||
const auto index = std::abs(i->first) - 1;
|
||||
const auto row = (index / MatrixRowShift);
|
||||
|
|
|
@ -205,10 +205,18 @@ void HistoryFileMedia::setStatusSize(int newSize, int fullSize, int duration, qi
|
|||
}
|
||||
|
||||
void HistoryFileMedia::step_radial(TimeMs ms, bool timer) {
|
||||
const auto updateRadial = [&] {
|
||||
return _animation->radial.update(
|
||||
dataProgress(),
|
||||
dataFinished(),
|
||||
ms);
|
||||
};
|
||||
if (timer) {
|
||||
Auth().data().requestViewRepaint(_parent);
|
||||
if (!anim::Disabled() || updateRadial()) {
|
||||
Auth().data().requestViewRepaint(_parent);
|
||||
}
|
||||
} else {
|
||||
_animation->radial.update(dataProgress(), dataFinished(), ms);
|
||||
updateRadial();
|
||||
if (!_animation->radial.animating()) {
|
||||
checkAnimationFinished();
|
||||
}
|
||||
|
@ -1848,6 +1856,9 @@ QMargins HistoryDocument::bubbleMargins() const {
|
|||
}
|
||||
|
||||
void HistoryDocument::step_voiceProgress(float64 ms, bool timer) {
|
||||
if (anim::Disabled()) {
|
||||
ms += (2 * AudioVoiceMsgUpdateView);
|
||||
}
|
||||
if (auto voice = Get<HistoryDocumentVoice>()) {
|
||||
if (voice->_playback) {
|
||||
float64 dt = ms / (2 * AudioVoiceMsgUpdateView);
|
||||
|
|
|
@ -2917,7 +2917,7 @@ void HistoryWidget::saveEditMsg() {
|
|||
|
||||
const auto textWithTags = _field->getTextWithAppliedMarkdown();
|
||||
const auto prepareFlags = Ui::ItemTextOptions(
|
||||
_history,
|
||||
_history,
|
||||
Auth().user()).flags;
|
||||
auto sending = TextWithEntities();
|
||||
auto left = TextWithEntities { textWithTags.text, ConvertTextTagsToEntities(textWithTags.tags) };
|
||||
|
@ -3232,14 +3232,16 @@ void HistoryWidget::unreadMentionsAnimationFinish() {
|
|||
}
|
||||
|
||||
void HistoryWidget::step_recording(float64 ms, bool timer) {
|
||||
float64 dt = ms / AudioVoiceMsgUpdateView;
|
||||
const auto dt = anim::Disabled() ? 1. : (ms / AudioVoiceMsgUpdateView);
|
||||
if (dt >= 1) {
|
||||
_a_recording.stop();
|
||||
a_recordingLevel.finish();
|
||||
} else {
|
||||
a_recordingLevel.update(dt, anim::linear);
|
||||
}
|
||||
if (timer) update(_attachToggle->geometry());
|
||||
if (timer && !anim::Disabled()) {
|
||||
update(_attachToggle->geometry());
|
||||
}
|
||||
}
|
||||
|
||||
void HistoryWidget::chooseAttach() {
|
||||
|
|
|
@ -152,7 +152,7 @@ void TopBarWidget::updateConnectingState() {
|
|||
}
|
||||
|
||||
void TopBarWidget::step_connecting(TimeMs ms, bool timer) {
|
||||
if (timer) {
|
||||
if (timer && !anim::Disabled()) {
|
||||
update();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -326,11 +326,19 @@ bool Gif::isRadialAnimation(TimeMs ms) const {
|
|||
}
|
||||
|
||||
void Gif::step_radial(TimeMs ms, bool timer) {
|
||||
const auto document = getShownDocument();
|
||||
const auto updateRadial = [&] {
|
||||
return _animation->radial.update(
|
||||
document->progress(),
|
||||
!document->loading() || document->loaded(),
|
||||
ms);
|
||||
};
|
||||
if (timer) {
|
||||
update();
|
||||
if (!anim::Disabled() || updateRadial()) {
|
||||
update();
|
||||
}
|
||||
} else {
|
||||
DocumentData *document = getShownDocument();
|
||||
_animation->radial.update(document->progress(), !document->loading() || document->loaded(), ms);
|
||||
updateRadial();
|
||||
if (!_animation->radial.animating() && document->loaded()) {
|
||||
_animation.reset();
|
||||
}
|
||||
|
@ -819,13 +827,18 @@ void File::thumbAnimationCallback() {
|
|||
}
|
||||
|
||||
void File::step_radial(TimeMs ms, bool timer) {
|
||||
if (timer) {
|
||||
update();
|
||||
} else {
|
||||
_animation->radial.update(
|
||||
const auto updateRadial = [&] {
|
||||
return _animation->radial.update(
|
||||
_document->progress(),
|
||||
!_document->loading() || _document->loaded(),
|
||||
ms);
|
||||
};
|
||||
if (timer) {
|
||||
if (!anim::Disabled() || updateRadial()) {
|
||||
update();
|
||||
}
|
||||
} else {
|
||||
updateRadial();
|
||||
if (!_animation->radial.animating()) {
|
||||
checkAnimationFinished();
|
||||
}
|
||||
|
@ -1333,11 +1346,19 @@ bool Game::isRadialAnimation(TimeMs ms) const {
|
|||
}
|
||||
|
||||
void Game::step_radial(TimeMs ms, bool timer) {
|
||||
const auto document = getResultDocument();
|
||||
const auto updateRadial = [&] {
|
||||
return _radial->update(
|
||||
document->progress(),
|
||||
!document->loading() || document->loaded(),
|
||||
ms);
|
||||
};
|
||||
if (timer) {
|
||||
update();
|
||||
if (!anim::Disabled() || updateRadial()) {
|
||||
update();
|
||||
}
|
||||
} else {
|
||||
auto document = getResultDocument();
|
||||
_radial->update(document->progress(), !document->loading() || document->loaded(), ms);
|
||||
updateRadial();
|
||||
if (!_radial->animating() && document->loaded()) {
|
||||
_radial.reset();
|
||||
}
|
||||
|
|
|
@ -94,7 +94,7 @@ void Playback::setValue(float64 value, bool animated) {
|
|||
}
|
||||
|
||||
void Playback::step_value(float64 ms, bool timer) {
|
||||
auto dt = ms / kPlaybackAnimationDurationMs;
|
||||
auto dt = anim::Disabled() ? 1. : (ms / kPlaybackAnimationDurationMs);
|
||||
if (dt >= 1.) {
|
||||
_a_value.stop();
|
||||
a_value.finish();
|
||||
|
|
|
@ -480,8 +480,11 @@ auto MediaView::computeOverviewType() const
|
|||
}
|
||||
|
||||
void MediaView::step_state(TimeMs ms, bool timer) {
|
||||
if (anim::Disabled()) {
|
||||
ms += st::mediaviewShowDuration + st::mediaviewHideDuration;
|
||||
}
|
||||
bool result = false;
|
||||
for (Showing::iterator i = _animations.begin(); i != _animations.end();) {
|
||||
for (auto i = _animations.begin(); i != _animations.end();) {
|
||||
TimeMs start = i.value();
|
||||
switch (i.key()) {
|
||||
case OverLeftNav: update(_leftNav); break;
|
||||
|
@ -584,8 +587,11 @@ void MediaView::step_radial(TimeMs ms, bool timer) {
|
|||
return;
|
||||
}
|
||||
const auto wasAnimating = _radial.animating();
|
||||
_radial.update(radialProgress(), !radialLoading(), ms + radialTimeShift());
|
||||
if (timer && (wasAnimating || _radial.animating())) {
|
||||
const auto updated = _radial.update(
|
||||
radialProgress(),
|
||||
!radialLoading(),
|
||||
ms + radialTimeShift());
|
||||
if (timer && (wasAnimating || _radial.animating()) && (!anim::Disabled() || updated)) {
|
||||
update(radialRect());
|
||||
}
|
||||
const auto ready = _doc && _doc->loaded();
|
||||
|
|
|
@ -227,10 +227,15 @@ void RadialProgressItem::setLinks(ClickHandlerPtr &&openl, ClickHandlerPtr &&sav
|
|||
}
|
||||
|
||||
void RadialProgressItem::step_radial(TimeMs ms, bool timer) {
|
||||
const auto updateRadial = [&] {
|
||||
return _radial->update(dataProgress(), dataFinished(), ms);
|
||||
};
|
||||
if (timer) {
|
||||
Auth().data().requestItemRepaint(parent());
|
||||
if (!anim::Disabled() || updateRadial()) {
|
||||
Auth().data().requestItemRepaint(parent());
|
||||
}
|
||||
} else {
|
||||
_radial->update(dataProgress(), dataFinished(), ms);
|
||||
updateRadial();
|
||||
if (!_radial->animating()) {
|
||||
checkRadialFinished();
|
||||
}
|
||||
|
|
|
@ -203,11 +203,11 @@ TimeMs BackgroundRow::radialTimeShift() const {
|
|||
}
|
||||
|
||||
void BackgroundRow::step_radial(TimeMs ms, bool timer) {
|
||||
_radial.update(
|
||||
const auto updated = _radial.update(
|
||||
radialProgress(),
|
||||
!radialLoading(),
|
||||
ms + radialTimeShift());
|
||||
if (timer && _radial.animating()) {
|
||||
if (timer && _radial.animating() && (!anim::Disabled() || updated)) {
|
||||
rtlupdate(radialRect());
|
||||
}
|
||||
}
|
||||
|
|
|
@ -93,8 +93,10 @@ void stopManager() {
|
|||
Media::Clip::Finish();
|
||||
}
|
||||
|
||||
void registerClipManager(Media::Clip::Manager *manager) {
|
||||
manager->connect(manager, SIGNAL(callback(Media::Clip::Reader*,qint32,qint32)), _manager, SLOT(clipCallback(Media::Clip::Reader*,qint32,qint32)));
|
||||
void registerClipManager(not_null<Media::Clip::Manager*> manager) {
|
||||
Expects(_manager != nullptr);
|
||||
|
||||
_manager->registerClip(manager);
|
||||
}
|
||||
|
||||
bool Disabled() {
|
||||
|
@ -104,7 +106,35 @@ bool Disabled() {
|
|||
void SetDisabled(bool disabled) {
|
||||
AnimationsDisabled = disabled;
|
||||
if (disabled && _manager) {
|
||||
_manager->timeout();
|
||||
_manager->step();
|
||||
}
|
||||
}
|
||||
|
||||
void DrawStaticLoading(
|
||||
QPainter &p,
|
||||
QRectF rect,
|
||||
int stroke,
|
||||
QPen pen,
|
||||
QBrush brush) {
|
||||
PainterHighQualityEnabler hq(p);
|
||||
|
||||
p.setBrush(brush);
|
||||
pen.setWidthF(stroke);
|
||||
pen.setCapStyle(Qt::RoundCap);
|
||||
pen.setJoinStyle(Qt::RoundJoin);
|
||||
p.setPen(pen);
|
||||
p.drawEllipse(rect);
|
||||
|
||||
const auto center = rect.center();
|
||||
const auto first = QPointF(center.x(), rect.y() + 1.5 * stroke);
|
||||
const auto delta = center.y() - first.y();
|
||||
const auto second = QPointF(center.x() + delta * 2 / 3., center.y());
|
||||
if (delta > 0) {
|
||||
QPainterPath path;
|
||||
path.moveTo(first);
|
||||
path.lineTo(center);
|
||||
path.lineTo(second);
|
||||
p.drawPath(path);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -125,19 +155,19 @@ void BasicAnimation::stop() {
|
|||
_manager->stop(this);
|
||||
}
|
||||
|
||||
AnimationManager::AnimationManager() : _timer(this), _iterating(false) {
|
||||
AnimationManager::AnimationManager() : _timer(this) {
|
||||
_timer.setSingleShot(false);
|
||||
connect(&_timer, SIGNAL(timeout()), this, SLOT(timeout()));
|
||||
connect(&_timer, &QTimer::timeout, this, &AnimationManager::step);
|
||||
}
|
||||
|
||||
void AnimationManager::start(BasicAnimation *obj) {
|
||||
if (_iterating) {
|
||||
_starting.insert(obj);
|
||||
if (!_stopping.isEmpty()) {
|
||||
_stopping.remove(obj);
|
||||
if (!_stopping.empty()) {
|
||||
_stopping.erase(obj);
|
||||
}
|
||||
} else {
|
||||
if (_objects.isEmpty()) {
|
||||
if (_objects.empty()) {
|
||||
_timer.start(AnimationTimerDelta);
|
||||
}
|
||||
_objects.insert(obj);
|
||||
|
@ -147,8 +177,8 @@ void AnimationManager::start(BasicAnimation *obj) {
|
|||
void AnimationManager::stop(BasicAnimation *obj) {
|
||||
if (_iterating) {
|
||||
_stopping.insert(obj);
|
||||
if (!_starting.isEmpty()) {
|
||||
_starting.remove(obj);
|
||||
if (!_starting.empty()) {
|
||||
_starting.erase(obj);
|
||||
}
|
||||
} else {
|
||||
auto i = _objects.find(obj);
|
||||
|
@ -161,25 +191,33 @@ void AnimationManager::stop(BasicAnimation *obj) {
|
|||
}
|
||||
}
|
||||
|
||||
void AnimationManager::timeout() {
|
||||
void AnimationManager::registerClip(not_null<Media::Clip::Manager*> clip) {
|
||||
connect(
|
||||
clip,
|
||||
&Media::Clip::Manager::callback,
|
||||
this,
|
||||
&AnimationManager::clipCallback);
|
||||
}
|
||||
|
||||
void AnimationManager::step() {
|
||||
_iterating = true;
|
||||
auto ms = getms();
|
||||
for_const (auto object, _objects) {
|
||||
const auto ms = getms();
|
||||
for (const auto object : _objects) {
|
||||
if (!_stopping.contains(object)) {
|
||||
object->step(ms, true);
|
||||
}
|
||||
}
|
||||
_iterating = false;
|
||||
|
||||
if (!_starting.isEmpty()) {
|
||||
for_const (auto object, _starting) {
|
||||
_objects.insert(object);
|
||||
if (!_starting.empty()) {
|
||||
for (const auto object : _starting) {
|
||||
_objects.emplace(object);
|
||||
}
|
||||
_starting.clear();
|
||||
}
|
||||
if (!_stopping.isEmpty()) {
|
||||
for_const (auto object, _stopping) {
|
||||
_objects.remove(object);
|
||||
if (!_stopping.empty()) {
|
||||
for (const auto object : _stopping) {
|
||||
_objects.erase(object);
|
||||
}
|
||||
_stopping.clear();
|
||||
}
|
||||
|
@ -188,7 +226,13 @@ void AnimationManager::timeout() {
|
|||
}
|
||||
}
|
||||
|
||||
void AnimationManager::clipCallback(Media::Clip::Reader *reader, qint32 threadIndex, qint32 notification) {
|
||||
Media::Clip::Reader::callback(reader, threadIndex, Media::Clip::Notification(notification));
|
||||
void AnimationManager::clipCallback(
|
||||
Media::Clip::Reader *reader,
|
||||
qint32 threadIndex,
|
||||
qint32 notification) {
|
||||
Media::Clip::Reader::callback(
|
||||
reader,
|
||||
threadIndex,
|
||||
Media::Clip::Notification(notification));
|
||||
}
|
||||
|
||||
|
|
|
@ -9,6 +9,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
|
||||
#include <QtCore/QTimer>
|
||||
#include <QtGui/QColor>
|
||||
#include "base/binary_guard.h"
|
||||
#include "base/flat_set.h"
|
||||
|
||||
namespace Media {
|
||||
namespace Clip {
|
||||
|
@ -161,7 +163,7 @@ private:
|
|||
|
||||
void startManager();
|
||||
void stopManager();
|
||||
void registerClipManager(Media::Clip::Manager *manager);
|
||||
void registerClipManager(not_null<Media::Clip::Manager*> manager);
|
||||
|
||||
TG_FORCE_INLINE int interpolate(int a, int b, float64 b_ratio) {
|
||||
return qRound(a + float64(b - a) * b_ratio);
|
||||
|
@ -416,6 +418,13 @@ QPainterPath path(QPointF (&from)[N]) {
|
|||
bool Disabled();
|
||||
void SetDisabled(bool disabled);
|
||||
|
||||
void DrawStaticLoading(
|
||||
QPainter &p,
|
||||
QRectF rect,
|
||||
int stroke,
|
||||
QPen pen,
|
||||
QBrush brush = Qt::NoBrush);
|
||||
|
||||
};
|
||||
|
||||
class BasicAnimation;
|
||||
|
@ -662,12 +671,14 @@ private:
|
|||
struct Data {
|
||||
template <typename Lambda>
|
||||
Data(float64 from, Lambda updateCallback)
|
||||
: value(from, from)
|
||||
, a_animation(animation(this, &Data::step))
|
||||
, updateCallback(std::move(updateCallback)) {
|
||||
: value(from, from)
|
||||
, a_animation(animation(this, &Data::step))
|
||||
, updateCallback(std::move(updateCallback)) {
|
||||
}
|
||||
void step(float64 ms, bool timer) {
|
||||
auto dt = (ms >= duration || anim::Disabled()) ? 1. : (ms / duration);
|
||||
const auto dt = (ms >= duration || anim::Disabled())
|
||||
? 1.
|
||||
: (ms / duration);
|
||||
if (dt >= 1) {
|
||||
value.finish();
|
||||
a_animation.stop();
|
||||
|
@ -690,23 +701,23 @@ private:
|
|||
};
|
||||
|
||||
class AnimationManager : public QObject {
|
||||
Q_OBJECT
|
||||
|
||||
public:
|
||||
AnimationManager();
|
||||
|
||||
void start(BasicAnimation *obj);
|
||||
void stop(BasicAnimation *obj);
|
||||
|
||||
public slots:
|
||||
void timeout();
|
||||
|
||||
void clipCallback(Media::Clip::Reader *reader, qint32 threadIndex, qint32 notification);
|
||||
void registerClip(not_null<Media::Clip::Manager*> clip);
|
||||
void step();
|
||||
|
||||
private:
|
||||
using AnimatingObjects = OrderedSet<BasicAnimation*>;
|
||||
AnimatingObjects _objects, _starting, _stopping;
|
||||
void clipCallback(
|
||||
Media::Clip::Reader *reader,
|
||||
qint32 threadIndex,
|
||||
qint32 notification);
|
||||
|
||||
base::flat_set<BasicAnimation*> _objects, _starting, _stopping;
|
||||
QTimer _timer;
|
||||
bool _iterating;
|
||||
bool _iterating = false;
|
||||
|
||||
};
|
||||
|
|
|
@ -11,6 +11,7 @@ namespace Ui {
|
|||
namespace {
|
||||
|
||||
constexpr auto kPointCount = 12;
|
||||
constexpr auto kStaticLoadingValue = float64(-666);
|
||||
|
||||
//
|
||||
// 1 3
|
||||
|
@ -90,7 +91,26 @@ void transformLoadingCross(float64 loading, std::array<QPointF, kPointCount> &po
|
|||
|
||||
} // namespace
|
||||
|
||||
void CrossAnimation::paint(Painter &p, const style::CrossAnimation &st, style::color color, int x, int y, int outerWidth, float64 shown, float64 loading) {
|
||||
void CrossAnimation::paintStaticLoading(
|
||||
Painter &p,
|
||||
const style::CrossAnimation &st,
|
||||
style::color color,
|
||||
int x,
|
||||
int y,
|
||||
int outerWidth,
|
||||
float64 shown) {
|
||||
paint(p, st, color, x, y, outerWidth, shown, kStaticLoadingValue);
|
||||
}
|
||||
|
||||
void CrossAnimation::paint(
|
||||
Painter &p,
|
||||
const style::CrossAnimation &st,
|
||||
style::color color,
|
||||
int x,
|
||||
int y,
|
||||
int outerWidth,
|
||||
float64 shown,
|
||||
float64 loading) {
|
||||
PainterHighQualityEnabler hq(p);
|
||||
|
||||
auto sqrt2 = sqrt(2.);
|
||||
|
@ -117,7 +137,8 @@ void CrossAnimation::paint(Painter &p, const style::CrossAnimation &st, style::c
|
|||
} };
|
||||
auto pathDeleteSize = kPointCount;
|
||||
|
||||
auto loadingArcLength = 0;
|
||||
const auto staticLoading = (loading == kStaticLoadingValue);
|
||||
auto loadingArcLength = staticLoading ? FullArcLength : 0;
|
||||
if (loading > 0.) {
|
||||
transformLoadingCross(loading, pathDelete, pathDeleteSize);
|
||||
|
||||
|
@ -125,44 +146,50 @@ void CrossAnimation::paint(Painter &p, const style::CrossAnimation &st, style::c
|
|||
loadingArcLength = qRound(-loadingArc * 2 * FullArcLength);
|
||||
}
|
||||
|
||||
if (shown < 1.) {
|
||||
auto alpha = -(shown - 1.) * M_PI_2;
|
||||
auto cosalpha = cos(alpha);
|
||||
auto sinalpha = sin(alpha);
|
||||
auto shiftx = deleteLeft + (deleteWidth / 2.);
|
||||
auto shifty = deleteTop + (deleteHeight / 2.);
|
||||
for (auto &point : pathDelete) {
|
||||
auto x = point.x() - shiftx;
|
||||
auto y = point.y() - shifty;
|
||||
point.setX(shiftx + x * cosalpha - y * sinalpha);
|
||||
point.setY(shifty + y * cosalpha + x * sinalpha);
|
||||
if (!staticLoading) {
|
||||
if (shown < 1.) {
|
||||
auto alpha = -(shown - 1.) * M_PI_2;
|
||||
auto cosalpha = cos(alpha);
|
||||
auto sinalpha = sin(alpha);
|
||||
auto shiftx = deleteLeft + (deleteWidth / 2.);
|
||||
auto shifty = deleteTop + (deleteHeight / 2.);
|
||||
for (auto &point : pathDelete) {
|
||||
auto x = point.x() - shiftx;
|
||||
auto y = point.y() - shifty;
|
||||
point.setX(shiftx + x * cosalpha - y * sinalpha);
|
||||
point.setY(shifty + y * cosalpha + x * sinalpha);
|
||||
}
|
||||
}
|
||||
QPainterPath path;
|
||||
path.moveTo(pathDelete[0]);
|
||||
for (int i = 1; i != pathDeleteSize; ++i) {
|
||||
path.lineTo(pathDelete[i]);
|
||||
}
|
||||
path.lineTo(pathDelete[0]);
|
||||
p.fillPath(path, color);
|
||||
}
|
||||
QPainterPath path;
|
||||
path.moveTo(pathDelete[0]);
|
||||
for (int i = 1; i != pathDeleteSize; ++i) {
|
||||
path.lineTo(pathDelete[i]);
|
||||
}
|
||||
path.lineTo(pathDelete[0]);
|
||||
p.fillPath(path, color);
|
||||
|
||||
if (loadingArcLength != 0) {
|
||||
auto loadingArcStart = FullArcLength / 8;
|
||||
auto roundSkip = (st.size * (1 - sqrt2) + 2 * sqrt2 * deleteSkip + st.stroke) / 2;
|
||||
auto roundPart = QRectF(x + roundSkip, y + roundSkip, st.size - 2 * roundSkip, st.size - 2 * roundSkip);
|
||||
if (shown < 1.) {
|
||||
loadingArcStart -= qRound(-(shown - 1.) * FullArcLength / 4.);
|
||||
if (staticLoading) {
|
||||
anim::DrawStaticLoading(p, roundPart, st.stroke, color);
|
||||
} else {
|
||||
auto loadingArcStart = FullArcLength / 8;
|
||||
if (shown < 1.) {
|
||||
loadingArcStart -= qRound(-(shown - 1.) * FullArcLength / 4.);
|
||||
}
|
||||
if (loadingArcLength < 0) {
|
||||
loadingArcStart += loadingArcLength;
|
||||
loadingArcLength = -loadingArcLength;
|
||||
}
|
||||
|
||||
p.setBrush(Qt::NoBrush);
|
||||
auto pen = color->p;
|
||||
pen.setWidthF(st.stroke);
|
||||
pen.setCapStyle(Qt::RoundCap);
|
||||
p.setPen(pen);
|
||||
p.drawArc(roundPart, loadingArcStart, loadingArcLength);
|
||||
}
|
||||
p.setBrush(Qt::NoBrush);
|
||||
auto pen = color->p;
|
||||
pen.setWidthF(st.stroke);
|
||||
pen.setCapStyle(Qt::RoundCap);
|
||||
p.setPen(pen);
|
||||
if (loadingArcLength < 0) {
|
||||
loadingArcStart += loadingArcLength;
|
||||
loadingArcLength = -loadingArcLength;
|
||||
}
|
||||
p.drawArc(roundPart, loadingArcStart, loadingArcLength);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -13,7 +13,23 @@ namespace Ui {
|
|||
|
||||
class CrossAnimation {
|
||||
public:
|
||||
static void paint(Painter &p, const style::CrossAnimation &st, style::color color, int x, int y, int outerWidth, float64 shown, float64 loading = 0.);
|
||||
static void paint(
|
||||
Painter &p,
|
||||
const style::CrossAnimation &st,
|
||||
style::color color,
|
||||
int x,
|
||||
int y,
|
||||
int outerWidth,
|
||||
float64 shown,
|
||||
float64 loading = 0.);
|
||||
static void paintStaticLoading(
|
||||
Painter &p,
|
||||
const style::CrossAnimation &st,
|
||||
style::color color,
|
||||
int x,
|
||||
int y,
|
||||
int outerWidth,
|
||||
float64 shown);
|
||||
|
||||
};
|
||||
|
||||
|
|
|
@ -23,9 +23,10 @@ void RadialAnimation::start(float64 prg) {
|
|||
_animation.start();
|
||||
}
|
||||
|
||||
void RadialAnimation::update(float64 prg, bool finished, TimeMs ms) {
|
||||
auto iprg = qRound(qMax(prg, 0.0001) * AlmostFullArcLength);
|
||||
if (iprg != qRound(a_arcEnd.to())) {
|
||||
bool RadialAnimation::update(float64 prg, bool finished, TimeMs ms) {
|
||||
const auto iprg = qRound(qMax(prg, 0.0001) * AlmostFullArcLength);
|
||||
const auto result = (iprg != qRound(a_arcEnd.to()));
|
||||
if (result) {
|
||||
a_arcEnd.start(iprg);
|
||||
_lastStart = _lastTime;
|
||||
}
|
||||
|
@ -34,7 +35,12 @@ void RadialAnimation::update(float64 prg, bool finished, TimeMs ms) {
|
|||
auto dt = float64(ms - _lastStart);
|
||||
auto fulldt = float64(ms - _firstStart);
|
||||
_opacity = qMin(fulldt / st::radialDuration, 1.);
|
||||
if (!finished) {
|
||||
if (anim::Disabled()) {
|
||||
a_arcEnd.update(1., anim::linear);
|
||||
if (finished) {
|
||||
stop();
|
||||
}
|
||||
} else if (!finished) {
|
||||
a_arcEnd.update(1. - (st::radialDuration / (st::radialDuration + dt)), anim::linear);
|
||||
} else if (dt >= st::radialDuration) {
|
||||
a_arcEnd.update(1., anim::linear);
|
||||
|
@ -46,6 +52,7 @@ void RadialAnimation::update(float64 prg, bool finished, TimeMs ms) {
|
|||
}
|
||||
auto fromstart = fulldt / st::radialPeriod;
|
||||
a_arcStart.update(fromstart - std::floor(fromstart), anim::linear);
|
||||
return result;
|
||||
}
|
||||
|
||||
void RadialAnimation::stop() {
|
||||
|
@ -69,7 +76,9 @@ void RadialAnimation::draw(Painter &p, const QRect &inner, int32 thickness, styl
|
|||
p.setPen(pen);
|
||||
|
||||
auto len = MinArcLength + qRound(a_arcEnd.current());
|
||||
auto from = QuarterArcLength - qRound(a_arcStart.current()) - len;
|
||||
auto from = QuarterArcLength
|
||||
- len
|
||||
- (anim::Disabled() ? 0 : qRound(a_arcStart.current()));
|
||||
if (rtl()) {
|
||||
from = QuarterArcLength - (from - QuarterArcLength) - len;
|
||||
if (from < 0) from += FullArcLength;
|
||||
|
@ -104,6 +113,9 @@ void InfiniteRadialAnimation::start() {
|
|||
|
||||
void InfiniteRadialAnimation::stop() {
|
||||
const auto now = getms();
|
||||
if (anim::Disabled()) {
|
||||
_workFinished = now;
|
||||
}
|
||||
if (!_workFinished) {
|
||||
const auto zero = _workStarted - _st.sineDuration;
|
||||
const auto index = (now - zero + _st.sinePeriod - _st.sineShift)
|
||||
|
@ -138,26 +150,32 @@ void InfiniteRadialAnimation::draw(
|
|||
auto o = p.opacity();
|
||||
p.setOpacity(o * state.shown);
|
||||
|
||||
auto pen = _st.color->p;
|
||||
auto was = p.pen();
|
||||
pen.setWidth(_st.thickness);
|
||||
pen.setCapStyle(Qt::RoundCap);
|
||||
p.setPen(pen);
|
||||
const auto rect = rtlrect(
|
||||
position.x(),
|
||||
position.y(),
|
||||
size.width(),
|
||||
size.height(),
|
||||
outerWidth);
|
||||
const auto was = p.pen();
|
||||
const auto brush = p.brush();
|
||||
if (anim::Disabled()) {
|
||||
anim::DrawStaticLoading(p, rect, _st.thickness, _st.color);
|
||||
} else {
|
||||
auto pen = _st.color->p;
|
||||
pen.setWidth(_st.thickness);
|
||||
pen.setCapStyle(Qt::RoundCap);
|
||||
p.setPen(pen);
|
||||
|
||||
{
|
||||
PainterHighQualityEnabler hq(p);
|
||||
p.drawArc(
|
||||
rtlrect(
|
||||
position.x(),
|
||||
position.y(),
|
||||
size.width(),
|
||||
size.height(),
|
||||
outerWidth),
|
||||
state.arcFrom,
|
||||
state.arcLength);
|
||||
{
|
||||
PainterHighQualityEnabler hq(p);
|
||||
p.drawArc(
|
||||
rect,
|
||||
state.arcFrom,
|
||||
state.arcLength);
|
||||
}
|
||||
}
|
||||
|
||||
p.setPen(was);
|
||||
p.setBrush(brush);
|
||||
p.setOpacity(o);
|
||||
}
|
||||
|
||||
|
@ -173,6 +191,10 @@ auto InfiniteRadialAnimation::computeState() -> State {
|
|||
linear,
|
||||
FullArcLength };
|
||||
}
|
||||
if (anim::Disabled()) {
|
||||
const auto shown = 1.;
|
||||
return { 1., 0, FullArcLength };
|
||||
}
|
||||
const auto min = int(std::round(FullArcLength * _st.arcMin));
|
||||
const auto max = int(std::round(FullArcLength * _st.arcMax));
|
||||
if (now <= _workStarted) {
|
||||
|
|
|
@ -25,7 +25,7 @@ public:
|
|||
}
|
||||
|
||||
void start(float64 prg);
|
||||
void update(float64 prg, bool finished, TimeMs ms);
|
||||
bool update(float64 prg, bool finished, TimeMs ms);
|
||||
void stop();
|
||||
|
||||
void step(TimeMs ms);
|
||||
|
|
|
@ -190,6 +190,23 @@ bool SendActionAnimation::Impl::supports(Type type) const {
|
|||
return Implementations->value(type, &TypingAnimation::kMeta) == metaData();
|
||||
}
|
||||
|
||||
void SendActionAnimation::Impl::paint(
|
||||
Painter &p,
|
||||
style::color color,
|
||||
int x,
|
||||
int y,
|
||||
int outerWidth,
|
||||
TimeMs ms) {
|
||||
paintFrame(
|
||||
p,
|
||||
color,
|
||||
x,
|
||||
y,
|
||||
outerWidth,
|
||||
anim::Disabled() ? 0 : (qMax(ms - _started, 0LL) % _period));
|
||||
}
|
||||
|
||||
|
||||
void SendActionAnimation::start(Type type) {
|
||||
if (!_impl || !_impl->supports(type)) {
|
||||
_impl = createByType(type);
|
||||
|
|
|
@ -44,9 +44,13 @@ public:
|
|||
bool supports(Type type) const;
|
||||
|
||||
virtual int width() const = 0;
|
||||
void paint(Painter &p, style::color color, int x, int y, int outerWidth, TimeMs ms) {
|
||||
paintFrame(p, color, x, y, outerWidth, qMax(ms - _started, 0LL) % _period);
|
||||
}
|
||||
void paint(
|
||||
Painter &p,
|
||||
style::color color,
|
||||
int x,
|
||||
int y,
|
||||
int outerWidth,
|
||||
TimeMs ms);
|
||||
|
||||
virtual ~Impl() = default;
|
||||
|
||||
|
|
|
@ -182,28 +182,50 @@ void EmojiButton::paintEvent(QPaintEvent *e) {
|
|||
p.fillRect(e->rect(), st::historyComposeAreaBg);
|
||||
paintRipple(p, _st.rippleAreaPosition.x(), _st.rippleAreaPosition.y(), ms, _rippleOverride ? &(*_rippleOverride)->c : nullptr);
|
||||
|
||||
const auto over = isOver();
|
||||
const auto loadingState = _loading
|
||||
? _loading->computeState()
|
||||
: Ui::InfiniteRadialAnimation::State{ 0., 0, FullArcLength };
|
||||
p.setOpacity(1. - loadingState.shown);
|
||||
if (loadingState.shown < 1.) {
|
||||
p.setOpacity(1. - loadingState.shown);
|
||||
|
||||
auto over = isOver();
|
||||
auto icon = _iconOverride ? _iconOverride : &(over ? _st.iconOver : _st.icon);
|
||||
icon->paint(p, _st.iconPosition, width());
|
||||
auto icon = _iconOverride ? _iconOverride : &(over ? _st.iconOver : _st.icon);
|
||||
icon->paint(p, _st.iconPosition, width());
|
||||
|
||||
p.setOpacity(1.);
|
||||
auto pen = _colorOverride ? (*_colorOverride)->p : (over ? st::historyEmojiCircleFgOver : st::historyEmojiCircleFg)->p;
|
||||
pen.setWidth(st::historyEmojiCircleLine);
|
||||
pen.setCapStyle(Qt::RoundCap);
|
||||
p.setPen(pen);
|
||||
p.setBrush(Qt::NoBrush);
|
||||
p.setOpacity(1.);
|
||||
}
|
||||
|
||||
PainterHighQualityEnabler hq(p);
|
||||
QRect inner(QPoint((width() - st::historyEmojiCircle.width()) / 2, st::historyEmojiCircleTop), st::historyEmojiCircle);
|
||||
if (loadingState.arcLength < FullArcLength) {
|
||||
p.drawArc(inner, loadingState.arcFrom, loadingState.arcLength);
|
||||
const auto color = (_colorOverride
|
||||
? *_colorOverride
|
||||
: (over
|
||||
? st::historyEmojiCircleFgOver
|
||||
: st::historyEmojiCircleFg));
|
||||
if (_loading && anim::Disabled()) {
|
||||
anim::DrawStaticLoading(
|
||||
p,
|
||||
inner,
|
||||
st::historyEmojiCircleLine,
|
||||
color);
|
||||
} else {
|
||||
p.drawEllipse(inner);
|
||||
auto pen = color->p;
|
||||
pen.setWidth(st::historyEmojiCircleLine);
|
||||
pen.setCapStyle(Qt::RoundCap);
|
||||
p.setPen(pen);
|
||||
p.setBrush(Qt::NoBrush);
|
||||
|
||||
PainterHighQualityEnabler hq(p);
|
||||
if (loadingState.arcLength < FullArcLength) {
|
||||
p.drawArc(inner, loadingState.arcFrom, loadingState.arcLength);
|
||||
} else {
|
||||
p.drawEllipse(inner);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void EmojiButton::step_loading(TimeMs ms, bool timer) {
|
||||
if (timer && !anim::Disabled()) {
|
||||
update();
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -215,8 +237,10 @@ void EmojiButton::setLoading(bool loading) {
|
|||
}
|
||||
if (loading) {
|
||||
_loading->start();
|
||||
update();
|
||||
} else if (_loading) {
|
||||
_loading->stop();
|
||||
update();
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -65,11 +65,7 @@ protected:
|
|||
QPoint prepareRippleStartPosition() const override;
|
||||
|
||||
private:
|
||||
void step_loading(TimeMs ms, bool timer) {
|
||||
if (timer) {
|
||||
update();
|
||||
}
|
||||
}
|
||||
void step_loading(TimeMs ms, bool timer);
|
||||
|
||||
const style::IconButton &_st;
|
||||
|
||||
|
|
|
@ -179,7 +179,7 @@ private:
|
|||
|
||||
class PainterHighQualityEnabler {
|
||||
public:
|
||||
PainterHighQualityEnabler(Painter &p) : _painter(p) {
|
||||
PainterHighQualityEnabler(QPainter &p) : _painter(p) {
|
||||
static constexpr QPainter::RenderHint Hints[] = {
|
||||
QPainter::Antialiasing,
|
||||
QPainter::SmoothPixmapTransform,
|
||||
|
@ -187,8 +187,8 @@ public:
|
|||
QPainter::HighQualityAntialiasing
|
||||
};
|
||||
|
||||
auto hints = _painter.renderHints();
|
||||
for_const (auto hint, Hints) {
|
||||
const auto hints = _painter.renderHints();
|
||||
for (const auto hint : Hints) {
|
||||
if (!(hints & hint)) {
|
||||
_hints |= hint;
|
||||
}
|
||||
|
@ -197,8 +197,12 @@ public:
|
|||
_painter.setRenderHints(_hints);
|
||||
}
|
||||
}
|
||||
PainterHighQualityEnabler(const PainterHighQualityEnabler &other) = delete;
|
||||
PainterHighQualityEnabler &operator=(const PainterHighQualityEnabler &other) = delete;
|
||||
|
||||
PainterHighQualityEnabler(
|
||||
const PainterHighQualityEnabler &other) = delete;
|
||||
PainterHighQualityEnabler &operator=(
|
||||
const PainterHighQualityEnabler &other) = delete;
|
||||
|
||||
~PainterHighQualityEnabler() {
|
||||
if (_hints) {
|
||||
_painter.setRenderHints(_hints, false);
|
||||
|
@ -206,7 +210,7 @@ public:
|
|||
}
|
||||
|
||||
private:
|
||||
Painter &_painter;
|
||||
QPainter &_painter;
|
||||
QPainter::RenderHints _hints = 0;
|
||||
|
||||
};
|
||||
|
|
|
@ -538,8 +538,8 @@ CrossButton::CrossButton(QWidget *parent, const style::CrossButton &st) : Ripple
|
|||
void CrossButton::step_loading(TimeMs ms, bool timer) {
|
||||
if (stopLoadingAnimation(ms)) {
|
||||
_a_loading.stop();
|
||||
}
|
||||
if (timer) {
|
||||
update();
|
||||
} else if (timer && !anim::Disabled()) {
|
||||
update();
|
||||
}
|
||||
}
|
||||
|
@ -584,11 +584,30 @@ void CrossButton::paintEvent(QPaintEvent *e) {
|
|||
if (_a_loading.animating()) {
|
||||
if (stopLoadingAnimation(ms)) {
|
||||
_a_loading.stop();
|
||||
} else if (anim::Disabled()) {
|
||||
CrossAnimation::paintStaticLoading(
|
||||
p,
|
||||
_st.cross,
|
||||
over ? _st.crossFgOver : _st.crossFg,
|
||||
_st.crossPosition.x(),
|
||||
_st.crossPosition.y(),
|
||||
width(),
|
||||
shown);
|
||||
return;
|
||||
} else {
|
||||
loading = ((ms - _loadingStartMs) % _st.loadingPeriod) / float64(_st.loadingPeriod);
|
||||
loading = ((ms - _loadingStartMs) % _st.loadingPeriod)
|
||||
/ float64(_st.loadingPeriod);
|
||||
}
|
||||
}
|
||||
CrossAnimation::paint(p, _st.cross, over ? _st.crossFgOver : _st.crossFg, _st.crossPosition.x(), _st.crossPosition.y(), width(), shown, loading);
|
||||
CrossAnimation::paint(
|
||||
p,
|
||||
_st.cross,
|
||||
over ? _st.crossFgOver : _st.crossFg,
|
||||
_st.crossPosition.x(),
|
||||
_st.crossPosition.y(),
|
||||
width(),
|
||||
shown,
|
||||
loading);
|
||||
}
|
||||
|
||||
bool CrossButton::stopLoadingAnimation(TimeMs ms) {
|
||||
|
@ -617,6 +636,9 @@ void CrossButton::setLoadingAnimation(bool enabled) {
|
|||
_a_loading.stop();
|
||||
}
|
||||
}
|
||||
if (anim::Disabled()) {
|
||||
update();
|
||||
}
|
||||
}
|
||||
|
||||
void CrossButton::onStateChanged(State was, StateChangeSource source) {
|
||||
|
|
|
@ -393,6 +393,9 @@ void Widget::opacityAnimationCallback() {
|
|||
}
|
||||
|
||||
void Widget::step_shift(float64 ms, bool timer) {
|
||||
if (anim::Disabled()) {
|
||||
ms += st::notifyFastAnim;
|
||||
}
|
||||
float64 dt = ms / float64(st::notifyFastAnim);
|
||||
if (dt >= 1) {
|
||||
a_shift.finish();
|
||||
|
|
|
@ -59,7 +59,7 @@ void Progress::paintEvent(QPaintEvent *e) {
|
|||
}
|
||||
|
||||
void Progress::step(TimeMs ms, bool timer) {
|
||||
if (timer) {
|
||||
if (timer && !anim::Disabled()) {
|
||||
update();
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue