Added ability to scroll vertical layout during reordering.
This commit is contained in:
parent
730816ef52
commit
d1619b728e
2 changed files with 62 additions and 3 deletions
|
|
@ -14,9 +14,18 @@
|
||||||
|
|
||||||
namespace Ui {
|
namespace Ui {
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
constexpr auto kScrollFactor = 0.05;
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
VerticalLayoutReorder::VerticalLayoutReorder(
|
VerticalLayoutReorder::VerticalLayoutReorder(
|
||||||
not_null<VerticalLayout*> layout)
|
not_null<VerticalLayout*> layout,
|
||||||
: _layout(layout) {
|
not_null<ScrollArea*> scroll)
|
||||||
|
: _layout(layout)
|
||||||
|
, _scroll(scroll)
|
||||||
|
, _scrollAnimation([=] { updateScrollCallback(); }) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void VerticalLayoutReorder::cancel() {
|
void VerticalLayoutReorder::cancel() {
|
||||||
|
|
@ -100,6 +109,8 @@ void VerticalLayoutReorder::updateOrder(int index, QPoint position) {
|
||||||
current.shift = current.finalShift = shift;
|
current.shift = current.finalShift = shift;
|
||||||
_layout->setVerticalShift(index, shift);
|
_layout->setVerticalShift(index, shift);
|
||||||
|
|
||||||
|
checkForScrollAnimation();
|
||||||
|
|
||||||
const auto count = _entries.size();
|
const auto count = _entries.size();
|
||||||
const auto currentHeight = current.widget->height();
|
const auto currentHeight = current.widget->height();
|
||||||
const auto currentMiddle = current.widget->y() + currentHeight / 2;
|
const auto currentMiddle = current.widget->y() + currentHeight / 2;
|
||||||
|
|
@ -153,6 +164,7 @@ void VerticalLayoutReorder::mouseRelease(
|
||||||
if (button != Qt::LeftButton) {
|
if (button != Qt::LeftButton) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
_scrollAnimation.stop();
|
||||||
finishCurrent();
|
finishCurrent();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -267,4 +279,41 @@ auto VerticalLayoutReorder::updates() const -> rpl::producer<Single> {
|
||||||
return _updates.events();
|
return _updates.events();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void VerticalLayoutReorder::updateScrollCallback() {
|
||||||
|
const auto delta = deltaFromEdge();
|
||||||
|
const auto oldTop = _scroll->scrollTop();
|
||||||
|
_scroll->scrollToY(oldTop + delta);
|
||||||
|
const auto newTop = _scroll->scrollTop();
|
||||||
|
|
||||||
|
_currentStart += oldTop - newTop;
|
||||||
|
if (newTop == 0 || newTop == _scroll->scrollTopMax()) {
|
||||||
|
_scrollAnimation.stop();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void VerticalLayoutReorder::checkForScrollAnimation() {
|
||||||
|
if (!deltaFromEdge() || _scrollAnimation.animating()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_scrollAnimation.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
int VerticalLayoutReorder::deltaFromEdge() {
|
||||||
|
Expects(_currentWidget != nullptr);
|
||||||
|
|
||||||
|
const auto globalPosition = _currentWidget->mapToGlobal(QPoint(0, 0));
|
||||||
|
const auto localTop = _scroll->mapFromGlobal(globalPosition).y();
|
||||||
|
const auto localBottom = localTop
|
||||||
|
+ _currentWidget->height()
|
||||||
|
- _scroll->height();
|
||||||
|
|
||||||
|
const auto isTopEdge = (localTop < 0);
|
||||||
|
const auto isBottomEdge = (localBottom > 0);
|
||||||
|
if (!isTopEdge && !isBottomEdge) {
|
||||||
|
_scrollAnimation.stop();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return int((isBottomEdge ? localBottom : localTop) * kScrollFactor);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace Ui
|
} // namespace Ui
|
||||||
|
|
|
||||||
|
|
@ -7,6 +7,7 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "ui/effects/animations.h"
|
#include "ui/effects/animations.h"
|
||||||
|
#include "ui/widgets/scroll_area.h"
|
||||||
|
|
||||||
namespace Ui {
|
namespace Ui {
|
||||||
|
|
||||||
|
|
@ -27,7 +28,9 @@ public:
|
||||||
State state = State::Started;
|
State state = State::Started;
|
||||||
};
|
};
|
||||||
|
|
||||||
VerticalLayoutReorder(not_null<VerticalLayout*> layout);
|
VerticalLayoutReorder(
|
||||||
|
not_null<VerticalLayout*> layout,
|
||||||
|
not_null<ScrollArea*> scroll);
|
||||||
|
|
||||||
void start();
|
void start();
|
||||||
void cancel();
|
void cancel();
|
||||||
|
|
@ -59,7 +62,14 @@ private:
|
||||||
void moveToShift(int index, int shift);
|
void moveToShift(int index, int shift);
|
||||||
void updateShift(not_null<RpWidget*> widget, int indexHint);
|
void updateShift(not_null<RpWidget*> widget, int indexHint);
|
||||||
|
|
||||||
|
void updateScrollCallback();
|
||||||
|
void checkForScrollAnimation();
|
||||||
|
int deltaFromEdge();
|
||||||
|
|
||||||
const not_null<Ui::VerticalLayout*> _layout;
|
const not_null<Ui::VerticalLayout*> _layout;
|
||||||
|
const not_null<Ui::ScrollArea*> _scroll;
|
||||||
|
|
||||||
|
Ui::Animations::Basic _scrollAnimation;
|
||||||
|
|
||||||
RpWidget *_currentWidget = nullptr;
|
RpWidget *_currentWidget = nullptr;
|
||||||
int _currentStart = 0;
|
int _currentStart = 0;
|
||||||
|
|
|
||||||
Loading…
Add table
Reference in a new issue