Add loading rows placeholder.

This commit is contained in:
John Preston 2020-01-16 15:04:21 +03:00
parent 80168a58a7
commit 5acb7448b7
2 changed files with 102 additions and 11 deletions

View File

@ -936,7 +936,7 @@ pollResultsQuestion: FlatLabel(defaultFlatLabel) {
pollResultsVotesCount: FlatLabel(defaultFlatLabel) { pollResultsVotesCount: FlatLabel(defaultFlatLabel) {
textFg: windowSubTextFg; textFg: windowSubTextFg;
} }
pollResultsHeaderPadding: margins(22px, 8px, 22px, 8px); pollResultsHeaderPadding: margins(22px, 22px, 22px, 8px);
pollResultsShowMore: SettingsButton { pollResultsShowMore: SettingsButton {
textFg: lightButtonFg; textFg: lightButtonFg;
textFgOver: lightButtonFgOver; textFgOver: lightButtonFgOver;

View File

@ -37,6 +37,21 @@ constexpr auto kFirstPage = 15;
constexpr auto kPerPage = 50; constexpr auto kPerPage = 50;
constexpr auto kLeavePreloaded = 5; constexpr auto kLeavePreloaded = 5;
class PeerListDummy final : public Ui::RpWidget {
public:
PeerListDummy(QWidget *parent, int count, const style::PeerList &st);
protected:
void paintEvent(QPaintEvent *e) override;
private:
const style::PeerList &_st;
int _count = 0;
std::vector<Ui::Animations::Simple> _animations;
};
class ListDelegate final : public PeerListContentDelegate { class ListDelegate final : public PeerListContentDelegate {
public: public:
void peerListSetTitle(rpl::producer<QString> title) override; void peerListSetTitle(rpl::producer<QString> title) override;
@ -53,6 +68,52 @@ public:
}; };
PeerListDummy::PeerListDummy(
QWidget *parent,
int count,
const style::PeerList &st)
: _st(st)
, _count(count) {
resize(width(), _count * _st.item.height);
}
void PeerListDummy::paintEvent(QPaintEvent *e) {
QPainter p(this);
PainterHighQualityEnabler hq(p);
const auto fill = e->rect();
const auto bottom = fill.top() + fill.height();
const auto from = floorclamp(fill.top(), _st.item.height, 0, _count);
const auto till = ceilclamp(bottom, _st.item.height, 0, _count);
p.translate(0, _st.item.height * from);
for (auto i = from; i != till; ++i) {
p.setBrush(st::windowBgRipple);
p.drawEllipse(
_st.item.photoPosition.x(),
_st.item.photoPosition.y(),
_st.item.photoSize,
_st.item.photoSize);
const auto small = int(1.5 * _st.item.photoSize);
const auto large = 2 * small;
const auto second = (i % 2) ? large : small;
const auto height = _st.item.nameStyle.font->height / 2;
const auto radius = height / 2;
const auto left = _st.item.namePosition.x();
const auto top = _st.item.namePosition.y()
+ (_st.item.nameStyle.font->height - height) / 2;
const auto skip = _st.item.namePosition.x()
- _st.item.photoPosition.x()
- _st.item.photoSize;
const auto next = left + small + skip;
p.drawRoundedRect(left, top, small, height, radius, radius);
p.drawRoundedRect(next, top, second, height, radius, radius);
p.translate(0, _st.item.height);
}
}
void ListDelegate::peerListSetTitle(rpl::producer<QString> title) { void ListDelegate::peerListSetTitle(rpl::producer<QString> title) {
} }
@ -106,8 +167,9 @@ public:
[[nodiscard]] auto showPeerInfoRequests() const [[nodiscard]] auto showPeerInfoRequests() const
-> rpl::producer<not_null<PeerData*>>; -> rpl::producer<not_null<PeerData*>>;
[[nodiscard]] rpl::producer<int> count() const;
[[nodiscard]] rpl::producer<int> fullCount() const; [[nodiscard]] rpl::producer<int> fullCount() const;
[[nodiscard]] rpl::producer<int> leftToLoad() const; [[nodiscard]] rpl::producer<int> loadMoreCount() const;
std::unique_ptr<PeerListState> saveState() const override; std::unique_ptr<PeerListState> saveState() const override;
void restoreState(std::unique_ptr<PeerListState> state) override; void restoreState(std::unique_ptr<PeerListState> state) override;
@ -140,8 +202,9 @@ private:
mtpRequestId _loadRequestId = 0; mtpRequestId _loadRequestId = 0;
QString _loadForOffset; QString _loadForOffset;
std::vector<not_null<UserData*>> _preloaded; std::vector<not_null<UserData*>> _preloaded;
rpl::variable<int> _leftToLoad; rpl::variable<int> _count = 0;
rpl::variable<int> _fullCount; rpl::variable<int> _fullCount;
rpl::variable<int> _leftToLoad;
rpl::event_stream<not_null<PeerData*>> _showPeerInfoRequests; rpl::event_stream<not_null<PeerData*>> _showPeerInfoRequests;
@ -159,8 +222,8 @@ ListController::ListController(
, _api(_session->api().instance()) { , _api(_session->api().instance()) {
const auto i = ranges::find(poll->answers, option, &PollAnswer::option); const auto i = ranges::find(poll->answers, option, &PollAnswer::option);
Assert(i != poll->answers.end()); Assert(i != poll->answers.end());
_leftToLoad = i->votes;
_fullCount = i->votes; _fullCount = i->votes;
_leftToLoad = i->votes;
} }
Main::Session &ListController::session() const { Main::Session &ListController::session() const {
@ -216,6 +279,7 @@ void ListController::loadMoreRows() {
} }
return data.vcount().v; return data.vcount().v;
}); });
_count = delegate()->peerListFullRowsCount();
if (_offset.isEmpty()) { if (_offset.isEmpty()) {
addPreloaded(); addPreloaded();
_fullCount = delegate()->peerListFullRowsCount(); _fullCount = delegate()->peerListFullRowsCount();
@ -241,6 +305,7 @@ void ListController::addPreloaded() {
for (const auto user : base::take(_preloaded)) { for (const auto user : base::take(_preloaded)) {
appendRow(user); appendRow(user);
} }
_leftToLoad = _fullCount.current() - delegate()->peerListFullRowsCount();
delegate()->peerListRefreshRows(); delegate()->peerListRefreshRows();
} }
@ -249,21 +314,33 @@ auto ListController::showPeerInfoRequests() const
return _showPeerInfoRequests.events(); return _showPeerInfoRequests.events();
} }
rpl::producer<int> ListController::leftToLoad() const { rpl::producer<int> ListController::count() const {
return _leftToLoad.value(); return _count.value();
} }
rpl::producer<int> ListController::fullCount() const { rpl::producer<int> ListController::fullCount() const {
return _fullCount.value(); return _fullCount.value();
} }
rpl::producer<int> ListController::loadMoreCount() const {
const auto initial = (_fullCount.current() <= kFirstPage)
? _fullCount.current()
: (kFirstPage - kLeavePreloaded);
return rpl::combine(
_count.value(),
_leftToLoad.value()
) | rpl::map([=](int count, int leftToLoad) {
return (count > 0) ? leftToLoad : (leftToLoad - initial);
});
}
auto ListController::saveState() const -> std::unique_ptr<PeerListState> { auto ListController::saveState() const -> std::unique_ptr<PeerListState> {
auto result = PeerListController::saveState(); auto result = PeerListController::saveState();
auto my = std::make_unique<SavedState>(); auto my = std::make_unique<SavedState>();
my->offset = _offset; my->offset = _offset;
my->leftToLoad = _leftToLoad.current();
my->fullCount = _fullCount.current(); my->fullCount = _fullCount.current();
my->leftToLoad = _leftToLoad.current();
my->preloaded = _preloaded; my->preloaded = _preloaded;
my->wasLoading = (_loadRequestId != 0); my->wasLoading = (_loadRequestId != 0);
my->loadForOffset = _loadForOffset; my->loadForOffset = _loadForOffset;
@ -284,11 +361,12 @@ void ListController::restoreState(std::unique_ptr<PeerListState> state) {
_offset = my->offset; _offset = my->offset;
_loadForOffset = my->loadForOffset; _loadForOffset = my->loadForOffset;
_preloaded = std::move(my->preloaded); _preloaded = std::move(my->preloaded);
_count = int(state->list.size());
_fullCount = my->fullCount;
_leftToLoad = my->leftToLoad;
if (my->wasLoading) { if (my->wasLoading) {
loadMoreRows(); loadMoreRows();
} }
_leftToLoad = my->leftToLoad;
_fullCount = my->fullCount;
PeerListController::restoreState(std::move(state)); PeerListController::restoreState(std::move(state));
} }
} }
@ -389,6 +467,19 @@ ListController *CreateAnswerRows(
delegate->setContent(content); delegate->setContent(content);
controller->setDelegate(delegate); controller->setDelegate(delegate);
const auto count = (answer.votes <= kFirstPage)
? answer.votes
: (kFirstPage - kLeavePreloaded);
const auto placeholder = container->add(object_ptr<PeerListDummy>(
container,
count,
st::infoCommonGroupsList));
controller->count(
) | rpl::filter(_1 > 0) | rpl::start_with_next([=] {
delete placeholder;
}, placeholder->lifetime());
const auto more = container->add( const auto more = container->add(
object_ptr<Ui::SlideWrap<Ui::SettingsButton>>( object_ptr<Ui::SlideWrap<Ui::SettingsButton>>(
container, container,
@ -396,13 +487,13 @@ ListController *CreateAnswerRows(
container, container,
tr::lng_polls_show_more( tr::lng_polls_show_more(
lt_count_decimal, lt_count_decimal,
controller->leftToLoad() | rpl::map(_1 + 0.), controller->loadMoreCount() | rpl::map(_1 + 0.),
Ui::Text::Upper), Ui::Text::Upper),
st::pollResultsShowMore))); st::pollResultsShowMore)));
more->entity()->setClickedCallback([=] { more->entity()->setClickedCallback([=] {
controller->allowLoadMore(); controller->allowLoadMore();
}); });
controller->leftToLoad( controller->loadMoreCount(
) | rpl::map(_1 > 0) | rpl::start_with_next([=](bool visible) { ) | rpl::map(_1 > 0) | rpl::start_with_next([=](bool visible) {
more->toggle(visible, anim::type::instant); more->toggle(visible, anim::type::instant);
}, more->lifetime()); }, more->lifetime());