Use new animations engine in all Basic-s.

This commit is contained in:
John Preston 2019-04-01 21:44:54 +04:00
parent cd3c1c6dc0
commit a6e96f9a28
41 changed files with 479 additions and 487 deletions

View File

@ -400,7 +400,7 @@ BackgroundPreviewBox::BackgroundPreviewBox(
lang(lng_background_text2),
true))
, _paper(paper)
, _radial(animation(this, &BackgroundPreviewBox::step_radial)) {
, _radial([=](crl::time now) { return radialAnimationCallback(now); }) {
subscribe(Auth().downloaderTaskFinished(), [=] { update(); });
}
@ -549,13 +549,8 @@ void BackgroundPreviewBox::paintImage(Painter &p, crl::time ms) {
}
void BackgroundPreviewBox::paintRadial(Painter &p, crl::time ms) {
bool radial = false;
float64 radialOpacity = 0;
if (_radial.animating()) {
_radial.step(ms);
radial = _radial.animating();
radialOpacity = _radial.opacity();
}
const auto radial = _radial.animating();
const auto radialOpacity = radial ? _radial.opacity() : 0.;
if (!radial) {
return;
}
@ -623,7 +618,7 @@ void BackgroundPreviewBox::paintDate(Painter &p) {
p.drawText(bubbleLeft + st::msgServicePadding.left(), bubbleTop + st::msgServicePadding.top() + st::msgServiceFont->ascent, text);
}
void BackgroundPreviewBox::step_radial(crl::time ms, bool timer) {
void BackgroundPreviewBox::radialAnimationCallback(crl::time now) {
Expects(_paper.document() != nullptr);
const auto document = _paper.document();
@ -631,9 +626,8 @@ void BackgroundPreviewBox::step_radial(crl::time ms, bool timer) {
const auto updated = _radial.update(
document->progress(),
!document->loading(),
ms);
if (timer
&& (wasAnimating || _radial.animating())
now);
if ((wasAnimating || _radial.animating())
&& (!anim::Disabled() || updated)) {
update(radialRect());
}

View File

@ -50,7 +50,7 @@ private:
void apply();
void share();
void step_radial(crl::time ms, bool timer);
void radialAnimationCallback(crl::time now);
QRect radialRect() const;
void checkLoadedDocument();

View File

@ -52,7 +52,7 @@ protected:
private:
void setupControls(View &&view);
int countAvailableWidth() const;
void step_radial(crl::time ms, bool timer);
void radialAnimationCallback();
void paintCheck(Painter &p, crl::time ms);
void showMenu();
@ -209,7 +209,7 @@ void ProxyRow::updateFields(View &&view) {
if (state == State::Connecting) {
if (!_progress) {
_progress = std::make_unique<Ui::InfiniteRadialAnimation>(
animation(this, &ProxyRow::step_radial),
[=] { radialAnimationCallback(); },
st::proxyCheckingAnimation);
}
_progress->start();
@ -219,7 +219,7 @@ void ProxyRow::updateFields(View &&view) {
if (state == State::Checking) {
if (!_checking) {
_checking = std::make_unique<Ui::InfiniteRadialAnimation>(
animation(this, &ProxyRow::step_radial),
[=] { radialAnimationCallback(); },
st::proxyCheckingAnimation);
_checking->start();
}
@ -241,8 +241,8 @@ void ProxyRow::updateFields(View &&view) {
update();
}
void ProxyRow::step_radial(crl::time ms, bool timer) {
if (timer && !anim::Disabled()) {
void ProxyRow::radialAnimationCallback() {
if (!anim::Disabled()) {
update();
}
}
@ -323,28 +323,22 @@ void ProxyRow::paintEvent(QPaintEvent *e) {
auto statusLeft = left;
if (_checking) {
_checking->step(ms);
if (_checking) {
_checking->draw(
p,
{
st::proxyCheckingPosition.x() + statusLeft,
st::proxyCheckingPosition.y() + top
},
width());
statusLeft += st::proxyCheckingPosition.x()
+ st::proxyCheckingAnimation.size.width()
+ st::proxyCheckingSkip;
}
_checking->draw(
p,
{
st::proxyCheckingPosition.x() + statusLeft,
st::proxyCheckingPosition.y() + top
},
width());
statusLeft += st::proxyCheckingPosition.x()
+ st::proxyCheckingAnimation.size.width()
+ st::proxyCheckingSkip;
}
p.drawTextLeft(statusLeft, top, width(), status);
top += st::normalFont->height + st::proxyRowPadding.bottom();
}
void ProxyRow::paintCheck(Painter &p, crl::time ms) {
if (_progress) {
_progress->step(ms);
}
const auto loading = _progress
? _progress->computeState()
: Ui::RadialState{ 0., 0, FullArcLength };

View File

@ -139,7 +139,7 @@ protected:
private:
QString titleText(const Database::TaggedSummary &data) const;
QString sizeText(const Database::TaggedSummary &data) const;
void step_radial(crl::time ms, bool timer);
void radialAnimationCallback();
Fn<QString(size_type)> _titleFactory;
object_ptr<Ui::FlatLabel> _title;
@ -186,7 +186,7 @@ void LocalStorageBox::Row::toggleProgress(bool shown) {
_clearing.destroy();
} else if (!_progress) {
_progress = std::make_unique<Ui::InfiniteRadialAnimation>(
animation(this, &Row::step_radial),
[=] { radialAnimationCallback(); },
st::proxyCheckingAnimation);
_progress->start();
_clearing = object_ptr<Ui::FlatLabel>(
@ -201,8 +201,8 @@ void LocalStorageBox::Row::toggleProgress(bool shown) {
}
}
void LocalStorageBox::Row::step_radial(crl::time ms, bool timer) {
if (timer && !anim::Disabled()) {
void LocalStorageBox::Row::radialAnimationCallback() {
if (!anim::Disabled()) {
RpWidget::update();
}
}
@ -247,7 +247,6 @@ void LocalStorageBox::Row::paintEvent(QPaintEvent *e) {
const auto padding = st::localStorageRowPadding;
const auto height = st::localStorageRowHeight;
const auto bottom = height - padding.bottom() - _description->height();
_progress->step(crl::now());
_progress->draw(
p,
{

View File

@ -44,8 +44,8 @@ int stickerPacksCount(bool includeArchivedOfficial) {
auto result = 0;
auto &order = Auth().data().stickerSetsOrder();
auto &sets = Auth().data().stickerSets();
for (auto i = 0, l = order.size(); i < l; ++i) {
auto it = sets.constFind(order.at(i));
for (const auto setId : order) {
const auto it = sets.constFind(setId);
if (it != sets.cend()) {
if (!(it->flags & MTPDstickerSet::Flag::f_archived) || ((it->flags & MTPDstickerSet::Flag::f_official) && includeArchivedOfficial)) {
++result;
@ -653,7 +653,9 @@ StickersBox::Inner::Row::~Row() = default;
StickersBox::Inner::Inner(QWidget *parent, StickersBox::Section section) : TWidget(parent)
, _section(section)
, _rowHeight(st::contactsPadding.top() + st::contactsPhotoSize + st::contactsPadding.bottom())
, _a_shifting(animation(this, &Inner::step_shifting))
, _shiftingAnimation([=](crl::time now) {
return shiftingAnimationCallback(now);
})
, _itemsTop(st::membersMarginTop)
, _addText(lang(lng_stickers_featured_add).toUpper())
, _addWidth(st::stickersTrendingAdd.font->width(_addText))
@ -665,7 +667,9 @@ StickersBox::Inner::Inner(QWidget *parent, StickersBox::Section section) : TWidg
StickersBox::Inner::Inner(QWidget *parent, not_null<ChannelData*> megagroup) : TWidget(parent)
, _section(StickersBox::Section::Installed)
, _rowHeight(st::contactsPadding.top() + st::contactsPhotoSize + st::contactsPadding.bottom())
, _a_shifting(animation(this, &Inner::step_shifting))
, _shiftingAnimation([=](crl::time now) {
return shiftingAnimationCallback(now);
})
, _itemsTop(st::membersMarginTop)
, _megagroupSet(megagroup)
, _megagroupSetInput(_megagroupSet->mgInfo->stickerSet)
@ -710,10 +714,6 @@ void StickersBox::Inner::setInnerFocus() {
void StickersBox::Inner::paintEvent(QPaintEvent *e) {
Painter p(this);
if (_a_shifting.animating()) {
_a_shifting.step();
}
auto clip = e->rect();
auto ms = crl::now();
p.fillRect(clip, st::boxBg);
@ -1071,7 +1071,7 @@ void StickersBox::Inner::onUpdateSelected() {
auto local = mapFromGlobal(_mouse);
if (_dragging >= 0) {
auto shift = 0;
auto ms = crl::now();
auto now = crl::now();
int firstSetIndex = 0;
if (_rows.at(firstSetIndex)->isRecentSet()) {
++firstSetIndex;
@ -1081,27 +1081,27 @@ void StickersBox::Inner::onUpdateSelected() {
for (int32 from = _dragging, to = _dragging + shift; from > to; --from) {
qSwap(_rows[from], _rows[from - 1]);
_rows[from]->yadd = anim::value(_rows[from]->yadd.current() - _rowHeight, 0);
_animStartTimes[from] = ms;
_shiftingStartTimes[from] = now;
}
} else if (_dragStart.y() < local.y() && _dragging + 1 < _rows.size()) {
shift = floorclamp(local.y() - _dragStart.y() + (_rowHeight / 2), _rowHeight, 0, _rows.size() - _dragging - 1);
for (int32 from = _dragging, to = _dragging + shift; from < to; ++from) {
qSwap(_rows[from], _rows[from + 1]);
_rows[from]->yadd = anim::value(_rows[from]->yadd.current() + _rowHeight, 0);
_animStartTimes[from] = ms;
_shiftingStartTimes[from] = now;
}
}
if (shift) {
_dragging += shift;
_above = _dragging;
_dragStart.setY(_dragStart.y() + shift * _rowHeight);
if (!_a_shifting.animating()) {
_a_shifting.start();
if (!_shiftingAnimation.animating()) {
_shiftingAnimation.start();
}
}
_rows[_dragging]->yadd = anim::value(local.y() - _dragStart.y(), local.y() - _dragStart.y());
_animStartTimes[_dragging] = 0;
_a_shifting.step(ms, true);
_shiftingStartTimes[_dragging] = 0;
shiftingAnimationCallback(now);
auto countDraggingScrollDelta = [this, local] {
if (local.y() < _visibleTop) {
@ -1181,10 +1181,10 @@ void StickersBox::Inner::mouseReleaseEvent(QMouseEvent *e) {
} else if (_dragging >= 0) {
QPoint local(mapFromGlobal(_mouse));
_rows[_dragging]->yadd.start(0.);
_aboveShadowFadeStart = _animStartTimes[_dragging] = crl::now();
_aboveShadowFadeStart = _shiftingStartTimes[_dragging] = crl::now();
_aboveShadowFadeOpacity = anim::value(aboveShadowOpacity(), 0);
if (!_a_shifting.animating()) {
_a_shifting.start();
if (!_shiftingAnimation.animating()) {
_shiftingAnimation.start();
}
_dragging = _started = -1;
@ -1262,64 +1262,64 @@ void StickersBox::Inner::leaveToChildEvent(QEvent *e, QWidget *child) {
onUpdateSelected();
}
void StickersBox::Inner::step_shifting(crl::time ms, bool timer) {
bool StickersBox::Inner::shiftingAnimationCallback(crl::time now) {
if (anim::Disabled()) {
ms += st::stickersRowDuration;
now += st::stickersRowDuration;
}
auto animating = false;
auto updateMin = -1;
auto updateMax = 0;
for (auto i = 0, l = _animStartTimes.size(); i < l; ++i) {
auto start = _animStartTimes.at(i);
for (auto i = 0, count = int(_shiftingStartTimes.size()); i != count; ++i) {
const auto start = _shiftingStartTimes[i];
if (start) {
if (updateMin < 0) updateMin = i;
if (updateMin < 0) {
updateMin = i;
}
updateMax = i;
if (start + st::stickersRowDuration > ms && ms >= start) {
_rows[i]->yadd.update(float64(ms - start) / st::stickersRowDuration, anim::sineInOut);
if (start + st::stickersRowDuration > now && now >= start) {
_rows[i]->yadd.update(float64(now - start) / st::stickersRowDuration, anim::sineInOut);
animating = true;
} else {
_rows[i]->yadd.finish();
_animStartTimes[i] = 0;
_shiftingStartTimes[i] = 0;
}
}
}
if (_aboveShadowFadeStart) {
if (updateMin < 0 || updateMin > _above) updateMin = _above;
if (updateMax < _above) updateMin = _above;
if (_aboveShadowFadeStart + st::stickersRowDuration > ms && ms > _aboveShadowFadeStart) {
_aboveShadowFadeOpacity.update(float64(ms - _aboveShadowFadeStart) / st::stickersRowDuration, anim::sineInOut);
if (_aboveShadowFadeStart + st::stickersRowDuration > now && now > _aboveShadowFadeStart) {
_aboveShadowFadeOpacity.update(float64(now - _aboveShadowFadeStart) / st::stickersRowDuration, anim::sineInOut);
animating = true;
} else {
_aboveShadowFadeOpacity.finish();
_aboveShadowFadeStart = 0;
}
}
if (timer) {
if (_dragging >= 0) {
if (updateMin < 0 || updateMin > _dragging) {
updateMin = _dragging;
}
if (updateMax < _dragging) updateMax = _dragging;
}
if (updateMin == 1 && _rows[0]->isRecentSet()) {
updateMin = 0; // Repaint from the very top of the content.
}
if (updateMin >= 0) {
update(0, _itemsTop + _rowHeight * (updateMin - 1), width(), _rowHeight * (updateMax - updateMin + 3));
if (_dragging >= 0) {
if (updateMin < 0 || updateMin > _dragging) {
updateMin = _dragging;
}
if (updateMax < _dragging) updateMax = _dragging;
}
if (updateMin == 1 && _rows[0]->isRecentSet()) {
updateMin = 0; // Repaint from the very top of the content.
}
if (updateMin >= 0) {
update(0, _itemsTop + _rowHeight * (updateMin - 1), width(), _rowHeight * (updateMax - updateMin + 3));
}
if (!animating) {
_above = _dragging;
_a_shifting.stop();
}
return animating;
}
void StickersBox::Inner::clear() {
_rows.clear();
_animStartTimes.clear();
_shiftingStartTimes.clear();
_aboveShadowFadeStart = 0;
_aboveShadowFadeOpacity = anim::value();
_a_shifting.stop();
_shiftingAnimation.stop();
_above = _dragging = _started = -1;
setSelected(SelectedRow());
setPressed(SelectedRow());
@ -1476,7 +1476,7 @@ void StickersBox::Inner::rebuild() {
return Auth().data().archivedStickerSetsOrder();
})();
_rows.reserve(order.size() + 1);
_animStartTimes.reserve(order.size() + 1);
_shiftingStartTimes.reserve(order.size() + 1);
auto &sets = Auth().data().stickerSets();
if (_megagroupSet) {
@ -1625,7 +1625,7 @@ void StickersBox::Inner::rebuildAppendSet(const Stickers::Set &set, int maxNameW
removed,
pixw,
pixh));
_animStartTimes.push_back(0);
_shiftingStartTimes.push_back(0);
}
void StickersBox::Inner::fillSetCover(const Stickers::Set &set, ImagePtr *thumbnail, DocumentData **outSticker, int *outWidth, int *outHeight) const {

View File

@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "base/timer.h"
#include "mtproto/sender.h"
#include "chat_helpers/stickers.h"
#include "ui/effects/animations.h"
#include "ui/widgets/input_fields.h"
class ConfirmBox;
@ -270,7 +271,7 @@ private:
QRect relativeButtonRect(bool removeButton) const;
void ensureRipple(const style::RippleAnimation &st, QImage mask, bool removeButton);
void step_shifting(crl::time ms, bool timer);
bool shiftingAnimationCallback(crl::time now);
void paintRow(Painter &p, Row *set, int index, crl::time ms);
void paintFakeButton(Painter &p, Row *set, int index, crl::time ms);
void clear();
@ -297,10 +298,10 @@ private:
int32 _rowHeight;
std::vector<std::unique_ptr<Row>> _rows;
QList<crl::time> _animStartTimes;
std::vector<crl::time> _shiftingStartTimes;
crl::time _aboveShadowFadeStart = 0;
anim::value _aboveShadowFadeOpacity;
BasicAnimation _a_shifting;
Ui::Animations::Basic _shiftingAnimation;
Fn<void(uint64 setId)> _installSetCallback;
Fn<void()> _loadMoreCallback;

View File

@ -124,7 +124,7 @@ private:
void setupHandler();
void load();
void step_radial(crl::time ms, bool timer);
void radialAnimationCallback();
int _id = 0;
bool _switching = false;
@ -582,8 +582,8 @@ void Row::setupPreview(const Set &set) {
}
}
void Row::step_radial(crl::time ms, bool timer) {
if (timer && !anim::Disabled()) {
void Row::radialAnimationCallback() {
if (!anim::Disabled()) {
update();
}
}
@ -633,7 +633,7 @@ void Row::updateAnimation(crl::time ms) {
: 0.;
if (!_loading) {
_loading = std::make_unique<Ui::RadialAnimation>(
animation(this, &Row::step_radial));
[=] { radialAnimationCallback(); });
_loading->start(progress);
} else {
_loading->update(progress, false, crl::now());

View File

@ -11,6 +11,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "data/data_session.h"
#include "data/data_channel.h"
#include "ui/widgets/buttons.h"
#include "ui/effects/animations.h"
#include "ui/effects/ripple_animation.h"
#include "ui/image/image.h"
#include "boxes/stickers_box.h"
@ -105,7 +106,7 @@ private:
template <typename Callback>
void enumerateVisibleIcons(Callback callback);
void step_icons(crl::time ms, bool timer);
bool iconsAnimationCallback(crl::time now);
void setSelectedIcon(
int newSelected,
ValidateIconAnimations animations);
@ -134,7 +135,7 @@ private:
int _iconSel = 0;
OverState _iconDown = SpecialOver::None;
bool _iconsDragging = false;
BasicAnimation _a_icons;
Ui::Animations::Basic _iconsAnimation;
QPoint _iconsMousePos, _iconsMouseDown;
int _iconsLeft = 0;
int _iconsRight = 0;
@ -180,7 +181,9 @@ StickersListWidget::Set::~Set() = default;
StickersListWidget::Footer::Footer(not_null<StickersListWidget*> parent) : InnerFooter(parent)
, _pan(parent)
, _a_icons(animation(this, &Footer::step_icons)) {
, _iconsAnimation([=](crl::time now) {
return iconsAnimationCallback(now);
}) {
setMouseTracking(true);
_iconsLeft = _iconsRight = st::emojiCategorySkip + st::stickerIconWidth;
@ -323,11 +326,11 @@ void StickersListWidget::Footer::setSelectedIcon(
_iconsMax);
if (animations == ValidateIconAnimations::None) {
_iconsX = anim::value(iconsXFinal, iconsXFinal);
_a_icons.stop();
_iconsAnimation.stop();
} else {
_iconsX.start(iconsXFinal);
_iconsStartAnim = crl::now();
_a_icons.start();
_iconsAnimation.start();
}
updateSelected();
update();
@ -336,7 +339,7 @@ void StickersListWidget::Footer::setSelectedIcon(
void StickersListWidget::Footer::processHideFinished() {
_iconOver = _iconDown = SpecialOver::None;
_iconsStartAnim = 0;
_a_icons.stop();
_iconsAnimation.stop();
_iconsX.finish();
_iconSelX.finish();
_horizontal = false;
@ -469,7 +472,7 @@ void StickersListWidget::Footer::mouseMoveEvent(QMouseEvent *e) {
if (newX != qRound(_iconsX.current())) {
_iconsX = anim::value(newX, newX);
_iconsStartAnim = 0;
_a_icons.stop();
_iconsAnimation.stop();
update();
}
}
@ -504,7 +507,7 @@ void StickersListWidget::Footer::finishDragging() {
if (newX != qRound(_iconsX.current())) {
_iconsX = anim::value(newX, newX);
_iconsStartAnim = 0;
_a_icons.stop();
_iconsAnimation.stop();
update();
}
_iconsDragging = false;
@ -539,7 +542,7 @@ void StickersListWidget::Footer::scrollByWheelEvent(
if (newX != qRound(_iconsX.current())) {
_iconsX = anim::value(newX, newX);
_iconsStartAnim = 0;
_a_icons.stop();
_iconsAnimation.stop();
updateSelected();
update();
}
@ -601,7 +604,7 @@ void StickersListWidget::Footer::refreshIconsGeometry(
_iconsX.finish();
_iconSelX.finish();
_iconsStartAnim = 0;
_a_icons.stop();
_iconsAnimation.stop();
_iconsMax = std::max(
_iconsLeft + int(_icons.size()) * st::stickerIconWidth + _iconsRight - width(),
0);
@ -673,13 +676,13 @@ void StickersListWidget::Footer::paintSetIcon(
}
}
void StickersListWidget::Footer::step_icons(crl::time ms, bool timer) {
bool StickersListWidget::Footer::iconsAnimationCallback(crl::time now) {
if (anim::Disabled()) {
ms += st::stickerIconMove;
now += st::stickerIconMove;
}
if (_iconsStartAnim) {
auto dt = (ms - _iconsStartAnim) / float64(st::stickerIconMove);
if (dt >= 1) {
const auto dt = (now - _iconsStartAnim) / float64(st::stickerIconMove);
if (dt >= 1.) {
_iconsStartAnim = 0;
_iconsX.finish();
_iconSelX.finish();
@ -689,11 +692,9 @@ void StickersListWidget::Footer::step_icons(crl::time ms, bool timer) {
}
}
if (timer) update();
update();
if (!_iconsStartAnim) {
_a_icons.stop();
}
return (_iconsStartAnim != 0);
}
StickersListWidget::StickersListWidget(QWidget *parent, not_null<Window::Controller*> controller) : Inner(parent, controller)

View File

@ -46,7 +46,6 @@ enum {
MediaOverviewStartPerPage = 5,
AudioVoiceMsgMaxLength = 100 * 60, // 100 minutes
AudioVoiceMsgUpdateView = 100, // 100ms
AudioVoiceMsgChannels = 2, // stereo
StickerMaxSize = 2048, // 2048x2048 is a max image size for sticker

View File

@ -71,7 +71,9 @@ DialogsInner::DialogsInner(QWidget *parent, not_null<Window::Controller*> contro
, _dialogs(std::make_unique<Dialogs::IndexedList>(Dialogs::SortMode::Date))
, _contactsNoDialogs(std::make_unique<Dialogs::IndexedList>(Dialogs::SortMode::Name))
, _contacts(std::make_unique<Dialogs::IndexedList>(Dialogs::SortMode::Name))
, _a_pinnedShifting(animation(this, &DialogsInner::step_pinnedShifting))
, _pinnedShiftAnimation([=](crl::time now) {
return pinnedShiftAnimationCallback(now);
})
, _addContactLnk(this, lang(lng_add_contact_button))
, _cancelSearchInChat(this, st::dialogsCancelSearchInPeer)
, _cancelSearchFromUser(this, st::dialogsCancelSearchInPeer) {
@ -255,10 +257,6 @@ void DialogsInner::paintRegion(Painter &p, const QRegion &region, bool paintingO
auto fullWidth = getFullWidth();
auto ms = crl::now();
if (_state == State::Default) {
if (_a_pinnedShifting.animating()) {
_a_pinnedShifting.step(ms, false);
}
auto rows = shownDialogs();
auto dialogsClip = r;
if (_dialogsImportant) {
@ -855,7 +853,7 @@ void DialogsInner::mousePressEvent(QMouseEvent *e) {
} else if (_pressed) {
auto row = _pressed;
row->addRipple(e->pos() - QPoint(0, dialogsOffset() + _pressed->pos() * st::dialogsRowHeight), QSize(getFullWidth(), st::dialogsRowHeight), [this, row] {
if (!_a_pinnedShifting.animating()) {
if (!_pinnedShiftAnimation.animating()) {
row->entry()->updateChatListEntry();
}
});
@ -901,7 +899,7 @@ void DialogsInner::checkReorderPinnedStart(QPoint localPosition) {
_pinnedOrder = Auth().data().pinnedDialogsOrder();
_pinnedRows[_draggingIndex].yadd = anim::value(0, localPosition.y() - _dragStart.y());
_pinnedRows[_draggingIndex].animStartTime = crl::now();
_a_pinnedShifting.start();
_pinnedShiftAnimation.start();
}
}
}
@ -959,7 +957,7 @@ void DialogsInner::finishReorderPinned() {
}
_draggingIndex = -1;
if (!_a_pinnedShifting.animating()) {
if (!_pinnedShiftAnimation.animating()) {
_pinnedRows.clear();
_aboveIndex = -1;
}
@ -969,7 +967,7 @@ void DialogsInner::finishReorderPinned() {
}
void DialogsInner::stopReorderPinned() {
_a_pinnedShifting.stop();
_pinnedShiftAnimation.stop();
finishReorderPinned();
}
@ -990,7 +988,7 @@ int DialogsInner::updateReorderIndexGetCount() {
_draggingIndex = index;
_aboveIndex = _draggingIndex;
while (count > _pinnedRows.size()) {
_pinnedRows.push_back(PinnedRow());
_pinnedRows.emplace_back();
}
while (count < _pinnedRows.size()) {
_pinnedRows.pop_back();
@ -1007,7 +1005,7 @@ bool DialogsInner::updateReorderPinned(QPoint localPosition) {
auto yaddWas = _pinnedRows[_draggingIndex].yadd.current();
auto shift = 0;
auto ms = crl::now();
auto now = crl::now();
auto rowHeight = st::dialogsRowHeight;
if (_dragStart.y() > localPosition.y() && _draggingIndex > 0) {
shift = -floorclamp(_dragStart.y() - localPosition.y() + (rowHeight / 2), rowHeight, 0, _draggingIndex);
@ -1016,7 +1014,7 @@ bool DialogsInner::updateReorderPinned(QPoint localPosition) {
shownDialogs()->movePinned(_dragging, -1);
std::swap(_pinnedRows[from], _pinnedRows[from - 1]);
_pinnedRows[from].yadd = anim::value(_pinnedRows[from].yadd.current() - rowHeight, 0);
_pinnedRows[from].animStartTime = ms;
_pinnedRows[from].animStartTime = now;
}
} else if (_dragStart.y() < localPosition.y() && _draggingIndex + 1 < pinnedCount) {
shift = floorclamp(localPosition.y() - _dragStart.y() + (rowHeight / 2), rowHeight, 0, pinnedCount - _draggingIndex - 1);
@ -1025,15 +1023,15 @@ bool DialogsInner::updateReorderPinned(QPoint localPosition) {
shownDialogs()->movePinned(_dragging, 1);
std::swap(_pinnedRows[from], _pinnedRows[from + 1]);
_pinnedRows[from].yadd = anim::value(_pinnedRows[from].yadd.current() + rowHeight, 0);
_pinnedRows[from].animStartTime = ms;
_pinnedRows[from].animStartTime = now;
}
}
if (shift) {
_draggingIndex += shift;
_aboveIndex = _draggingIndex;
_dragStart.setY(_dragStart.y() + shift * rowHeight);
if (!_a_pinnedShifting.animating()) {
_a_pinnedShifting.start();
if (!_pinnedShiftAnimation.animating()) {
_pinnedShiftAnimation.start();
}
}
_aboveTopShift = qCeil(_pinnedRows[_aboveIndex].yadd.current());
@ -1041,22 +1039,22 @@ bool DialogsInner::updateReorderPinned(QPoint localPosition) {
if (!_pinnedRows[_draggingIndex].animStartTime) {
_pinnedRows[_draggingIndex].yadd.finish();
}
_a_pinnedShifting.step(ms, true);
pinnedShiftAnimationCallback(now);
auto countDraggingScrollDelta = [this, localPosition] {
const auto delta = [&] {
if (localPosition.y() < _visibleTop) {
return localPosition.y() - _visibleTop;
}
return 0;
};
}();
emit draggingScrollDelta(countDraggingScrollDelta());
emit draggingScrollDelta(delta);
return true;
}
void DialogsInner::step_pinnedShifting(crl::time ms, bool timer) {
bool DialogsInner::pinnedShiftAnimationCallback(crl::time now) {
if (anim::Disabled()) {
ms += st::stickersRowDuration;
now += st::stickersRowDuration;
}
auto wasAnimating = false;
@ -1069,8 +1067,8 @@ void DialogsInner::step_pinnedShifting(crl::time ms, bool timer) {
wasAnimating = true;
if (updateMin < 0) updateMin = i;
updateMax = i;
if (start + st::stickersRowDuration > ms && ms >= start) {
_pinnedRows[i].yadd.update(float64(ms - start) / st::stickersRowDuration, anim::sineInOut);
if (start + st::stickersRowDuration > now && now >= start) {
_pinnedRows[i].yadd.update(float64(now - start) / st::stickersRowDuration, anim::sineInOut);
animating = true;
} else {
_pinnedRows[i].yadd.finish();
@ -1078,33 +1076,31 @@ void DialogsInner::step_pinnedShifting(crl::time ms, bool timer) {
}
}
}
if (timer || (wasAnimating && !animating)) {
updateReorderIndexGetCount();
if (_draggingIndex >= 0) {
if (updateMin < 0 || updateMin > _draggingIndex) {
updateMin = _draggingIndex;
}
if (updateMax < _draggingIndex) updateMax = _draggingIndex;
updateReorderIndexGetCount();
if (_draggingIndex >= 0) {
if (updateMin < 0 || updateMin > _draggingIndex) {
updateMin = _draggingIndex;
}
if (updateMin >= 0) {
auto top = pinnedOffset();
auto updateFrom = top + st::dialogsRowHeight * (updateMin - 1);
auto updateHeight = st::dialogsRowHeight * (updateMax - updateMin + 3);
if (base::in_range(_aboveIndex, 0, _pinnedRows.size())) {
// Always include currently dragged chat in its current and old positions.
auto aboveRowBottom = top + (_aboveIndex + 1) * st::dialogsRowHeight;
auto aboveTopShift = qCeil(_pinnedRows[_aboveIndex].yadd.current());
accumulate_max(updateHeight, (aboveRowBottom - updateFrom) + _aboveTopShift);
accumulate_max(updateHeight, (aboveRowBottom - updateFrom) + aboveTopShift);
_aboveTopShift = aboveTopShift;
}
update(0, updateFrom, getFullWidth(), updateHeight);
if (updateMax < _draggingIndex) updateMax = _draggingIndex;
}
if (updateMin >= 0) {
auto top = pinnedOffset();
auto updateFrom = top + st::dialogsRowHeight * (updateMin - 1);
auto updateHeight = st::dialogsRowHeight * (updateMax - updateMin + 3);
if (base::in_range(_aboveIndex, 0, _pinnedRows.size())) {
// Always include currently dragged chat in its current and old positions.
auto aboveRowBottom = top + (_aboveIndex + 1) * st::dialogsRowHeight;
auto aboveTopShift = qCeil(_pinnedRows[_aboveIndex].yadd.current());
accumulate_max(updateHeight, (aboveRowBottom - updateFrom) + _aboveTopShift);
accumulate_max(updateHeight, (aboveRowBottom - updateFrom) + aboveTopShift);
_aboveTopShift = aboveTopShift;
}
update(0, updateFrom, getFullWidth(), updateHeight);
}
if (!animating) {
_aboveIndex = _draggingIndex;
_a_pinnedShifting.stop();
}
return animating;
}
void DialogsInner::mouseReleaseEvent(QMouseEvent *e) {
@ -1120,8 +1116,8 @@ void DialogsInner::mousePressReleased(
if (_draggingIndex >= 0) {
_pinnedRows[_draggingIndex].yadd.start(0.);
_pinnedRows[_draggingIndex].animStartTime = crl::now();
if (!_a_pinnedShifting.animating()) {
_a_pinnedShifting.start();
if (!_pinnedShiftAnimation.animating()) {
_pinnedShiftAnimation.start();
}
}
finishReorderPinned();

View File

@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "dialogs/dialogs_widget.h"
#include "dialogs/dialogs_key.h"
#include "data/data_messages.h"
#include "ui/effects/animations.h"
#include "base/flags.h"
namespace Dialogs {
@ -283,7 +284,7 @@ private:
void stopReorderPinned();
int countPinnedIndex(Dialogs::Row *ofRow);
void savePinnedOrder();
void step_pinnedShifting(crl::time ms, bool timer);
bool pinnedShiftAnimationCallback(crl::time now);
void handleChatMigration(not_null<ChatData*> chat);
not_null<Window::Controller*> _controller;
@ -313,7 +314,7 @@ private:
crl::time animStartTime = 0;
};
std::vector<PinnedRow> _pinnedRows;
BasicAnimation _a_pinnedShifting;
Ui::Animations::Basic _pinnedShiftAnimation;
std::deque<Dialogs::Key> _pinnedOrder;
// Remember the last currently dragged row top shift for updating area.

View File

@ -66,7 +66,7 @@ protected:
void onStateChanged(State was, StateChangeSource source) override;
private:
void step_radial(crl::time ms, bool timer);
void radialAnimationCallback();
QString _text;
const style::FlatButton &_st;
@ -95,8 +95,8 @@ void DialogsWidget::BottomButton::setText(const QString &text) {
update();
}
void DialogsWidget::BottomButton::step_radial(crl::time ms, bool timer) {
if (timer && !anim::Disabled() && width() < st::columnMinimalWidthLeft) {
void DialogsWidget::BottomButton::radialAnimationCallback() {
if (!anim::Disabled() && width() < st::columnMinimalWidthLeft) {
update();
}
}
@ -106,7 +106,7 @@ void DialogsWidget::BottomButton::onStateChanged(State was, StateChangeSource so
if ((was & StateFlag::Disabled) != (state() & StateFlag::Disabled)) {
_loading = isDisabled()
? std::make_unique<Ui::InfiniteRadialAnimation>(
animation(this, &BottomButton::step_radial),
[=] { radialAnimationCallback(); },
st::dialogsLoadMoreLoading)
: nullptr;
if (_loading) {

View File

@ -396,7 +396,9 @@ ReplyKeyboard::ReplyKeyboard(
not_null<const HistoryItem*> item,
std::unique_ptr<Style> &&s)
: _item(item)
, _a_selected(animation(this, &ReplyKeyboard::step_selected))
, _selectedAnimation([=](crl::time now) {
return selectedAnimationCallback(now);
})
, _st(std::move(s)) {
if (const auto markup = _item->Get<HistoryMessageReplyMarkup>()) {
const auto context = _item->fullId();
@ -640,20 +642,20 @@ void ReplyKeyboard::startAnimation(int i, int j, int direction) {
_animations.emplace(indexForAnimation, crl::now());
}
if (notStarted && !_a_selected.animating()) {
_a_selected.start();
if (notStarted && !_selectedAnimation.animating()) {
_selectedAnimation.start();
}
}
void ReplyKeyboard::step_selected(crl::time ms, bool timer) {
bool ReplyKeyboard::selectedAnimationCallback(crl::time now) {
if (anim::Disabled()) {
ms += st::botKbDuration;
now += st::botKbDuration;
}
for (auto i = _animations.begin(); i != _animations.end();) {
const auto index = std::abs(i->first) - 1;
const auto row = (index / MatrixRowShift);
const auto col = index % MatrixRowShift;
const auto dt = float64(ms - i->second) / st::botKbDuration;
const auto dt = float64(now - i->second) / st::botKbDuration;
if (dt >= 1) {
_rows[row][col].howMuchOver = (i->first > 0) ? 1 : 0;
i = _animations.erase(i);
@ -662,10 +664,8 @@ void ReplyKeyboard::step_selected(crl::time ms, bool timer) {
++i;
}
}
if (timer) _st->repaint(_item);
if (_animations.empty()) {
_a_selected.stop();
}
_st->repaint(_item);
return !_animations.empty();
}
void ReplyKeyboard::clearSelection() {
@ -676,7 +676,7 @@ void ReplyKeyboard::clearSelection() {
_rows[row][col].howMuchOver = 0;
}
_animations.clear();
_a_selected.stop();
_selectedAnimation.stop();
}
int ReplyKeyboard::Style::buttonSkip() const {
@ -857,19 +857,24 @@ HistoryDocumentCaptioned::HistoryDocumentCaptioned()
: _caption(st::msgFileMinWidth - st::msgPadding.left() - st::msgPadding.right()) {
}
HistoryDocumentVoicePlayback::HistoryDocumentVoicePlayback(const HistoryDocument *that)
: a_progress(0., 0.)
, _a_progress(animation(const_cast<HistoryDocument*>(that), &HistoryDocument::step_voiceProgress)) {
HistoryDocumentVoicePlayback::HistoryDocumentVoicePlayback(
const HistoryDocument *that)
: progress(0., 0.)
, progressAnimation([=](crl::time now) {
const auto nonconst = const_cast<HistoryDocument*>(that);
return nonconst->voiceProgressAnimationCallback(now);
}) {
}
void HistoryDocumentVoice::ensurePlayback(const HistoryDocument *that) const {
void HistoryDocumentVoice::ensurePlayback(
const HistoryDocument *that) const {
if (!_playback) {
_playback = std::make_unique<HistoryDocumentVoicePlayback>(that);
}
}
void HistoryDocumentVoice::checkPlaybackFinished() const {
if (_playback && !_playback->_a_progress.animating()) {
if (_playback && !_playback->progressAnimation.animating()) {
_playback.reset();
}
}

View File

@ -9,6 +9,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/history_item.h"
#include "ui/empty_userpic.h"
#include "ui/effects/animations.h"
class HistoryDocument;
struct WebPageData;
@ -329,7 +330,7 @@ private:
ButtonCoords findButtonCoordsByClickHandler(const ClickHandlerPtr &p);
void step_selected(crl::time ms, bool timer);
bool selectedAnimationCallback(crl::time now);
const not_null<const HistoryItem*> _item;
int _width = 0;
@ -337,7 +338,7 @@ private:
std::vector<std::vector<Button>> _rows;
base::flat_map<int, crl::time> _animations;
BasicAnimation _a_selected;
Ui::Animations::Basic _selectedAnimation;
std::unique_ptr<Style> _st;
ClickHandlerPtr _savedPressed;
@ -383,9 +384,9 @@ struct HistoryDocumentNamed : public RuntimeComponent<HistoryDocumentNamed, Hist
struct HistoryDocumentVoicePlayback {
HistoryDocumentVoicePlayback(const HistoryDocument *that);
int32 _position = 0;
anim::value a_progress;
BasicAnimation _a_progress;
int32 position = 0;
anim::value progress;
Ui::Animations::Basic progressAnimation;
};
class HistoryDocumentVoice : public RuntimeComponent<HistoryDocumentVoice, HistoryDocument> {

View File

@ -105,6 +105,7 @@ constexpr auto kCancelTypingActionTimeout = crl::time(5000);
constexpr auto kSaveDraftTimeout = 1000;
constexpr auto kSaveDraftAnywayTimeout = 5000;
constexpr auto kSaveCloudDraftIdleTimeout = 14000;
constexpr auto kRecordingUpdateDelta = crl::time(100);
ApiWrap::RequestMessageDataCallback replyEditMessageDataCallback() {
return [](ChannelData *channel, MsgId msgId) {
@ -189,7 +190,10 @@ HistoryWidget::HistoryWidget(
, _send(this)
, _unblock(this, lang(lng_unblock_button).toUpper(), st::historyUnblock)
, _botStart(this, lang(lng_bot_start).toUpper(), st::historyComposeButton)
, _joinChannel(this, lang(lng_profile_join_channel).toUpper(), st::historyComposeButton)
, _joinChannel(
this,
lang(lng_profile_join_channel).toUpper(),
st::historyComposeButton)
, _muteUnmute(this, lang(lng_channel_mute).toUpper(), st::historyComposeButton)
, _attachToggle(this, st::historyAttach)
, _tabbedSelectorToggle(this, st::historyAttachEmoji)
@ -202,7 +206,9 @@ HistoryWidget::HistoryWidget(
Ui::InputField::Mode::MultiLine,
langFactory(lng_message_ph))
, _recordCancelWidth(st::historyRecordFont->width(lang(lng_record_cancel)))
, _a_recording(animation(this, &HistoryWidget::step_recording))
, _recordingAnimation([=](crl::time now) {
return recordingAnimationCallback(now);
})
, _kbScroll(this, st::botKbScroll)
, _tabbedPanel(this, controller)
, _tabbedSelector(_tabbedPanel->getSelector())
@ -1235,8 +1241,8 @@ void HistoryWidget::onRecordUpdate(quint16 level, qint32 samples) {
return;
}
a_recordingLevel.start(level);
_a_recording.start();
_recordingLevel.start(level);
_recordingAnimation.start();
_recordingSamples = samples;
if (samples < 0 || samples >= Media::Player::kDefaultFrequency * AudioVoiceMsgMaxLength) {
stopRecording(_peer && samples > 0 && _inField);
@ -1364,7 +1370,7 @@ void HistoryWidget::setReplyReturns(PeerId peer, const QList<MsgId> &replyReturn
_replyReturns = replyReturns;
if (_replyReturns.isEmpty()) {
_replyReturn = 0;
_replyReturn = nullptr;
} else if (_replyReturns.back() < 0 && -_replyReturns.back() < ServerMaxMsgId) {
_replyReturn = App::histItemById(0, -_replyReturns.back());
} else {
@ -1373,7 +1379,7 @@ void HistoryWidget::setReplyReturns(PeerId peer, const QList<MsgId> &replyReturn
while (!_replyReturns.isEmpty() && !_replyReturn) {
_replyReturns.pop_back();
if (_replyReturns.isEmpty()) {
_replyReturn = 0;
_replyReturn = nullptr;
} else if (_replyReturns.back() < 0 && -_replyReturns.back() < ServerMaxMsgId) {
_replyReturn = App::histItemById(0, -_replyReturns.back());
} else {
@ -1383,11 +1389,11 @@ void HistoryWidget::setReplyReturns(PeerId peer, const QList<MsgId> &replyReturn
}
void HistoryWidget::calcNextReplyReturn() {
_replyReturn = 0;
_replyReturn = nullptr;
while (!_replyReturns.isEmpty() && !_replyReturn) {
_replyReturns.pop_back();
if (_replyReturns.isEmpty()) {
_replyReturn = 0;
_replyReturn = nullptr;
} else if (_replyReturns.back() < 0 && -_replyReturns.back() < ServerMaxMsgId) {
_replyReturn = App::histItemById(0, -_replyReturns.back());
} else {
@ -2784,7 +2790,7 @@ bool HistoryWidget::saveEditMsgFail(History *history, const RPCError &error, mtp
}
}
QString err = error.type();
const auto &err = error.type();
if (err == qstr("MESSAGE_ID_INVALID") || err == qstr("CHAT_ADMIN_REQUIRED") || err == qstr("MESSAGE_EDIT_TIME_EXPIRED")) {
Ui::show(Box<InformBox>(lang(lng_edit_error)));
} else if (err == qstr("MESSAGE_NOT_MODIFIED")) {
@ -2989,17 +2995,20 @@ void HistoryWidget::unreadMentionsAnimationFinish() {
updateUnreadMentionsPosition();
}
void HistoryWidget::step_recording(float64 ms, bool timer) {
const auto dt = anim::Disabled() ? 1. : (ms / AudioVoiceMsgUpdateView);
if (dt >= 1) {
_a_recording.stop();
a_recordingLevel.finish();
bool HistoryWidget::recordingAnimationCallback(crl::time now) {
const auto dt = anim::Disabled()
? 1.
: ((now - _recordingAnimation.started())
/ float64(kRecordingUpdateDelta));
if (dt >= 1.) {
_recordingLevel.finish();
} else {
a_recordingLevel.update(dt, anim::linear);
_recordingLevel.update(dt, anim::linear);
}
if (timer && !anim::Disabled()) {
if (!anim::Disabled()) {
update(_attachToggle->geometry());
}
return (dt < 1.);
}
void HistoryWidget::chooseAttach() {
@ -3080,7 +3089,9 @@ void HistoryWidget::leaveEventHook(QEvent *e) {
_attachDragState = DragState::None;
updateDragAreas();
}
if (hasMouseTracking()) mouseMoveEvent(0);
if (hasMouseTracking()) {
mouseMoveEvent(nullptr);
}
}
void HistoryWidget::mouseMoveEvent(QMouseEvent *e) {
@ -3158,8 +3169,8 @@ void HistoryWidget::mouseReleaseEvent(QMouseEvent *e) {
void HistoryWidget::stopRecording(bool send) {
emit Media::Capture::instance()->stop(send);
a_recordingLevel = anim::value();
_a_recording.stop();
_recordingLevel = anim::value();
_recordingAnimation.stop();
_recording = false;
_recordingSamples = 0;
@ -3581,7 +3592,7 @@ void HistoryWidget::onKbToggle(bool manual) {
_field->setMaxHeight(st::historyComposeFieldMaxHeight);
_kbReplyTo = 0;
_kbReplyTo = nullptr;
if (!readyToForward() && (!_previewData || _previewData->pendingTill < 0) && !_editMsgId && !_replyToId) {
_fieldBarCancel->hide();
updateMouseTracking();
@ -3604,7 +3615,9 @@ void HistoryWidget::onKbToggle(bool manual) {
_field->setMaxHeight(st::historyComposeFieldMaxHeight);
_kbReplyTo = (_peer->isChat() || _peer->isChannel() || _keyboard->forceReply()) ? App::histItemById(_keyboard->forMsgId()) : 0;
_kbReplyTo = (_peer->isChat() || _peer->isChannel() || _keyboard->forceReply())
? App::histItemById(_keyboard->forMsgId())
: nullptr;
if (_kbReplyTo && !_editMsgId && !_replyToId && fieldEnabled) {
updateReplyToName();
updateReplyEditText(_kbReplyTo);
@ -3621,7 +3634,9 @@ void HistoryWidget::onKbToggle(bool manual) {
int32 maxh = qMin(_keyboard->height(), st::historyComposeFieldMaxHeight - (st::historyComposeFieldMaxHeight / 2));
_field->setMaxHeight(st::historyComposeFieldMaxHeight - maxh);
_kbReplyTo = (_peer->isChat() || _peer->isChannel() || _keyboard->forceReply()) ? App::histItemById(_keyboard->forMsgId()) : 0;
_kbReplyTo = (_peer->isChat() || _peer->isChannel() || _keyboard->forceReply())
? App::histItemById(_keyboard->forMsgId())
: nullptr;
if (_kbReplyTo && !_editMsgId && !_replyToId) {
updateReplyToName();
updateReplyEditText(_kbReplyTo);
@ -6461,7 +6476,7 @@ void HistoryWidget::drawRecording(Painter &p, float64 recordActive) {
p.setPen(Qt::NoPen);
p.setBrush(st::historyRecordSignalColor);
auto delta = qMin(a_recordingLevel.current() / 0x4000, 1.);
auto delta = qMin(_recordingLevel.current() / 0x4000, 1.);
auto d = 2 * qRound(st::historyRecordSignalMin + (delta * (st::historyRecordSignalMax - st::historyRecordSignalMin)));
{
PainterHighQualityEnabler hq(p);

View File

@ -12,6 +12,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "chat_helpers/field_autocomplete.h"
#include "window/section_widget.h"
#include "ui/widgets/input_fields.h"
#include "ui/effects/animations.h"
#include "ui/rp_widget.h"
#include "base/flags.h"
#include "base/timer.h"
@ -219,7 +220,7 @@ public:
void updatePreview();
void previewCancel();
void step_recording(float64 ms, bool timer);
bool recordingAnimationCallback(crl::time now);
void stopRecording(bool send);
void onListEscapePressed();
@ -768,9 +769,9 @@ private:
rpl::lifetime _uploaderSubscriptions;
// This can animate for a very long time (like in music playing),
// so it should be a BasicAnimation, not an Animation.
BasicAnimation _a_recording;
anim::value a_recordingLevel;
// so it should be a Basic, not a Simple animation.
Ui::Animations::Basic _recordingAnimation;
anim::value _recordingLevel;
bool kbWasHidden() const;

View File

@ -25,6 +25,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
namespace {
constexpr auto kAudioVoiceMsgUpdateView = crl::time(100);
using TextState = HistoryView::TextState;
} // namespace
@ -364,7 +366,7 @@ void HistoryDocument::draw(Painter &p, const QRect &r, TextSelection selection,
if (voice->seeking()) {
return voice->seekingCurrent();
} else if (voice->_playback) {
return voice->_playback->a_progress.current();
return voice->_playback->progress.current();
}
return 0.;
})();
@ -707,15 +709,15 @@ bool HistoryDocument::updateStatusText() const {
if (auto voice = Get<HistoryDocumentVoice>()) {
bool was = (voice->_playback != nullptr);
voice->ensurePlayback(this);
if (!was || state.position != voice->_playback->_position) {
if (!was || state.position != voice->_playback->position) {
auto prg = state.length ? snap(float64(state.position) / state.length, 0., 1.) : 0.;
if (voice->_playback->_position < state.position) {
voice->_playback->a_progress.start(prg);
if (voice->_playback->position < state.position) {
voice->_playback->progress.start(prg);
} else {
voice->_playback->a_progress = anim::value(0., prg);
voice->_playback->progress = anim::value(0., prg);
}
voice->_playback->_position = state.position;
voice->_playback->_a_progress.start();
voice->_playback->position = state.position;
voice->_playback->progressAnimation.start();
}
voice->_lastDurationMs = static_cast<int>((state.length * 1000LL) / state.frequency); // Bad :(
}
@ -759,24 +761,25 @@ bool HistoryDocument::hideForwardedFrom() const {
return _data->isSong();
}
void HistoryDocument::step_voiceProgress(float64 ms, bool timer) {
bool HistoryDocument::voiceProgressAnimationCallback(crl::time now) {
if (anim::Disabled()) {
ms += (2 * AudioVoiceMsgUpdateView);
now += (2 * kAudioVoiceMsgUpdateView);
}
if (auto voice = Get<HistoryDocumentVoice>()) {
if (const auto voice = Get<HistoryDocumentVoice>()) {
if (voice->_playback) {
float64 dt = ms / (2 * AudioVoiceMsgUpdateView);
if (dt >= 1) {
voice->_playback->_a_progress.stop();
voice->_playback->a_progress.finish();
const auto dt = (now - voice->_playback->progressAnimation.started())
/ float64(2 * kAudioVoiceMsgUpdateView);
if (dt >= 1.) {
voice->_playback->progressAnimation.stop();
voice->_playback->progress.finish();
} else {
voice->_playback->a_progress.update(qMin(dt, 1.), anim::linear);
}
if (timer) {
history()->owner().requestViewRepaint(_parent);
voice->_playback->progress.update(qMin(dt, 1.), anim::linear);
}
history()->owner().requestViewRepaint(_parent);
return (dt < 1.);
}
}
return false;
}
void HistoryDocument::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) {
@ -793,8 +796,8 @@ void HistoryDocument::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool
currentProgress);
voice->ensurePlayback(this);
voice->_playback->_position = 0;
voice->_playback->a_progress = anim::value(currentProgress, currentProgress);
voice->_playback->position = 0;
voice->_playback->progress = anim::value(currentProgress, currentProgress);
}
voice->stopSeeking();
}

View File

@ -48,7 +48,7 @@ public:
QMargins bubbleMargins() const override;
bool hideForwardedFrom() const override;
void step_voiceProgress(float64 ms, bool timer);
bool voiceProgressAnimationCallback(crl::time now);
void clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) override;

View File

@ -67,28 +67,28 @@ void HistoryFileMedia::setStatusSize(int newSize, int fullSize, int duration, qi
}
}
void HistoryFileMedia::step_radial(crl::time ms, bool timer) {
const auto updateRadial = [&] {
bool HistoryFileMedia::radialAnimationCallback(crl::time now) const {
const auto radialUpdated = [&] {
return _animation->radial.update(
dataProgress(),
dataFinished(),
ms);
};
if (timer) {
if (!anim::Disabled() || updateRadial()) {
history()->owner().requestViewRepaint(_parent);
}
} else {
updateRadial();
if (!_animation->radial.animating()) {
checkAnimationFinished();
}
now);
}();
if (!anim::Disabled() || radialUpdated) {
history()->owner().requestViewRepaint(_parent);
}
if (!_animation->radial.animating()) {
checkAnimationFinished();
return false;
}
return true;
}
void HistoryFileMedia::ensureAnimation() const {
if (!_animation) {
_animation = std::make_unique<AnimationData>(animation(const_cast<HistoryFileMedia*>(this), &HistoryFileMedia::step_radial));
_animation = std::make_unique<AnimationData>([=](crl::time now) {
return radialAnimationCallback(now);
});
}
}

View File

@ -62,17 +62,16 @@ protected:
// duration = -1 - no duration, duration = -2 - "GIF" duration
void setStatusSize(int newSize, int fullSize, int duration, qint64 realDuration) const;
void step_radial(crl::time ms, bool timer);
bool radialAnimationCallback(crl::time now) const;
void thumbAnimationCallback();
void ensureAnimation() const;
void checkAnimationFinished() const;
bool isRadialAnimation(crl::time ms) const {
if (!_animation || !_animation->radial.animating()) return false;
_animation->radial.step(ms);
return _animation && _animation->radial.animating();
bool isRadialAnimation(crl::time now) const {
return _animation
&& _animation->radial.animating()
&& radialAnimationCallback(now);
}
bool isThumbAnimation(crl::time ms) const {
if (_animation) {
@ -89,9 +88,11 @@ protected:
virtual bool dataLoaded() const = 0;
struct AnimationData {
AnimationData(AnimationCallbacks &&radialCallbacks)
: radial(std::move(radialCallbacks)) {
template <typename Callback>
AnimationData(Callback &&radialCallback)
: radial(std::forward<Callback>(radialCallback)) {
}
Animation a_thumbOver;
Ui::RadialAnimation radial;
};

View File

@ -142,9 +142,10 @@ struct HistoryPoll::AnswersAnimation {
};
struct HistoryPoll::SendingAnimation {
template <typename Callback>
SendingAnimation(
const QByteArray &option,
AnimationCallbacks &&callbacks);
Callback &&callback);
QByteArray option;
Ui::InfiniteRadialAnimation animation;
@ -167,11 +168,14 @@ struct HistoryPoll::Answer {
mutable std::unique_ptr<Ui::RippleAnimation> ripple;
};
template <typename Callback>
HistoryPoll::SendingAnimation::SendingAnimation(
const QByteArray &option,
AnimationCallbacks &&callbacks)
Callback &&callback)
: option(option)
, animation(std::move(callbacks), st::historyPollRadialAnimation) {
, animation(
std::forward<Callback>(callback),
st::historyPollRadialAnimation) {
}
HistoryPoll::Answer::Answer() : text(st::msgMinWidth / 2) {
@ -397,9 +401,7 @@ void HistoryPoll::checkSendingAnimation() const {
}
_sendingAnimation = std::make_unique<SendingAnimation>(
sending,
animation(
const_cast<HistoryPoll*>(this),
&HistoryPoll::step_radial));
[=] { radialAnimationCallback(); });
_sendingAnimation->animation.start();
}
@ -561,8 +563,8 @@ void HistoryPoll::resetAnswersAnimation() const {
}
}
void HistoryPoll::step_radial(crl::time ms, bool timer) {
if (timer && !anim::Disabled()) {
void HistoryPoll::radialAnimationCallback() const {
if (!anim::Disabled()) {
history()->owner().requestViewRepaint(_parent);
}
}

View File

@ -109,7 +109,7 @@ private:
void saveStateInAnimation() const;
void startAnswersAnimation() const;
void resetAnswersAnimation() const;
void step_radial(crl::time ms, bool timer);
void radialAnimationCallback() const;
void toggleRipple(Answer &answer, bool pressed);

View File

@ -154,15 +154,15 @@ void TopBarWidget::updateConnectingState() {
}
} else if (!_connecting) {
_connecting = std::make_unique<Ui::InfiniteRadialAnimation>(
animation(this, &TopBarWidget::step_connecting),
[=] { connectingAnimationCallback(); },
st::topBarConnectingAnimation);
_connecting->start();
update();
}
}
void TopBarWidget::step_connecting(crl::time ms, bool timer) {
if (timer && !anim::Disabled()) {
void TopBarWidget::connectingAnimationCallback() {
if (!anim::Disabled()) {
update();
}
}
@ -362,9 +362,6 @@ bool TopBarWidget::paintConnectingState(
int top,
int outerWidth,
crl::time ms) {
if (_connecting) {
_connecting->step(ms);
}
if (!_connecting) {
return false;
}

View File

@ -84,7 +84,7 @@ private:
void updateConnectingState();
void updateAdaptiveLayout();
int countSelectedButtonsTop(float64 selectedShown);
void step_connecting(crl::time ms, bool timer);
void connectingAnimationCallback();
void paintTopBar(Painter &p, crl::time ms);
void paintStatus(

View File

@ -321,35 +321,34 @@ void Gif::prepareThumbnail(QSize size, QSize frame) const {
void Gif::ensureAnimation() const {
if (!_animation) {
_animation = std::make_unique<AnimationData>(animation(const_cast<Gif*>(this), &Gif::step_radial));
_animation = std::make_unique<AnimationData>([=](crl::time now) {
return radialAnimationCallback(now);
});
}
}
bool Gif::isRadialAnimation(crl::time ms) const {
if (!_animation || !_animation->radial.animating()) return false;
_animation->radial.step(ms);
return _animation && _animation->radial.animating();
bool Gif::isRadialAnimation(crl::time now) const {
return _animation
&& _animation->radial.animating()
&& radialAnimationCallback(now);
}
void Gif::step_radial(crl::time ms, bool timer) {
bool Gif::radialAnimationCallback(crl::time now) const {
const auto document = getShownDocument();
const auto updateRadial = [&] {
const auto radialUpdated = [&] {
return _animation->radial.update(
document->progress(),
!document->loading() || document->loaded(),
ms);
};
if (timer) {
if (!anim::Disabled() || updateRadial()) {
update();
}
} else {
updateRadial();
if (!_animation->radial.animating() && document->loaded()) {
_animation.reset();
}
now);
}();
if (!anim::Disabled() || radialUpdated) {
update();
}
if (!_animation->radial.animating() && document->loaded()) {
_animation.reset();
return false;
}
return true;
}
void Gif::clipCallback(Media::Clip::Notification notification) {
@ -845,28 +844,28 @@ void File::thumbAnimationCallback() {
update();
}
void File::step_radial(crl::time ms, bool timer) {
const auto updateRadial = [&] {
bool File::radialAnimationCallback(crl::time now) const {
const auto radialUpdated = [&] {
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();
}
now);
}();
if (!anim::Disabled() || radialUpdated) {
update();
}
if (!_animation->radial.animating()) {
checkAnimationFinished();
return false;
}
return true;
}
void File::ensureAnimation() const {
if (!_animation) {
_animation = std::make_unique<AnimationData>(animation(const_cast<File*>(this), &File::step_radial));
_animation = std::make_unique<AnimationData>([=](crl::time now) {
return radialAnimationCallback(now);
});
}
}
@ -1251,7 +1250,9 @@ void Game::paint(Painter &p, const QRect &clip, const PaintContext *context) con
bool animating = (_gif && _gif->started());
if (displayLoading) {
if (!_radial) {
_radial = std::make_unique<Ui::RadialAnimation>(animation(const_cast<Game*>(this), &Game::step_radial));
_radial = std::make_unique<Ui::RadialAnimation>([=](crl::time now) {
return radialAnimationCallback(now);
});
}
if (!_radial->animating()) {
_radial->start(document->progress());
@ -1360,31 +1361,28 @@ void Game::validateThumbnail(Image *image, QSize size, bool good) const {
size.height());
}
bool Game::isRadialAnimation(crl::time ms) const {
if (!_radial || !_radial->animating()) return false;
_radial->step(ms);
return _radial && _radial->animating();
bool Game::isRadialAnimation(crl::time now) const {
return _radial
&& _radial->animating()
&& radialAnimationCallback(now);
}
void Game::step_radial(crl::time ms, bool timer) {
bool Game::radialAnimationCallback(crl::time now) const {
const auto document = getResultDocument();
const auto updateRadial = [&] {
const auto radialUpdated = [&] {
return _radial->update(
document->progress(),
!document->loading() || document->loaded(),
ms);
};
if (timer) {
if (!anim::Disabled() || updateRadial()) {
update();
}
} else {
updateRadial();
if (!_radial->animating() && document->loaded()) {
_radial.reset();
}
now);
}();
if (!anim::Disabled() || radialUpdated) {
update();
}
if (!_radial->animating() && document->loaded()) {
_radial.reset();
return false;
}
return true;
}
void Game::clipCallback(Media::Clip::Notification notification) {

View File

@ -92,17 +92,17 @@ private:
void prepareThumbnail(QSize size, QSize frame) const;
void ensureAnimation() const;
bool isRadialAnimation(crl::time ms) const;
void step_radial(crl::time ms, bool timer);
bool isRadialAnimation(crl::time now) const;
bool radialAnimationCallback(crl::time now) const;
void clipCallback(Media::Clip::Notification notification);
struct AnimationData {
AnimationData(AnimationCallbacks &&callbacks)
: over(false)
, radial(std::move(callbacks)) {
template <typename Callback>
AnimationData(Callback &&callback)
: radial(std::forward<Callback>(callback)) {
}
bool over;
bool over = false;
Animation _a_over;
Ui::RadialAnimation radial;
};
@ -250,17 +250,16 @@ public:
private:
void thumbAnimationCallback();
void step_radial(crl::time ms, bool timer);
bool radialAnimationCallback(crl::time now) const;
void ensureAnimation() const;
void checkAnimationFinished() const;
bool updateStatusText() const;
bool isRadialAnimation(crl::time ms) const {
if (!_animation || !_animation->radial.animating()) return false;
_animation->radial.step(ms);
return _animation && _animation->radial.animating();
bool isRadialAnimation(crl::time now) const {
return _animation
&& _animation->radial.animating()
&& radialAnimationCallback(now);
}
bool isThumbAnimation(crl::time ms) const {
if (_animation) {
@ -273,7 +272,9 @@ private:
}
struct AnimationData {
AnimationData(AnimationCallbacks &&radialCallbacks) : radial(std::move(radialCallbacks)) {
template <typename Callback>
AnimationData(Callback &&radialCallback)
: radial(std::forward<Callback>(radialCallback)) {
}
Animation a_thumbOver;
Ui::RadialAnimation radial;
@ -360,8 +361,8 @@ private:
void prepareThumbnail(QSize size) const;
void validateThumbnail(Image *image, QSize size, bool good) const;
bool isRadialAnimation(crl::time ms) const;
void step_radial(crl::time ms, bool timer);
bool isRadialAnimation(crl::time now) const;
bool radialAnimationCallback(crl::time now) const;
void clipCallback(Media::Clip::Notification notification);

View File

@ -91,7 +91,7 @@ void ItemBase::preload() const {
}
}
void ItemBase::update() {
void ItemBase::update() const {
if (_position >= 0) {
context()->inlineItemRepaint(this);
}

View File

@ -78,7 +78,7 @@ public:
virtual void preload() const;
void update();
void update() const;
void layoutChanged();
// ClickHandlerHost interface

View File

@ -23,6 +23,7 @@ constexpr auto kCaptureFrequency = Player::kDefaultFrequency;
constexpr auto kCaptureSkipDuration = crl::time(400);
constexpr auto kCaptureFadeInDuration = crl::time(300);
constexpr auto kCaptureBufferSlice = 256 * 1024;
constexpr auto kCaptureUpdateDelta = crl::time(100);
Instance *CaptureInstance = nullptr;
@ -517,7 +518,7 @@ void Instance::Inner::onTimeout() {
}
}
qint32 samplesFull = d->fullSamples + _captured.size() / sizeof(short), samplesSinceUpdate = samplesFull - d->lastUpdate;
if (samplesSinceUpdate > AudioVoiceMsgUpdateView * kCaptureFrequency / 1000) {
if (samplesSinceUpdate > kCaptureUpdateDelta * kCaptureFrequency / 1000) {
emit updated(d->levelMax, samplesFull);
d->lastUpdate = samplesFull;
d->levelMax = 0;

View File

@ -147,12 +147,13 @@ struct OverlayWidget::Collage {
};
struct OverlayWidget::Streamed {
template <typename Callback>
Streamed(
not_null<Data::Session*> owner,
std::unique_ptr<Streaming::Loader> loader,
QWidget *controlsParent,
not_null<PlaybackControls::Delegate*> controlsDelegate,
AnimationCallbacks loadingCallbacks);
Callback &&loadingCallback);
Streaming::Player player;
Streaming::Information info;
@ -160,22 +161,25 @@ struct OverlayWidget::Streamed {
bool waiting = false;
Ui::InfiniteRadialAnimation radial;
Animation fading;
Ui::Animations::Simple fading;
base::Timer timer;
QImage frameForDirectPaint;
bool resumeOnCallEnd = false;
};
template <typename Callback>
OverlayWidget::Streamed::Streamed(
not_null<Data::Session*> owner,
std::unique_ptr<Streaming::Loader> loader,
QWidget *controlsParent,
not_null<PlaybackControls::Delegate*> controlsDelegate,
AnimationCallbacks loadingCallbacks)
Callback &&loadingCallback)
: player(owner, std::move(loader))
, controls(controlsParent, controlsDelegate)
, radial(std::move(loadingCallbacks), st::mediaviewStreamingRadial) {
, radial(
std::forward<Callback>(loadingCallback),
st::mediaviewStreamingRadial) {
}
OverlayWidget::OverlayWidget()
@ -184,7 +188,7 @@ OverlayWidget::OverlayWidget()
, _docDownload(this, lang(lng_media_download), st::mediaviewFileLink)
, _docSaveAs(this, lang(lng_mediaview_save_as), st::mediaviewFileLink)
, _docCancel(this, lang(lng_cancel), st::mediaviewFileLink)
, _radial(animation(this, &OverlayWidget::step_radial))
, _radial([=](crl::time now) { return radialAnimationCallback(now); })
, _lastAction(-st::mediaviewDeltaFromLastAction, -st::mediaviewDeltaFromLastAction)
, _stateAnimation([=](crl::time now) { return stateAnimationCallback(now); })
, _dropdown(this, st::mediaviewDropdownMenu)
@ -704,8 +708,8 @@ bool OverlayWidget::updateControlsAnimation(crl::time now) {
return (dt < 1);
}
void OverlayWidget::step_waiting(crl::time ms, bool timer) {
if (timer && !anim::Disabled()) {
void OverlayWidget::waitingAnimationCallback() {
if (!anim::Disabled()) {
update(radialRect());
}
}
@ -802,17 +806,17 @@ crl::time OverlayWidget::radialTimeShift() const {
return _photo ? st::radialDuration : 0;
}
void OverlayWidget::step_radial(crl::time ms, bool timer) {
bool OverlayWidget::radialAnimationCallback(crl::time now) {
if ((!_doc && !_photo) || _streamed) {
_radial.stop();
return;
return false;
}
const auto wasAnimating = _radial.animating();
const auto updated = _radial.update(
radialProgress(),
!radialLoading(),
ms + radialTimeShift());
if (timer && (wasAnimating || _radial.animating()) && (!anim::Disabled() || updated)) {
now + radialTimeShift());
if ((wasAnimating || _radial.animating())
&& (!anim::Disabled() || updated)) {
update(radialRect());
}
const auto ready = _doc && _doc->loaded();
@ -833,6 +837,7 @@ void OverlayWidget::step_radial(crl::time ms, bool timer) {
}
}
}
return true;
}
void OverlayWidget::zoomIn() {
@ -1986,7 +1991,7 @@ void OverlayWidget::createStreamingObjects() {
_doc->createStreamingLoader(fileOrigin()),
this,
static_cast<PlaybackControls::Delegate*>(this),
animation(this, &OverlayWidget::step_waiting));
[=] { waitingAnimationCallback(); });
if (videoIsGifv()) {
_streamed->controls.hide();
@ -2449,13 +2454,8 @@ void OverlayWidget::paintEvent(QPaintEvent *e) {
}
}
bool radial = false;
float64 radialOpacity = 0;
if (_radial.animating()) {
_radial.step(ms);
radial = _radial.animating();
radialOpacity = _radial.opacity();
}
const auto radial = _radial.animating();
const auto radialOpacity = radial ? _radial.opacity() : 0.;
paintRadialLoading(p, radial, radialOpacity);
}
if (_saveMsgStarted && _saveMsg.intersects(r)) {
@ -2491,13 +2491,8 @@ void OverlayWidget::paintEvent(QPaintEvent *e) {
if (_docRect.intersects(r)) {
p.fillRect(_docRect, st::mediaviewFileBg);
if (_docIconRect.intersects(r)) {
bool radial = false;
float64 radialOpacity = 0;
if (_radial.animating()) {
_radial.step(ms);
radial = _radial.animating();
radialOpacity = _radial.opacity();
}
const auto radial = _radial.animating();
const auto radialOpacity = radial ? _radial.opacity() : 0.;
if (!_doc || !_doc->hasThumbnail()) {
p.fillRect(_docIconRect, _docIconColor);
if ((!_doc || _doc->loaded()) && (!radial || radialOpacity < 1) && _docIcon) {
@ -2721,12 +2716,9 @@ void OverlayWidget::paintRadialLoading(
bool radial,
float64 radialOpacity) {
if (_streamed) {
const auto ms = crl::now();
_streamed->radial.step(ms);
if (!_streamed->radial.animating()) {
return;
}
_streamed->fading.step(ms);
if (!_streamed->fading.animating() && !_streamed->waiting) {
if (!_streamed->waiting) {
_streamed->radial.stop(anim::type::instant);
@ -2783,7 +2775,7 @@ void OverlayWidget::paintRadialLoadingContent(
if (_streamed) {
paintBg(
_streamed->fading.current(_streamed->waiting ? 1. : 0.),
_streamed->fading.value(_streamed->waiting ? 1. : 0.),
st::radialBg);
_streamed->radial.draw(p, arc.topLeft(), arc.size(), width());
return;

View File

@ -276,8 +276,8 @@ private:
void clearControlsState();
bool stateAnimationCallback(crl::time ms);
void step_radial(crl::time ms, bool timer);
void step_waiting(crl::time ms, bool timer);
bool radialAnimationCallback(crl::time now);
void waitingAnimationCallback();
bool updateControlsAnimation(crl::time now);
void zoomIn();

View File

@ -218,29 +218,29 @@ void RadialProgressItem::setLinks(
_cancell = std::move(cancell);
}
void RadialProgressItem::step_radial(crl::time ms, bool timer) {
const auto updateRadial = [&] {
return _radial->update(dataProgress(), dataFinished(), ms);
};
if (timer) {
if (!anim::Disabled() || updateRadial()) {
Auth().data().requestItemRepaint(parent());
}
} else {
updateRadial();
if (!_radial->animating()) {
checkRadialFinished();
}
bool RadialProgressItem::radialAnimationCallback(crl::time now) const {
const auto radialUpdated = [&] {
return _radial->update(dataProgress(), dataFinished(), now);
}();
if (!anim::Disabled() || radialUpdated) {
Auth().data().requestItemRepaint(parent());
}
if (!_radial->animating()) {
checkRadialFinished();
return false;
}
return true;
}
void RadialProgressItem::ensureRadial() {
if (!_radial) {
_radial = std::make_unique<Ui::RadialAnimation>(animation(const_cast<RadialProgressItem*>(this), &RadialProgressItem::step_radial));
_radial = std::make_unique<Ui::RadialAnimation>([=](crl::time now) {
return radialAnimationCallback(now);
});
}
}
void RadialProgressItem::checkRadialFinished() {
void RadialProgressItem::checkRadialFinished() const {
if (_radial && !_radial->animating() && dataLoaded()) {
_radial.reset();
}

View File

@ -129,16 +129,15 @@ protected:
ClickHandlerPtr &&cancell);
void setDocumentLinks(not_null<DocumentData*> document);
void step_radial(crl::time ms, bool timer);
bool radialAnimationCallback(crl::time now) const;
void ensureRadial();
void checkRadialFinished();
void checkRadialFinished() const;
bool isRadialAnimation(crl::time ms) const {
if (!_radial || !_radial->animating()) return false;
_radial->step(ms);
return _radial && _radial->animating();
bool isRadialAnimation(crl::time now) const {
return _radial
&& _radial->animating()
&& radialAnimationCallback(now);
}
virtual float64 dataProgress() const = 0;
@ -148,7 +147,7 @@ protected:
return false;
}
std::unique_ptr<Ui::RadialAnimation> _radial;
mutable std::unique_ptr<Ui::RadialAnimation> _radial;
Animation _a_iconOver;
};

View File

@ -58,7 +58,7 @@ private:
QRect radialRect() const;
void radialStart();
crl::time radialTimeShift() const;
void step_radial(crl::time ms, bool timer);
void radialAnimationCallback(crl::time now);
QPixmap _background;
object_ptr<Ui::LinkButton> _chooseFromGallery;
@ -114,7 +114,7 @@ BackgroundRow::BackgroundRow(QWidget *parent) : RpWidget(parent)
lang(lng_settings_bg_from_gallery),
st::settingsLink)
, _chooseFromFile(this, lang(lng_settings_bg_from_file), st::settingsLink)
, _radial(animation(this, &BackgroundRow::step_radial)) {
, _radial([=](crl::time now) { radialAnimationCallback(now); }) {
updateImage();
_chooseFromGallery->addClickHandler([] {
@ -139,13 +139,8 @@ BackgroundRow::BackgroundRow(QWidget *parent) : RpWidget(parent)
void BackgroundRow::paintEvent(QPaintEvent *e) {
Painter p(this);
bool radial = false;
float64 radialOpacity = 0;
if (_radial.animating()) {
_radial.step(crl::now());
radial = _radial.animating();
radialOpacity = _radial.opacity();
}
const auto radial = _radial.animating();
const auto radialOpacity = radial ? _radial.opacity() : 0.;
if (radial) {
const auto backThumb = App::main()->newBackgroundThumb();
if (!backThumb) {
@ -249,12 +244,12 @@ crl::time BackgroundRow::radialTimeShift() const {
return st::radialDuration;
}
void BackgroundRow::step_radial(crl::time ms, bool timer) {
void BackgroundRow::radialAnimationCallback(crl::time now) {
const auto updated = _radial.update(
radialProgress(),
!radialLoading(),
ms + radialTimeShift());
if (timer && _radial.animating() && (!anim::Disabled() || updated)) {
now + radialTimeShift());
if (!anim::Disabled() || updated) {
rtlupdate(radialRect());
}
}

View File

@ -11,27 +11,23 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
namespace Ui {
RadialAnimation::RadialAnimation(AnimationCallbacks &&callbacks)
: a_arcStart(0, FullArcLength)
, _animation(std::move(callbacks)) {
}
void RadialAnimation::start(float64 prg) {
_firstStart = _lastStart = _lastTime = crl::now();
int32 iprg = qRound(qMax(prg, 0.0001) * AlmostFullArcLength), iprgstrict = qRound(prg * AlmostFullArcLength);
a_arcEnd = anim::value(iprgstrict, iprg);
const auto iprg = qRound(qMax(prg, 0.0001) * AlmostFullArcLength);
const auto iprgstrict = qRound(prg * AlmostFullArcLength);
_arcEnd = anim::value(iprgstrict, iprg);
_animation.start();
}
bool RadialAnimation::update(float64 prg, bool finished, crl::time ms) {
const auto iprg = qRound(qMax(prg, 0.0001) * AlmostFullArcLength);
const auto result = (iprg != qRound(a_arcEnd.to()));
const auto result = (iprg != qRound(_arcEnd.to()));
if (_finished != finished) {
a_arcEnd.start(iprg);
_arcEnd.start(iprg);
_finished = finished;
_lastStart = _lastTime;
} else if (result) {
a_arcEnd.start(iprg);
_arcEnd.start(iprg);
_lastStart = _lastTime;
}
_lastTime = ms;
@ -43,35 +39,31 @@ bool RadialAnimation::update(float64 prg, bool finished, crl::time ms) {
: fulldt;
_opacity = qMin(opacitydt / st::radialDuration, 1.);
if (anim::Disabled()) {
a_arcEnd.update(1., anim::linear);
_arcEnd.update(1., anim::linear);
if (finished) {
stop();
}
} else if (!finished) {
a_arcEnd.update(1. - (st::radialDuration / (st::radialDuration + dt)), anim::linear);
_arcEnd.update(1. - (st::radialDuration / (st::radialDuration + dt)), anim::linear);
} else if (dt >= st::radialDuration) {
a_arcEnd.update(1., anim::linear);
_arcEnd.update(1., anim::linear);
stop();
} else {
auto r = dt / st::radialDuration;
a_arcEnd.update(r, anim::linear);
_arcEnd.update(r, anim::linear);
_opacity *= 1 - r;
}
auto fromstart = fulldt / st::radialPeriod;
a_arcStart.update(fromstart - std::floor(fromstart), anim::linear);
_arcStart.update(fromstart - std::floor(fromstart), anim::linear);
return result;
}
void RadialAnimation::stop() {
_firstStart = _lastStart = _lastTime = 0;
a_arcEnd = anim::value();
_arcEnd = anim::value();
_animation.stop();
}
void RadialAnimation::step(crl::time ms) {
_animation.step(ms);
}
void RadialAnimation::draw(
Painter &p,
const QRect &inner,
@ -98,10 +90,10 @@ void RadialAnimation::draw(
}
RadialState RadialAnimation::computeState() const {
auto length = MinArcLength + qRound(a_arcEnd.current());
auto length = MinArcLength + qRound(_arcEnd.current());
auto from = QuarterArcLength
- length
- (anim::Disabled() ? 0 : qRound(a_arcStart.current()));
- (anim::Disabled() ? 0 : qRound(_arcStart.current()));
if (rtl()) {
from = QuarterArcLength - (from - QuarterArcLength) - length;
if (from < 0) from += FullArcLength;
@ -109,13 +101,6 @@ RadialState RadialAnimation::computeState() const {
return { _opacity, from, length };
}
InfiniteRadialAnimation::InfiniteRadialAnimation(
AnimationCallbacks &&callbacks,
const style::InfiniteRadialAnimation &st)
: _st(st)
, _animation(std::move(callbacks)) {
}
void InfiniteRadialAnimation::start(crl::time skip) {
const auto now = crl::now();
if (_workFinished <= now && (_workFinished || !_workStarted)) {
@ -145,10 +130,6 @@ void InfiniteRadialAnimation::stop(anim::type animated) {
}
}
void InfiniteRadialAnimation::step(crl::time ms) {
_animation.step(ms);
}
void InfiniteRadialAnimation::draw(
Painter &p,
QPoint position,

View File

@ -7,6 +7,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#pragma once
#include "ui/effects/animations.h"
namespace style {
struct InfiniteRadialAnimation;
} // namespace style
@ -21,7 +23,8 @@ struct RadialState {
class RadialAnimation {
public:
RadialAnimation(AnimationCallbacks &&callbacks);
template <typename Callback>
RadialAnimation(Callback &&callback);
float64 opacity() const {
return _opacity;
@ -34,11 +37,6 @@ public:
bool update(float64 prg, bool finished, crl::time ms);
void stop();
void step(crl::time ms);
void step() {
step(crl::now());
}
void draw(
Painter &p,
const QRect &inner,
@ -52,17 +50,25 @@ private:
crl::time _lastStart = 0;
crl::time _lastTime = 0;
float64 _opacity = 0.;
anim::value a_arcEnd;
anim::value a_arcStart;
BasicAnimation _animation;
anim::value _arcEnd;
anim::value _arcStart;
Ui::Animations::Basic _animation;
bool _finished = false;
};
template <typename Callback>
inline RadialAnimation::RadialAnimation(Callback &&callback)
: _arcStart(0, FullArcLength)
, _animation(std::forward<Callback>(callback)) {
}
class InfiniteRadialAnimation {
public:
template <typename Callback>
InfiniteRadialAnimation(
AnimationCallbacks &&callbacks,
Callback &&callback,
const style::InfiniteRadialAnimation &st);
bool animating() const {
@ -72,11 +78,6 @@ public:
void start(crl::time skip = 0);
void stop(anim::type animated = anim::type::normal);
void step(crl::time ms);
void step() {
step(crl::now());
}
void draw(
Painter &p,
QPoint position,
@ -93,8 +94,16 @@ private:
const style::InfiniteRadialAnimation &_st;
crl::time _workStarted = 0;
crl::time _workFinished = 0;
BasicAnimation _animation;
Ui::Animations::Basic _animation;
};
template <typename Callback>
inline InfiniteRadialAnimation::InfiniteRadialAnimation(
Callback &&callback,
const style::InfiniteRadialAnimation &st)
: _st(st)
, _animation(std::forward<Callback>(callback)) {
}
} // namespace Ui

View File

@ -242,8 +242,8 @@ void EmojiButton::paintEvent(QPaintEvent *e) {
}
}
void EmojiButton::step_loading(crl::time ms, bool timer) {
if (timer && !anim::Disabled()) {
void EmojiButton::loadingAnimationCallback() {
if (!anim::Disabled()) {
update();
}
}
@ -251,7 +251,7 @@ void EmojiButton::step_loading(crl::time ms, bool timer) {
void EmojiButton::setLoading(bool loading) {
if (loading && !_loading) {
_loading = std::make_unique<Ui::InfiniteRadialAnimation>(
animation(this, &EmojiButton::step_loading),
[=] { loadingAnimationCallback(); },
st::defaultInfiniteRadialAnimation);
}
if (loading) {

View File

@ -65,7 +65,7 @@ protected:
QPoint prepareRippleStartPosition() const override;
private:
void step_loading(crl::time ms, bool timer);
void loadingAnimationCallback();
const style::IconButton &_st;

View File

@ -277,10 +277,11 @@ void Manager::removeWidget(internal::Widget *remove) {
if (remove == _hideAll.get()) {
_hideAll.reset();
} else if (remove) {
auto it = std::find_if(_notifications.cbegin(), _notifications.cend(), [remove](auto &item) {
return item.get() == remove;
});
if (it != _notifications.cend()) {
const auto it = ranges::find(
_notifications,
remove,
&std::unique_ptr<Notification>::get);
if (it != end(_notifications)) {
_notifications.erase(it);
_positionsOutdated = true;
}
@ -289,13 +290,13 @@ void Manager::removeWidget(internal::Widget *remove) {
}
void Manager::doShowNotification(HistoryItem *item, int forwardedCount) {
_queuedNotifications.push_back(QueuedNotification(item, forwardedCount));
_queuedNotifications.emplace_back(item, forwardedCount);
showNextFromQueue();
}
void Manager::doClearAll() {
_queuedNotifications.clear();
for_const (auto &notification, _notifications) {
for (const auto &notification : _notifications) {
notification->unlinkHistory();
}
showNextFromQueue();
@ -356,8 +357,10 @@ Widget::Widget(Manager *manager, QPoint startPosition, int shift, Direction shif
, _manager(manager)
, _startPosition(startPosition)
, _direction(shiftDirection)
, a_shift(shift)
, _a_shift(animation(this, &Widget::step_shift)) {
, _shift(shift)
, _shiftAnimation([=](crl::time now) {
return shiftAnimationCallback(now);
}) {
setWindowOpacity(0.);
setWindowFlags(Qt::WindowFlags(Qt::FramelessWindowHint)
@ -392,17 +395,19 @@ void Widget::opacityAnimationCallback() {
}
}
void Widget::step_shift(float64 ms, bool timer) {
bool Widget::shiftAnimationCallback(crl::time now) {
if (anim::Disabled()) {
ms += st::notifyFastAnim;
now += st::notifyFastAnim;
}
float64 dt = ms / float64(st::notifyFastAnim);
if (dt >= 1) {
a_shift.finish();
const auto dt = (now - _shiftAnimation.started())
/ float64(st::notifyFastAnim);
if (dt >= 1.) {
_shift.finish();
} else {
a_shift.update(dt, anim::linear);
_shift.update(dt, anim::linear);
}
moveByShift();
return (dt < 1.);
}
void Widget::hideSlow() {
@ -443,8 +448,8 @@ void Widget::updateOpacity() {
}
void Widget::changeShift(int top) {
a_shift.start(top);
_a_shift.start();
_shift.start(top);
_shiftAnimation.start();
}
void Widget::updatePosition(QPoint startPosition, Direction shiftDirection) {
@ -466,7 +471,7 @@ void Widget::updateGeometry(int x, int y, int width, int height) {
}
void Widget::addToShift(int add) {
a_shift.add(add);
_shift.add(add);
moveByShift();
}
@ -475,7 +480,7 @@ void Widget::moveByShift() {
}
QPoint Widget::computePosition(int height) const {
auto realShift = qRound(a_shift.current());
auto realShift = qRound(_shift.current());
if (_direction == Direction::Up) {
realShift = -realShift - height;
}
@ -678,7 +683,7 @@ void Notification::updateNotifyDisplay() {
}
if (!options.hideMessageText) {
const HistoryItem *textCachedFor = 0;
const HistoryItem *textCachedFor = nullptr;
Text itemTextCache(itemWidth);
QRect r(st::notifyPhotoPos.x() + st::notifyPhotoSize + st::notifyTextLeft, st::notifyItemTop + st::msgNameFont->height, itemWidth, 2 * st::dialogsTextFont->height);
if (_item) {

View File

@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#pragma once
#include "window/notifications_manager.h"
#include "ui/effects/animations.h"
#include "base/timer.h"
namespace Ui {
@ -115,7 +116,7 @@ public:
void updateOpacity();
void changeShift(int top);
int currentShift() const {
return a_shift.current();
return _shift.current();
}
void updatePosition(QPoint startPosition, Direction shiftDirection);
void addToHeight(int add);
@ -139,7 +140,7 @@ private:
void destroyDelayed();
void moveByShift();
void hideAnimated(float64 duration, const anim::transition &func);
void step_shift(float64 ms, bool timer);
bool shiftAnimationCallback(crl::time now);
Manager *_manager = nullptr;
@ -150,8 +151,8 @@ private:
QPoint _startPosition;
Direction _direction;
anim::value a_shift;
BasicAnimation _a_shift;
anim::value _shift;
Ui::Animations::Basic _shiftAnimation;
};

View File

@ -30,7 +30,7 @@ protected:
void paintEvent(QPaintEvent *e) override;
private:
void step(crl::time ms, bool timer);
void animationStep();
Ui::InfiniteRadialAnimation _animation;
@ -38,7 +38,7 @@ private:
Progress::Progress(QWidget *parent)
: RpWidget(parent)
, _animation(animation(this, &Progress::step), st::connectingRadial) {
, _animation([=] { animationStep(); }, st::connectingRadial) {
setAttribute(Qt::WA_OpaquePaintEvent);
setAttribute(Qt::WA_TransparentForMouseEvents);
resize(st::connectingRadial.size);
@ -58,8 +58,8 @@ void Progress::paintEvent(QPaintEvent *e) {
width());
}
void Progress::step(crl::time ms, bool timer) {
if (timer && !anim::Disabled()) {
void Progress::animationStep() {
if (!anim::Disabled()) {
update();
}
}