624 lines
17 KiB
C++
624 lines
17 KiB
C++
/*
|
|
This file is part of Telegram Desktop,
|
|
the official desktop application for the Telegram messaging service.
|
|
|
|
For license and copyright information please follow this link:
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|
*/
|
|
#include "ui/chat/attach/attach_album_preview.h"
|
|
|
|
#include "ui/chat/attach/attach_album_thumbnail.h"
|
|
#include "ui/chat/attach/attach_prepare.h"
|
|
#include "ui/widgets/popup_menu.h"
|
|
#include "ui/painter.h"
|
|
#include "lang/lang_keys.h"
|
|
#include "styles/style_chat.h"
|
|
#include "styles/style_boxes.h"
|
|
#include "styles/style_layers.h"
|
|
#include "styles/style_menu_icons.h"
|
|
|
|
#include <QtWidgets/QApplication>
|
|
|
|
namespace Ui {
|
|
namespace {
|
|
|
|
constexpr auto kDragDuration = crl::time(200);
|
|
|
|
} // namespace
|
|
|
|
AlbumPreview::AlbumPreview(
|
|
QWidget *parent,
|
|
gsl::span<Ui::PreparedFile> items,
|
|
SendFilesWay way)
|
|
: RpWidget(parent)
|
|
, _sendWay(way)
|
|
, _dragTimer([=] { switchToDrag(); }) {
|
|
setMouseTracking(true);
|
|
prepareThumbs(items);
|
|
updateSize();
|
|
updateFileRows();
|
|
}
|
|
|
|
AlbumPreview::~AlbumPreview() = default;
|
|
|
|
void AlbumPreview::setSendWay(SendFilesWay way) {
|
|
if (_sendWay != way) {
|
|
cancelDrag();
|
|
_sendWay = way;
|
|
}
|
|
updateSize();
|
|
updateFileRows();
|
|
update();
|
|
}
|
|
|
|
void AlbumPreview::updateFileRows() {
|
|
Expects(_order.size() == _thumbs.size());
|
|
|
|
const auto isFile = !_sendWay.sendImagesAsPhotos();
|
|
auto top = 0;
|
|
for (auto i = 0; i < _order.size(); i++) {
|
|
const auto &thumb = _thumbs[_order[i]];
|
|
thumb->setButtonVisible(isFile && !thumb->isCompressedSticker());
|
|
thumb->moveButtons(top);
|
|
top += thumb->fileHeight() + st::sendMediaRowSkip;
|
|
}
|
|
}
|
|
|
|
base::flat_set<int> AlbumPreview::collectSpoileredIndices() {
|
|
auto result = base::flat_set<int>();
|
|
result.reserve(_thumbs.size());
|
|
auto i = 0;
|
|
for (const auto &thumb : _thumbs) {
|
|
if (thumb->hasSpoiler()) {
|
|
result.emplace(i);
|
|
}
|
|
++i;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
bool AlbumPreview::canHaveSpoiler(int index) const {
|
|
return _sendWay.sendImagesAsPhotos();
|
|
}
|
|
|
|
void AlbumPreview::toggleSpoilers(bool enabled) {
|
|
for (auto &thumb : _thumbs) {
|
|
thumb->setSpoiler(enabled);
|
|
}
|
|
}
|
|
|
|
std::vector<int> AlbumPreview::takeOrder() {
|
|
//Expects(_thumbs.size() == _order.size());
|
|
//Expects(_itemsShownDimensions.size() == _order.size());
|
|
|
|
auto reordered = std::vector<std::unique_ptr<AlbumThumbnail>>();
|
|
auto reorderedShownDimensions = std::vector<QSize>();
|
|
reordered.reserve(_thumbs.size());
|
|
reorderedShownDimensions.reserve(_itemsShownDimensions.size());
|
|
for (auto index : _order) {
|
|
reordered.push_back(std::move(_thumbs[index]));
|
|
reorderedShownDimensions.push_back(_itemsShownDimensions[index]);
|
|
}
|
|
_thumbs = std::move(reordered);
|
|
_itemsShownDimensions = std::move(reorderedShownDimensions);
|
|
return std::exchange(_order, defaultOrder());
|
|
}
|
|
|
|
auto AlbumPreview::generateOrderedLayout() const
|
|
-> std::vector<GroupMediaLayout> {
|
|
auto layout = LayoutMediaGroup(
|
|
_itemsShownDimensions,
|
|
st::sendMediaPreviewSize,
|
|
st::historyGroupWidthMin / 2,
|
|
st::historyGroupSkip / 2);
|
|
Assert(layout.size() == _order.size());
|
|
return layout;
|
|
}
|
|
|
|
std::vector<int> AlbumPreview::defaultOrder(int count) const {
|
|
if (count < 0) {
|
|
count = _order.size();
|
|
}
|
|
return ranges::views::ints(0, count) | ranges::to_vector;
|
|
}
|
|
|
|
void AlbumPreview::prepareThumbs(gsl::span<Ui::PreparedFile> items) {
|
|
_order = defaultOrder(items.size());
|
|
_itemsShownDimensions = ranges::views::all(
|
|
_order
|
|
) | ranges::views::transform([&](int index) {
|
|
return items[index].shownDimensions;
|
|
}) | ranges::to_vector;
|
|
|
|
const auto count = int(_order.size());
|
|
const auto layout = generateOrderedLayout();
|
|
_thumbs.reserve(count);
|
|
for (auto i = 0; i != count; ++i) {
|
|
_thumbs.push_back(std::make_unique<AlbumThumbnail>(
|
|
items[i],
|
|
layout[i],
|
|
this,
|
|
[=] { update(); },
|
|
[=] { changeThumbByIndex(thumbIndex(thumbUnderCursor())); },
|
|
[=] { deleteThumbByIndex(thumbIndex(thumbUnderCursor())); }));
|
|
if (_thumbs.back()->isCompressedSticker()) {
|
|
_hasMixedFileHeights = true;
|
|
}
|
|
}
|
|
_thumbsHeight = countLayoutHeight(layout);
|
|
_photosHeight = ranges::accumulate(ranges::views::all(
|
|
_thumbs
|
|
) | ranges::views::transform([](const auto &thumb) {
|
|
return thumb->photoHeight();
|
|
}), 0) + (count - 1) * st::sendMediaRowSkip;
|
|
|
|
if (!_hasMixedFileHeights) {
|
|
_filesHeight = count * _thumbs.front()->fileHeight()
|
|
+ (count - 1) * st::sendMediaRowSkip;
|
|
} else {
|
|
_filesHeight = ranges::accumulate(ranges::views::all(
|
|
_thumbs
|
|
) | ranges::views::transform([](const auto &thumb) {
|
|
return thumb->fileHeight();
|
|
}), 0) + (count - 1) * st::sendMediaRowSkip;
|
|
}
|
|
}
|
|
|
|
int AlbumPreview::contentLeft() const {
|
|
return (st::boxWideWidth - st::sendMediaPreviewSize) / 2;
|
|
}
|
|
|
|
int AlbumPreview::contentTop() const {
|
|
return 0;
|
|
}
|
|
|
|
AlbumThumbnail *AlbumPreview::findThumb(QPoint position) const {
|
|
position -= QPoint(contentLeft(), contentTop());
|
|
|
|
auto top = 0;
|
|
const auto isPhotosWay = _sendWay.sendImagesAsPhotos();
|
|
const auto skip = st::sendMediaRowSkip;
|
|
auto find = [&](const auto &thumb) {
|
|
if (_sendWay.groupFiles() && _sendWay.sendImagesAsPhotos()) {
|
|
return thumb->containsPoint(position);
|
|
} else {
|
|
const auto bottom = top + (isPhotosWay
|
|
? thumb->photoHeight()
|
|
: thumb->fileHeight());
|
|
const auto isUnderTop = (position.y() > top);
|
|
top = bottom + skip;
|
|
return isUnderTop && (position.y() < bottom);
|
|
}
|
|
return false;
|
|
};
|
|
|
|
const auto i = ranges::find_if(_thumbs, std::move(find));
|
|
return (i == _thumbs.end()) ? nullptr : i->get();
|
|
}
|
|
|
|
not_null<AlbumThumbnail*> AlbumPreview::findClosestThumb(
|
|
QPoint position) const {
|
|
Expects(_draggedThumb != nullptr);
|
|
|
|
if (const auto exact = findThumb(position)) {
|
|
return exact;
|
|
}
|
|
auto result = _draggedThumb;
|
|
auto distance = _draggedThumb->distanceTo(position);
|
|
for (const auto &thumb : _thumbs) {
|
|
const auto check = thumb->distanceTo(position);
|
|
if (check < distance) {
|
|
distance = check;
|
|
result = thumb.get();
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
int AlbumPreview::orderIndex(
|
|
not_null<AlbumThumbnail*> thumb) const {
|
|
const auto i = ranges::find_if(_order, [&](int index) {
|
|
return (_thumbs[index].get() == thumb);
|
|
});
|
|
Assert(i != _order.end());
|
|
return int(i - _order.begin());
|
|
}
|
|
|
|
void AlbumPreview::cancelDrag() {
|
|
_thumbsHeightAnimation.stop();
|
|
_finishDragAnimation.stop();
|
|
_shrinkAnimation.stop();
|
|
if (_draggedThumb) {
|
|
_draggedThumb->moveInAlbum({ 0, 0 });
|
|
_draggedThumb = nullptr;
|
|
}
|
|
if (_suggestedThumb) {
|
|
const auto suggestedIndex = orderIndex(_suggestedThumb);
|
|
if (suggestedIndex > 0) {
|
|
_thumbs[_order[suggestedIndex - 1]]->suggestMove(0., [] {});
|
|
}
|
|
if (suggestedIndex < int(_order.size() - 1)) {
|
|
_thumbs[_order[suggestedIndex + 1]]->suggestMove(0., [] {});
|
|
}
|
|
_suggestedThumb->suggestMove(0., [] {});
|
|
_suggestedThumb->finishAnimations();
|
|
_suggestedThumb = nullptr;
|
|
}
|
|
_paintedAbove = nullptr;
|
|
update();
|
|
}
|
|
|
|
void AlbumPreview::finishDrag() {
|
|
Expects(_draggedThumb != nullptr);
|
|
Expects(_suggestedThumb != nullptr);
|
|
|
|
if (_suggestedThumb != _draggedThumb) {
|
|
const auto currentIndex = orderIndex(_draggedThumb);
|
|
const auto newIndex = orderIndex(_suggestedThumb);
|
|
const auto delta = (currentIndex < newIndex) ? 1 : -1;
|
|
const auto realIndex = _order[currentIndex];
|
|
for (auto i = currentIndex; i != newIndex; i += delta) {
|
|
_order[i] = _order[i + delta];
|
|
}
|
|
_order[newIndex] = realIndex;
|
|
const auto layout = generateOrderedLayout();
|
|
for (auto i = 0, count = int(_order.size()); i != count; ++i) {
|
|
_thumbs[_order[i]]->moveToLayout(layout[i]);
|
|
}
|
|
_finishDragAnimation.start([=] { update(); }, 0., 1., kDragDuration);
|
|
|
|
updateSizeAnimated(layout);
|
|
} else {
|
|
for (const auto &thumb : _thumbs) {
|
|
thumb->resetLayoutAnimation();
|
|
}
|
|
_draggedThumb->animateLayoutToInitial();
|
|
_finishDragAnimation.start([=] { update(); }, 0., 1., kDragDuration);
|
|
}
|
|
}
|
|
|
|
int AlbumPreview::countLayoutHeight(
|
|
const std::vector<GroupMediaLayout> &layout) const {
|
|
const auto accumulator = [](int current, const auto &item) {
|
|
return std::max(current, item.geometry.y() + item.geometry.height());
|
|
};
|
|
return ranges::accumulate(layout, 0, accumulator);
|
|
}
|
|
|
|
void AlbumPreview::updateSizeAnimated(
|
|
const std::vector<GroupMediaLayout> &layout) {
|
|
const auto newHeight = countLayoutHeight(layout);
|
|
if (newHeight != _thumbsHeight) {
|
|
_thumbsHeightAnimation.start(
|
|
[=] { updateSize(); },
|
|
_thumbsHeight,
|
|
newHeight,
|
|
kDragDuration);
|
|
_thumbsHeight = newHeight;
|
|
}
|
|
}
|
|
|
|
void AlbumPreview::updateSize() {
|
|
const auto newHeight = [&] {
|
|
if (!_sendWay.sendImagesAsPhotos()) {
|
|
return _filesHeight;
|
|
} else if (!_sendWay.groupFiles()) {
|
|
return _photosHeight;
|
|
} else {
|
|
return int(base::SafeRound(_thumbsHeightAnimation.value(
|
|
_thumbsHeight)));
|
|
}
|
|
}();
|
|
if (height() != newHeight) {
|
|
resize(st::boxWideWidth, newHeight);
|
|
}
|
|
}
|
|
|
|
void AlbumPreview::paintEvent(QPaintEvent *e) {
|
|
Painter p(this);
|
|
|
|
if (!_sendWay.sendImagesAsPhotos()) {
|
|
paintFiles(p, e->rect());
|
|
} else if (!_sendWay.groupFiles()) {
|
|
paintPhotos(p, e->rect());
|
|
} else {
|
|
paintAlbum(p);
|
|
}
|
|
}
|
|
|
|
void AlbumPreview::paintAlbum(Painter &p) const {
|
|
const auto shrink = _shrinkAnimation.value(_draggedThumb ? 1. : 0.);
|
|
const auto moveProgress = _finishDragAnimation.value(1.);
|
|
const auto left = contentLeft();
|
|
const auto top = contentTop();
|
|
for (const auto &thumb : _thumbs) {
|
|
if (thumb.get() != _paintedAbove) {
|
|
thumb->paintInAlbum(p, left, top, shrink, moveProgress);
|
|
}
|
|
}
|
|
if (_paintedAbove) {
|
|
_paintedAbove->paintInAlbum(p, left, top, shrink, moveProgress);
|
|
}
|
|
}
|
|
|
|
void AlbumPreview::paintPhotos(Painter &p, QRect clip) const {
|
|
const auto left = (st::boxWideWidth - st::sendMediaPreviewSize) / 2;
|
|
auto top = 0;
|
|
const auto outerWidth = width();
|
|
for (const auto &thumb : _thumbs) {
|
|
const auto bottom = top + thumb->photoHeight();
|
|
const auto guard = gsl::finally([&] {
|
|
top = bottom + st::sendMediaRowSkip;
|
|
});
|
|
if (top >= clip.y() + clip.height()) {
|
|
break;
|
|
} else if (bottom <= clip.y()) {
|
|
continue;
|
|
}
|
|
thumb->paintPhoto(p, left, top, outerWidth);
|
|
}
|
|
}
|
|
|
|
void AlbumPreview::paintFiles(Painter &p, QRect clip) const {
|
|
const auto left = (st::boxWideWidth - st::sendMediaPreviewSize) / 2;
|
|
const auto outerWidth = width();
|
|
if (!_hasMixedFileHeights) {
|
|
const auto fileHeight = st::attachPreviewThumbLayout.thumbSize
|
|
+ st::sendMediaRowSkip;
|
|
const auto bottom = clip.y() + clip.height();
|
|
const auto from = std::clamp(
|
|
clip.y() / fileHeight,
|
|
0,
|
|
int(_thumbs.size()));
|
|
const auto till = std::clamp(
|
|
(bottom + fileHeight - 1) / fileHeight,
|
|
0,
|
|
int(_thumbs.size()));
|
|
|
|
auto top = from * fileHeight;
|
|
for (auto i = from; i != till; ++i) {
|
|
_thumbs[i]->paintFile(p, left, top, outerWidth);
|
|
top += fileHeight;
|
|
}
|
|
} else {
|
|
auto top = 0;
|
|
for (const auto &thumb : _thumbs) {
|
|
const auto bottom = top + thumb->fileHeight();
|
|
const auto guard = gsl::finally([&] {
|
|
top = bottom + st::sendMediaRowSkip;
|
|
});
|
|
if (top >= clip.y() + clip.height()) {
|
|
break;
|
|
} else if (bottom <= clip.y()) {
|
|
continue;
|
|
}
|
|
thumb->paintFile(p, left, top, outerWidth);
|
|
}
|
|
}
|
|
}
|
|
|
|
int AlbumPreview::thumbIndex(AlbumThumbnail *thumb) {
|
|
if (!thumb) {
|
|
return -1;
|
|
}
|
|
const auto thumbIt = ranges::find_if(_thumbs, [&](auto &t) {
|
|
return t.get() == thumb;
|
|
});
|
|
Expects(thumbIt != _thumbs.end());
|
|
return std::distance(_thumbs.begin(), thumbIt);
|
|
}
|
|
|
|
AlbumThumbnail *AlbumPreview::thumbUnderCursor() {
|
|
return findThumb(mapFromGlobal(QCursor::pos()));
|
|
}
|
|
|
|
void AlbumPreview::deleteThumbByIndex(int index) {
|
|
if (index < 0) {
|
|
return;
|
|
}
|
|
_thumbDeleted.fire(std::move(index));
|
|
}
|
|
|
|
void AlbumPreview::changeThumbByIndex(int index) {
|
|
if (index < 0) {
|
|
return;
|
|
}
|
|
_thumbChanged.fire(std::move(index));
|
|
}
|
|
|
|
void AlbumPreview::modifyThumbByIndex(int index) {
|
|
if (index < 0) {
|
|
return;
|
|
}
|
|
_thumbModified.fire(std::move(index));
|
|
}
|
|
|
|
void AlbumPreview::thumbButtonsCallback(
|
|
not_null<AlbumThumbnail*> thumb,
|
|
AttachButtonType type) {
|
|
const auto index = thumbIndex(thumb);
|
|
|
|
switch (type) {
|
|
case AttachButtonType::None: return;
|
|
case AttachButtonType::Edit: changeThumbByIndex(index); break;
|
|
case AttachButtonType::Delete: deleteThumbByIndex(index); break;
|
|
case AttachButtonType::Modify:
|
|
cancelDrag();
|
|
modifyThumbByIndex(index);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void AlbumPreview::mousePressEvent(QMouseEvent *e) {
|
|
if (_finishDragAnimation.animating()) {
|
|
return;
|
|
}
|
|
const auto position = e->pos();
|
|
cancelDrag();
|
|
if (const auto thumb = findThumb(position)) {
|
|
_draggedStartPosition = position;
|
|
_pressedThumb = thumb;
|
|
_pressedButtonType = thumb->buttonTypeFromPoint(position);
|
|
|
|
const auto isAlbum = _sendWay.sendImagesAsPhotos()
|
|
&& _sendWay.groupFiles();
|
|
if (!isAlbum || e->button() != Qt::LeftButton) {
|
|
_dragTimer.cancel();
|
|
return;
|
|
}
|
|
|
|
if (_pressedButtonType == AttachButtonType::None) {
|
|
switchToDrag();
|
|
} else if (_pressedButtonType == AttachButtonType::Modify) {
|
|
_dragTimer.callOnce(QApplication::startDragTime());
|
|
}
|
|
}
|
|
}
|
|
|
|
void AlbumPreview::mouseMoveEvent(QMouseEvent *e) {
|
|
if (!_sendWay.sendImagesAsPhotos() && !_hasMixedFileHeights) {
|
|
applyCursor(style::cur_default);
|
|
return;
|
|
}
|
|
if (_dragTimer.isActive()) {
|
|
_dragTimer.cancel();
|
|
switchToDrag();
|
|
}
|
|
const auto isAlbum = _sendWay.sendImagesAsPhotos()
|
|
&& _sendWay.groupFiles();
|
|
if (isAlbum && _draggedThumb) {
|
|
const auto position = e->pos();
|
|
_draggedThumb->moveInAlbum(position - _draggedStartPosition);
|
|
updateSuggestedDrag(_draggedThumb->center());
|
|
update();
|
|
} else {
|
|
const auto thumb = findThumb(e->pos());
|
|
const auto regularCursor = isAlbum
|
|
? style::cur_pointer
|
|
: style::cur_default;
|
|
const auto cursor = thumb
|
|
? (thumb->buttonsContainPoint(e->pos())
|
|
? style::cur_pointer
|
|
: regularCursor)
|
|
: style::cur_default;
|
|
applyCursor(cursor);
|
|
}
|
|
}
|
|
|
|
void AlbumPreview::applyCursor(style::cursor cursor) {
|
|
if (_cursor != cursor) {
|
|
_cursor = cursor;
|
|
setCursor(_cursor);
|
|
}
|
|
}
|
|
|
|
void AlbumPreview::updateSuggestedDrag(QPoint position) {
|
|
auto closest = findClosestThumb(position);
|
|
auto closestIndex = orderIndex(closest);
|
|
|
|
const auto draggedIndex = orderIndex(_draggedThumb);
|
|
const auto closestIsBeforePoint = closest->isPointAfter(position);
|
|
if (closestIndex < draggedIndex && closestIsBeforePoint) {
|
|
closest = _thumbs[_order[++closestIndex]].get();
|
|
} else if (closestIndex > draggedIndex && !closestIsBeforePoint) {
|
|
closest = _thumbs[_order[--closestIndex]].get();
|
|
}
|
|
|
|
if (_suggestedThumb == closest) {
|
|
return;
|
|
}
|
|
|
|
const auto last = int(_order.size()) - 1;
|
|
if (_suggestedThumb) {
|
|
const auto suggestedIndex = orderIndex(_suggestedThumb);
|
|
if (suggestedIndex < draggedIndex && suggestedIndex > 0) {
|
|
const auto previous = _thumbs[_order[suggestedIndex - 1]].get();
|
|
previous->suggestMove(0., [=] { update(); });
|
|
} else if (suggestedIndex > draggedIndex && suggestedIndex < last) {
|
|
const auto next = _thumbs[_order[suggestedIndex + 1]].get();
|
|
next->suggestMove(0., [=] { update(); });
|
|
}
|
|
_suggestedThumb->suggestMove(0., [=] { update(); });
|
|
}
|
|
_suggestedThumb = closest;
|
|
const auto suggestedIndex = closestIndex;
|
|
if (_suggestedThumb != _draggedThumb) {
|
|
const auto delta = (suggestedIndex < draggedIndex) ? 1. : -1.;
|
|
if (delta > 0. && suggestedIndex > 0) {
|
|
const auto previous = _thumbs[_order[suggestedIndex - 1]].get();
|
|
previous->suggestMove(-delta, [=] { update(); });
|
|
} else if (delta < 0. && suggestedIndex < last) {
|
|
const auto next = _thumbs[_order[suggestedIndex + 1]].get();
|
|
next->suggestMove(-delta, [=] { update(); });
|
|
}
|
|
_suggestedThumb->suggestMove(delta, [=] { update(); });
|
|
}
|
|
}
|
|
|
|
void AlbumPreview::mouseReleaseEvent(QMouseEvent *e) {
|
|
if (_draggedThumb) {
|
|
finishDrag();
|
|
_shrinkAnimation.start(
|
|
[=] { update(); },
|
|
1.,
|
|
0.,
|
|
AlbumThumbnail::kShrinkDuration);
|
|
_draggedThumb = nullptr;
|
|
_suggestedThumb = nullptr;
|
|
update();
|
|
} else if (const auto thumb = base::take(_pressedThumb)) {
|
|
const auto was = _pressedButtonType;
|
|
const auto now = thumb->buttonTypeFromPoint(e->pos());
|
|
if (e->button() == Qt::RightButton) {
|
|
showContextMenu(thumb, e->globalPos());
|
|
} else if (was == now) {
|
|
thumbButtonsCallback(thumb, now);
|
|
}
|
|
}
|
|
_pressedButtonType = AttachButtonType::None;
|
|
}
|
|
|
|
void AlbumPreview::showContextMenu(
|
|
not_null<AlbumThumbnail*> thumb,
|
|
QPoint position) {
|
|
if (!_sendWay.sendImagesAsPhotos()) {
|
|
return;
|
|
}
|
|
_menu = base::make_unique_q<Ui::PopupMenu>(
|
|
this,
|
|
st::popupMenuWithIcons);
|
|
|
|
const auto spoilered = thumb->hasSpoiler();
|
|
_menu->addAction(spoilered
|
|
? tr::lng_context_disable_spoiler(tr::now)
|
|
: tr::lng_context_spoiler_effect(tr::now), [=] {
|
|
thumb->setSpoiler(!spoilered);
|
|
}, spoilered ? &st::menuIconDisable : &st::menuIconSpoiler);
|
|
|
|
if (_menu->empty()) {
|
|
_menu = nullptr;
|
|
} else {
|
|
_menu->popup(position);
|
|
}
|
|
}
|
|
|
|
void AlbumPreview::switchToDrag() {
|
|
_paintedAbove
|
|
= _suggestedThumb
|
|
= _draggedThumb
|
|
= base::take(_pressedThumb);
|
|
_shrinkAnimation.start(
|
|
[=] { update(); },
|
|
0.,
|
|
1.,
|
|
AlbumThumbnail::kShrinkDuration);
|
|
applyCursor(style::cur_sizeall);
|
|
update();
|
|
}
|
|
|
|
rpl::producer<int> AlbumPreview::thumbModified() const {
|
|
return _thumbModified.events();
|
|
}
|
|
|
|
} // namespace Ui
|