mirror of https://github.com/procxx/kepka.git
Add progress animation to GIFs search.
Also display "no results" phrase.
This commit is contained in:
parent
2028116e22
commit
03a59b04be
|
@ -270,6 +270,7 @@ contactsSearchCancel: CrossButton {
|
|||
crossPosition: point(4px, 4px);
|
||||
|
||||
duration: 150;
|
||||
loadingPeriod: 1000;
|
||||
ripple: RippleAnimation(defaultRippleAnimation) {
|
||||
color: windowBgOver;
|
||||
}
|
||||
|
|
|
@ -137,6 +137,7 @@ dialogsCancelSearch: CrossButton {
|
|||
crossPosition: point(0px, 0px);
|
||||
|
||||
duration: 150;
|
||||
loadingPeriod: 1000;
|
||||
ripple: emptyRippleAnimation;
|
||||
}
|
||||
|
||||
|
|
|
@ -48,6 +48,9 @@ public:
|
|||
|
||||
void stealFocus();
|
||||
void returnFocus();
|
||||
void setLoading(bool loading) {
|
||||
_cancel->setLoadingAnimation(loading);
|
||||
}
|
||||
|
||||
protected:
|
||||
void paintEvent(QPaintEvent *e) override;
|
||||
|
@ -181,6 +184,7 @@ GifsListWidget::~GifsListWidget() {
|
|||
}
|
||||
|
||||
void GifsListWidget::cancelGifsSearch() {
|
||||
_footer->setLoading(false);
|
||||
if (_inlineRequestId) {
|
||||
request(_inlineRequestId).cancel();
|
||||
_inlineRequestId = 0;
|
||||
|
@ -192,6 +196,7 @@ void GifsListWidget::cancelGifsSearch() {
|
|||
}
|
||||
|
||||
void GifsListWidget::inlineResultsDone(const MTPmessages_BotResults &result) {
|
||||
_footer->setLoading(false);
|
||||
_inlineRequestId = 0;
|
||||
|
||||
auto it = _inlineCache.find(_inlineQuery);
|
||||
|
@ -754,9 +759,7 @@ bool GifsListWidget::refreshInlineRows(int32 *added) {
|
|||
auto it = _inlineCache.find(_inlineQuery);
|
||||
const InlineCacheEntry *entry = nullptr;
|
||||
if (it != _inlineCache.cend()) {
|
||||
if (!it->second->results.empty()) {
|
||||
entry = it->second.get();
|
||||
}
|
||||
entry = it->second.get();
|
||||
_inlineNextOffset = it->second->nextOffset;
|
||||
}
|
||||
auto result = refreshInlineRows(entry, false);
|
||||
|
@ -780,6 +783,7 @@ void GifsListWidget::searchForGifs(const QString &query) {
|
|||
}
|
||||
|
||||
if (_inlineQuery != query) {
|
||||
_footer->setLoading(false);
|
||||
if (_inlineRequestId) {
|
||||
request(_inlineRequestId).cancel();
|
||||
_inlineRequestId = 0;
|
||||
|
@ -813,8 +817,10 @@ void GifsListWidget::sendInlineRequest() {
|
|||
if (_inlineRequestId || !_inlineQueryPeer || _inlineNextQuery.isEmpty()) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (!_searchBot) {
|
||||
// Wait for the bot being resolved.
|
||||
_footer->setLoading(true);
|
||||
_inlineRequestTimer.start(kSearchRequestDelay);
|
||||
return;
|
||||
}
|
||||
|
@ -825,13 +831,18 @@ void GifsListWidget::sendInlineRequest() {
|
|||
auto it = _inlineCache.find(_inlineQuery);
|
||||
if (it != _inlineCache.cend()) {
|
||||
nextOffset = it->second->nextOffset;
|
||||
if (nextOffset.isEmpty()) return;
|
||||
if (nextOffset.isEmpty()) {
|
||||
_footer->setLoading(false);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
_footer->setLoading(true);
|
||||
_inlineRequestId = request(MTPmessages_GetInlineBotResults(MTP_flags(0), _searchBot->inputUser, _inlineQueryPeer->input, MTPInputGeoPoint(), MTP_string(_inlineQuery), MTP_string(nextOffset))).done([this](const MTPmessages_BotResults &result, mtpRequestId requestId) {
|
||||
inlineResultsDone(result);
|
||||
}).fail([this](const RPCError &error) {
|
||||
// show error?
|
||||
_footer->setLoading(false);
|
||||
_inlineRequestId = 0;
|
||||
}).handleAllErrors().send();
|
||||
}
|
||||
|
|
|
@ -21,19 +21,100 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
|||
#include "ui/effects/cross_animation.h"
|
||||
|
||||
namespace Ui {
|
||||
namespace {
|
||||
|
||||
void CrossAnimation::paint(Painter &p, const style::CrossAnimation &st, style::color color, int x, int y, int outerWidth, float64 shown) {
|
||||
constexpr auto kPointCount = 12;
|
||||
|
||||
//
|
||||
// 1 3
|
||||
// X X X X
|
||||
// X X X X
|
||||
// 0 X X 4
|
||||
// X X X X
|
||||
// X 2 X
|
||||
// X X
|
||||
// X X
|
||||
// 11 5
|
||||
// X X
|
||||
// X X
|
||||
// X 8 X
|
||||
// X X X X
|
||||
// 10 X X 6
|
||||
// X X X X
|
||||
// X X X X
|
||||
// 9 7
|
||||
//
|
||||
|
||||
void transformLoadingCross(float64 loading, std::array<QPointF, kPointCount> &points, int &paintPointsCount) {
|
||||
auto moveTo = [](QPointF &point, QPointF &to, float64 ratio) {
|
||||
point = point * (1. - ratio) + to * ratio;
|
||||
};
|
||||
auto moveFrom = [](QPointF &point, QPointF &from, float64 ratio) {
|
||||
point = from * (1. - ratio) + point * ratio;
|
||||
};
|
||||
auto paintPoints = [&points, &paintPointsCount](std::initializer_list<int> &&indices) {
|
||||
auto index = 0;
|
||||
for (auto paintIndex : indices) {
|
||||
points[index++] = points[paintIndex];
|
||||
}
|
||||
paintPointsCount = indices.size();
|
||||
};
|
||||
|
||||
if (loading < 0.125) {
|
||||
auto ratio = loading / 0.125;
|
||||
moveTo(points[6], points[5], ratio);
|
||||
moveTo(points[7], points[8], ratio);
|
||||
} else if (loading < 0.25) {
|
||||
auto ratio = (loading - 0.125) / 0.125;
|
||||
moveTo(points[9], points[8], ratio);
|
||||
moveTo(points[10], points[11], ratio);
|
||||
paintPoints({ 0, 1, 2, 3, 4, 9, 10, 11 });
|
||||
} else if (loading < 0.375) {
|
||||
auto ratio = (loading - 0.25) / 0.125;
|
||||
moveTo(points[0], points[11], ratio);
|
||||
moveTo(points[1], points[2], ratio);
|
||||
paintPoints({ 0, 1, 2, 3, 4, 8 });
|
||||
} else if (loading < 0.5) {
|
||||
auto ratio = (loading - 0.375) / 0.125;
|
||||
moveTo(points[8], points[4], ratio);
|
||||
moveTo(points[11], points[3], ratio);
|
||||
paintPoints({ 3, 4, 8, 11 });
|
||||
} else if (loading < 0.625) {
|
||||
auto ratio = (loading - 0.5) / 0.125;
|
||||
moveFrom(points[8], points[4], ratio);
|
||||
moveFrom(points[11], points[3], ratio);
|
||||
paintPoints({ 3, 4, 8, 11 });
|
||||
} else if (loading < 0.75) {
|
||||
auto ratio = (loading - 0.625) / 0.125;
|
||||
moveFrom(points[6], points[5], ratio);
|
||||
moveFrom(points[7], points[8], ratio);
|
||||
paintPoints({ 3, 4, 5, 6, 7, 11 });
|
||||
} else if (loading < 0.875) {
|
||||
auto ratio = (loading - 0.75) / 0.125;
|
||||
moveFrom(points[9], points[8], ratio);
|
||||
moveFrom(points[10], points[11], ratio);
|
||||
paintPoints({ 3, 4, 5, 6, 7, 8, 9, 10 });
|
||||
} else {
|
||||
auto ratio = (loading - 0.875) / 0.125;
|
||||
moveFrom(points[0], points[11], ratio);
|
||||
moveFrom(points[1], points[2], ratio);
|
||||
}
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
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 deleteScale = shown + st.minScale * (1. - shown);
|
||||
auto deleteSkip = deleteScale * st.skip + (1. - deleteScale) * (st.size / 2);
|
||||
auto sqrt2 = sqrt(2.);
|
||||
auto deleteScale = shown + st.minScale * (1. - shown);
|
||||
auto deleteSkip = (deleteScale * st.skip) + (1. - deleteScale) * (st.size / 2);
|
||||
auto deleteLeft = rtlpoint(x + deleteSkip, 0, outerWidth).x() + 0.;
|
||||
auto deleteTop = y + deleteSkip + 0.;
|
||||
auto deleteWidth = st.size - 2 * deleteSkip;
|
||||
auto deleteHeight = st.size - 2 * deleteSkip;
|
||||
auto deleteStroke = st.stroke / sqrt2;
|
||||
QPointF pathDelete[] = {
|
||||
std::array<QPointF, kPointCount> pathDelete = { {
|
||||
{ deleteLeft, deleteTop + deleteStroke },
|
||||
{ deleteLeft + deleteStroke, deleteTop },
|
||||
{ deleteLeft + (deleteWidth / 2.), deleteTop + (deleteHeight / 2.) - deleteStroke },
|
||||
|
@ -46,7 +127,17 @@ void CrossAnimation::paint(Painter &p, const style::CrossAnimation &st, style::c
|
|||
{ deleteLeft + deleteStroke, deleteTop + deleteHeight },
|
||||
{ deleteLeft, deleteTop + deleteHeight - deleteStroke },
|
||||
{ deleteLeft + (deleteWidth / 2.) - deleteStroke, deleteTop + (deleteHeight / 2.) },
|
||||
};
|
||||
} };
|
||||
auto pathDeleteSize = kPointCount;
|
||||
|
||||
auto loadingArcLength = 0;
|
||||
if (loading > 0.) {
|
||||
transformLoadingCross(loading, pathDelete, pathDeleteSize);
|
||||
|
||||
auto loadingArc = (loading >= 0.5) ? (loading - 1.) : loading;
|
||||
loadingArcLength = qRound(-loadingArc * 2 * FullArcLength);
|
||||
}
|
||||
|
||||
if (shown < 1.) {
|
||||
auto alpha = -(shown - 1.) * M_PI_2;
|
||||
auto cosalpha = cos(alpha);
|
||||
|
@ -62,10 +153,30 @@ void CrossAnimation::paint(Painter &p, const style::CrossAnimation &st, style::c
|
|||
}
|
||||
QPainterPath path;
|
||||
path.moveTo(pathDelete[0]);
|
||||
for (int i = 1; i != base::array_size(pathDelete); ++i) {
|
||||
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.);
|
||||
}
|
||||
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);
|
||||
}
|
||||
}
|
||||
|
||||
} // namespace Ui
|
||||
|
|
|
@ -26,7 +26,7 @@ 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);
|
||||
static void paint(Painter &p, const style::CrossAnimation &st, style::color color, int x, int y, int outerWidth, float64 shown, float64 loading = 0.);
|
||||
|
||||
};
|
||||
|
||||
|
|
|
@ -625,12 +625,22 @@ void LeftOutlineButton::paintEvent(QPaintEvent *e) {
|
|||
}
|
||||
|
||||
CrossButton::CrossButton(QWidget *parent, const style::CrossButton &st) : RippleButton(parent, st.ripple)
|
||||
, _st(st) {
|
||||
, _st(st)
|
||||
, _a_loading(animation(this, &CrossButton::step_loading)) {
|
||||
resize(_st.width, _st.height);
|
||||
setCursor(style::cur_pointer);
|
||||
hide();
|
||||
}
|
||||
|
||||
void CrossButton::step_loading(TimeMs ms, bool timer) {
|
||||
if (stopLoadingAnimation(ms)) {
|
||||
_a_loading.stop();
|
||||
}
|
||||
if (timer) {
|
||||
update();
|
||||
}
|
||||
}
|
||||
|
||||
void CrossButton::toggleAnimated(bool visible) {
|
||||
if (_shown == visible) {
|
||||
return;
|
||||
|
@ -659,7 +669,43 @@ void CrossButton::paintEvent(QPaintEvent *e) {
|
|||
|
||||
paintRipple(p, _st.crossPosition.x(), _st.crossPosition.y(), ms);
|
||||
|
||||
CrossAnimation::paint(p, _st.cross, over ? _st.crossFgOver : _st.crossFg, _st.crossPosition.x(), _st.crossPosition.y(), width(), shown);
|
||||
auto loading = 0.;
|
||||
if (_a_loading.animating()) {
|
||||
if (stopLoadingAnimation(ms)) {
|
||||
_a_loading.stop();
|
||||
} else {
|
||||
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);
|
||||
}
|
||||
|
||||
bool CrossButton::stopLoadingAnimation(TimeMs ms) {
|
||||
if (!_loadingStopMs) {
|
||||
return false;
|
||||
}
|
||||
auto stopPeriod = (_loadingStopMs - _loadingStartMs) / _st.loadingPeriod;
|
||||
auto currentPeriod = (ms - _loadingStartMs) / _st.loadingPeriod;
|
||||
if (currentPeriod != stopPeriod) {
|
||||
t_assert(currentPeriod > stopPeriod);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
void CrossButton::setLoadingAnimation(bool enabled) {
|
||||
if (enabled) {
|
||||
_loadingStopMs = 0;
|
||||
if (!_a_loading.animating()) {
|
||||
_loadingStartMs = getms();
|
||||
_a_loading.start();
|
||||
}
|
||||
} else if (_a_loading.animating()) {
|
||||
_loadingStopMs = getms();
|
||||
if (!((_loadingStopMs - _loadingStartMs) % _st.loadingPeriod)) {
|
||||
_a_loading.stop();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void CrossButton::onStateChanged(State was, StateChangeSource source) {
|
||||
|
|
|
@ -233,6 +233,7 @@ public:
|
|||
bool isShown() const {
|
||||
return _shown;
|
||||
}
|
||||
void setLoadingAnimation(bool enabled);
|
||||
|
||||
protected:
|
||||
void paintEvent(QPaintEvent *e) override;
|
||||
|
@ -243,6 +244,8 @@ protected:
|
|||
QPoint prepareRippleStartPosition() const override;
|
||||
|
||||
private:
|
||||
void step_loading(TimeMs ms, bool timer);
|
||||
bool stopLoadingAnimation(TimeMs ms);
|
||||
void animationCallback();
|
||||
|
||||
const style::CrossButton &_st;
|
||||
|
@ -250,6 +253,10 @@ private:
|
|||
bool _shown = false;
|
||||
Animation _a_show;
|
||||
|
||||
TimeMs _loadingStartMs = 0;
|
||||
TimeMs _loadingStopMs = 0;
|
||||
BasicAnimation _a_loading;
|
||||
|
||||
};
|
||||
|
||||
} // namespace Ui
|
||||
|
|
|
@ -315,6 +315,7 @@ CrossButton {
|
|||
crossFgOver:color;
|
||||
crossPosition: point;
|
||||
duration: int;
|
||||
loadingPeriod: int;
|
||||
|
||||
ripple: RippleAnimation;
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue