From 1fc7dabd3ea7a844372806aae24f5c6bfa1d4df8 Mon Sep 17 00:00:00 2001 From: John Preston Date: Mon, 25 Dec 2017 22:26:08 +0300 Subject: [PATCH] Allow media reordering when sending an album. --- Telegram/SourceFiles/app.cpp | 29 +- Telegram/SourceFiles/boxes/send_files_box.cpp | 905 +++++++++++++++--- Telegram/SourceFiles/boxes/send_files_box.h | 2 +- .../info/media/info_media_list_widget.cpp | 2 +- .../storage/storage_media_prepare.cpp | 28 +- .../storage/storage_media_prepare.h | 3 + Telegram/SourceFiles/ui/grouped_layout.cpp | 6 +- Telegram/SourceFiles/ui/images.cpp | 56 +- Telegram/SourceFiles/ui/images.h | 3 +- 9 files changed, 832 insertions(+), 202 deletions(-) diff --git a/Telegram/SourceFiles/app.cpp b/Telegram/SourceFiles/app.cpp index 8a544b17f..2639348cf 100644 --- a/Telegram/SourceFiles/app.cpp +++ b/Telegram/SourceFiles/app.cpp @@ -2015,15 +2015,6 @@ namespace { } } - int msgRadius() { - static int MsgRadius = ([]() { - return st::historyMessageRadius; - auto minMsgHeight = (st::msgPadding.top() + st::msgFont->height + st::msgPadding.bottom()); - return minMsgHeight / 2; - })(); - return MsgRadius; - } - void createMaskCorners() { QImage mask[4]; prepareCorners(SmallMaskCorners, st::buttonRadius, QColor(255, 255, 255), nullptr, mask); @@ -2031,7 +2022,7 @@ namespace { ::cornersMaskSmall[i] = mask[i].convertToFormat(QImage::Format_ARGB32_Premultiplied); ::cornersMaskSmall[i].setDevicePixelRatio(cRetinaFactor()); } - prepareCorners(LargeMaskCorners, msgRadius(), QColor(255, 255, 255), nullptr, mask); + prepareCorners(LargeMaskCorners, st::historyMessageRadius, QColor(255, 255, 255), nullptr, mask); for (int i = 0; i < 4; ++i) { ::cornersMaskLarge[i] = mask[i].convertToFormat(QImage::Format_ARGB32_Premultiplied); ::cornersMaskLarge[i].setDevicePixelRatio(cRetinaFactor()); @@ -2045,12 +2036,12 @@ namespace { prepareCorners(StickerCorners, st::dateRadius, st::msgServiceBg); prepareCorners(StickerSelectedCorners, st::dateRadius, st::msgServiceBgSelected); prepareCorners(SelectedOverlaySmallCorners, st::buttonRadius, st::msgSelectOverlay); - prepareCorners(SelectedOverlayLargeCorners, msgRadius(), st::msgSelectOverlay); + prepareCorners(SelectedOverlayLargeCorners, st::historyMessageRadius, st::msgSelectOverlay); prepareCorners(DateCorners, st::dateRadius, st::msgDateImgBg); prepareCorners(DateSelectedCorners, st::dateRadius, st::msgDateImgBgSelected); - prepareCorners(InShadowCorners, msgRadius(), st::msgInShadow); - prepareCorners(InSelectedShadowCorners, msgRadius(), st::msgInShadowSelected); - prepareCorners(ForwardCorners, msgRadius(), st::historyForwardChooseBg); + prepareCorners(InShadowCorners, st::historyMessageRadius, st::msgInShadow); + prepareCorners(InSelectedShadowCorners, st::historyMessageRadius, st::msgInShadowSelected); + prepareCorners(ForwardCorners, st::historyMessageRadius, st::historyForwardChooseBg); prepareCorners(MediaviewSaveCorners, st::mediaviewControllerRadius, st::mediaviewSaveMsgBg); prepareCorners(EmojiHoverCorners, st::buttonRadius, st::emojiPanHover); prepareCorners(StickerHoverCorners, st::buttonRadius, st::emojiPanHover); @@ -2062,10 +2053,10 @@ namespace { prepareCorners(Doc3Corners, st::buttonRadius, st::msgFile3Bg); prepareCorners(Doc4Corners, st::buttonRadius, st::msgFile4Bg); - prepareCorners(MessageInCorners, msgRadius(), st::msgInBg, &st::msgInShadow); - prepareCorners(MessageInSelectedCorners, msgRadius(), st::msgInBgSelected, &st::msgInShadowSelected); - prepareCorners(MessageOutCorners, msgRadius(), st::msgOutBg, &st::msgOutShadow); - prepareCorners(MessageOutSelectedCorners, msgRadius(), st::msgOutBgSelected, &st::msgOutShadowSelected); + prepareCorners(MessageInCorners, st::historyMessageRadius, st::msgInBg, &st::msgInShadow); + prepareCorners(MessageInSelectedCorners, st::historyMessageRadius, st::msgInBgSelected, &st::msgInShadowSelected); + prepareCorners(MessageOutCorners, st::historyMessageRadius, st::msgOutBg, &st::msgOutShadow); + prepareCorners(MessageOutSelectedCorners, st::historyMessageRadius, st::msgOutBgSelected, &st::msgOutShadowSelected); } void createCorners() { @@ -2663,7 +2654,7 @@ namespace { QImage images[4]; switch (radius) { case ImageRoundRadius::Small: prepareCorners(SmallMaskCorners, st::buttonRadius, bg, nullptr, images); break; - case ImageRoundRadius::Large: prepareCorners(LargeMaskCorners, msgRadius(), bg, nullptr, images); break; + case ImageRoundRadius::Large: prepareCorners(LargeMaskCorners, st::historyMessageRadius, bg, nullptr, images); break; default: p.fillRect(x, y, w, h, bg); return; } diff --git a/Telegram/SourceFiles/boxes/send_files_box.cpp b/Telegram/SourceFiles/boxes/send_files_box.cpp index 2fa19880e..aa179518a 100644 --- a/Telegram/SourceFiles/boxes/send_files_box.cpp +++ b/Telegram/SourceFiles/boxes/send_files_box.cpp @@ -40,6 +40,8 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org namespace { constexpr auto kMinPreviewWidth = 20; +constexpr auto kShrinkDuration = TimeMs(150); +constexpr auto kDragDuration = TimeMs(200); class SingleMediaPreview : public Ui::RpWidget { public: @@ -106,6 +108,456 @@ private: }; +class AlbumThumb { +public: + AlbumThumb( + const Storage::PreparedFile &file, + const Ui::GroupMediaLayout &layout); + + void moveToLayout(const Ui::GroupMediaLayout &layout); + + int photoHeight() const; + + void paintInAlbum( + Painter &p, + int left, + int top, + float64 shrinkProgress, + float64 moveProgress, + TimeMs ms); + void paintPhoto(Painter &p, int left, int top, int outerWidth); + void paintFile(Painter &p, int left, int top, int outerWidth); + + bool containsPoint(QPoint position) const; + int distanceTo(QPoint position) const; + bool isPointAfter(QPoint position) const; + void moveInAlbum(QPoint to); + QPoint center() const; + void suggestMove(float64 delta, base::lambda callback); + void finishAnimations(); + +private: + QRect countRealGeometry() const; + QRect countCurrentGeometry(float64 progress) const; + void prepareCache(QSize size, int shrink); + void drawSimpleFrame(Painter &p, QRect to, QSize size) const; + + Ui::GroupMediaLayout _layout; + QRect _fromGeometry; + const QImage _fullPreview; + const int _shrinkSize = 0; + QPixmap _albumImage; + QImage _albumCache; + QPoint _albumPosition; + RectParts _albumCorners = RectPart::None; + QPixmap _photo; + QPixmap _fileThumb; + QString _name; + QString _status; + int _nameWidth = 0; + int _statusWidth = 0; + bool _isVideo = false; + float64 _suggestedMove = 0.; + Animation _suggestedMoveAnimation; + int _lastShrinkValue = 0; + +}; + +AlbumThumb::AlbumThumb( + const Storage::PreparedFile &file, + const Ui::GroupMediaLayout &layout) +: _layout(layout) +, _fullPreview(file.preview) +, _shrinkSize(int(std::ceil(st::historyMessageRadius / 1.4))) +, _isVideo(file.type == Storage::PreparedFile::AlbumType::Video) { + Expects(!_fullPreview.isNull()); + + moveToLayout(layout); + + using Option = Images::Option; + const auto previewWidth = _fullPreview.width(); + const auto previewHeight = _fullPreview.height(); + const auto imageWidth = std::max( + previewWidth / cIntRetinaFactor(), + st::minPhotoSize); + const auto imageHeight = std::max( + previewHeight / cIntRetinaFactor(), + st::minPhotoSize); + _photo = App::pixmapFromImageInPlace(Images::prepare( + _fullPreview, + previewWidth, + previewHeight, + Option::RoundedLarge | Option::RoundedAll, + imageWidth, + imageHeight)); + + const auto idealSize = st::sendMediaFileThumbSize * cIntRetinaFactor(); + const auto fileThumbSize = (previewWidth > previewHeight) + ? QSize(previewWidth * idealSize / previewHeight, idealSize) + : QSize(idealSize, previewHeight * idealSize / previewWidth); + _fileThumb = App::pixmapFromImageInPlace(Images::prepare( + _fullPreview, + fileThumbSize.width(), + fileThumbSize.height(), + Option::RoundedSmall | Option::RoundedAll, + st::sendMediaFileThumbSize, + st::sendMediaFileThumbSize + )); + + const auto availableFileWidth = st::sendMediaPreviewSize + - st::sendMediaFileThumbSkip + - st::sendMediaFileThumbSize; + const auto filepath = file.path; + if (filepath.isEmpty()) { + _name = filedialogDefaultName( + qsl("image"), + qsl(".png"), + QString(), + true); + _status = qsl("%1x%2").arg( + _fullPreview.width() + ).arg( + _fullPreview.height() + ); + } else { + auto fileinfo = QFileInfo(filepath); + _name = fileinfo.fileName(); + _status = formatSizeText(fileinfo.size()); + } + _nameWidth = st::semiboldFont->width(_name); + if (_nameWidth > availableFileWidth) { + _name = st::semiboldFont->elided( + _name, + Qt::ElideMiddle); + _nameWidth = st::semiboldFont->width(_name); + } + _statusWidth = st::normalFont->width(_status); +} + +void AlbumThumb::moveToLayout(const Ui::GroupMediaLayout &layout) { + _fromGeometry = countRealGeometry(); + _layout = layout; + _suggestedMove = 0.; + _albumPosition = QPoint(0, 0); + + const auto width = _layout.geometry.width(); + const auto height = _layout.geometry.height(); + _albumCorners = Ui::GetCornersFromSides(_layout.sides); + using Option = Images::Option; + const auto options = Option::Smooth + | Option::RoundedLarge + | ((_albumCorners & RectPart::TopLeft) + ? Option::RoundedTopLeft + : Option::None) + | ((_albumCorners & RectPart::TopRight) + ? Option::RoundedTopRight + : Option::None) + | ((_albumCorners & RectPart::BottomLeft) + ? Option::RoundedBottomLeft + : Option::None) + | ((_albumCorners & RectPart::BottomRight) + ? Option::RoundedBottomRight + : Option::None); + const auto pixSize = Ui::GetImageScaleSizeForGeometry( + { _fullPreview.width(), _fullPreview.height() }, + { width, height }); + const auto pixWidth = pixSize.width() * cIntRetinaFactor(); + const auto pixHeight = pixSize.height() * cIntRetinaFactor(); + + _albumImage = App::pixmapFromImageInPlace(Images::prepare( + _fullPreview, + pixWidth, + pixHeight, + options, + width, + height)); +} + +int AlbumThumb::photoHeight() const { + return _photo.height() / cIntRetinaFactor(); +} + +void AlbumThumb::paintInAlbum( + Painter &p, + int left, + int top, + float64 shrinkProgress, + float64 moveProgress, + TimeMs ms) { + const auto shrink = anim::interpolate(0, _shrinkSize, shrinkProgress); + _suggestedMoveAnimation.step(ms); + _lastShrinkValue = shrink; + const auto geometry = countCurrentGeometry(moveProgress); + const auto x = left + geometry.x(); + const auto y = top + geometry.y(); + if (shrink > 0 || moveProgress < 1.) { + const auto size = geometry.size(); + if (shrinkProgress < 1 && _albumCorners != RectPart::None) { + prepareCache(size, shrink); + p.drawImage(x, y, _albumCache); + } else { + const auto to = QRect({ x, y }, size).marginsRemoved( + { shrink, shrink, shrink, shrink } + ); + drawSimpleFrame(p, to, size); + } + } else { + p.drawPixmap(x, y, _albumImage); + } + if (_isVideo) { + const auto inner = QRect( + x + (geometry.width() - st::msgFileSize) / 2, + y + (geometry.height() - st::msgFileSize) / 2, + st::msgFileSize, + st::msgFileSize); + { + PainterHighQualityEnabler hq(p); + p.setPen(Qt::NoPen); + p.setBrush(st::msgDateImgBg); + p.drawEllipse(inner); + } + st::historyFileThumbPlay.paintInCenter(p, inner); + } +} + +void AlbumThumb::prepareCache(QSize size, int shrink) { + const auto width = std::max( + _layout.geometry.width(), + _fromGeometry.width()); + const auto height = std::max( + _layout.geometry.height(), + _fromGeometry.height()); + const auto cacheSize = QSize(width, height) * cIntRetinaFactor(); + const auto initial = QRect(QPoint(), size); + + if (_albumCache.width() < cacheSize.width() + || _albumCache.height() < cacheSize.height()) { + _albumCache = QImage(cacheSize, QImage::Format_ARGB32_Premultiplied); + } + _albumCache.fill(Qt::transparent); + { + Painter p(&_albumCache); + const auto to = initial.marginsRemoved( + { shrink, shrink, shrink, shrink } + ); + drawSimpleFrame(p, to, size); + } + Images::prepareRound( + _albumCache, + ImageRoundRadius::Large, + _albumCorners, + initial); +} + +void AlbumThumb::drawSimpleFrame(Painter &p, QRect to, QSize size) const { + const auto fullWidth = _fullPreview.width(); + const auto fullHeight = _fullPreview.height(); + const auto previewSize = Ui::GetImageScaleSizeForGeometry( + { fullWidth, fullHeight }, + { size.width(), size.height() }); + const auto previewWidth = previewSize.width() * cIntRetinaFactor(); + const auto previewHeight = previewSize.height() * cIntRetinaFactor(); + const auto width = size.width() * cIntRetinaFactor(); + const auto height = size.height() * cIntRetinaFactor(); + const auto scaleWidth = to.width() / float64(width); + const auto scaleHeight = to.height() / float64(height); + const auto Round = [](float64 value) { + return int(std::round(value)); + }; + const auto [from, fillBlack] = [&] { + if (previewWidth < width && previewHeight < height) { + const auto toWidth = Round(previewWidth * scaleWidth); + const auto toHeight = Round(previewHeight * scaleHeight); + return std::make_pair( + QRect(0, 0, fullWidth, fullHeight), + QMargins( + (to.width() - toWidth) / 2, + (to.height() - toHeight) / 2, + to.width() - toWidth - (to.width() - toWidth) / 2, + to.height() - toHeight - (to.height() - toHeight) / 2)); + } else if (previewWidth * height > previewHeight * width) { + if (previewHeight >= height) { + const auto takeWidth = previewWidth * height / previewHeight; + const auto useWidth = fullWidth * width / takeWidth; + return std::make_pair( + QRect( + (fullWidth - useWidth) / 2, + 0, + useWidth, + fullHeight), + QMargins(0, 0, 0, 0)); + } else { + const auto takeWidth = previewWidth; + const auto useWidth = fullWidth * width / takeWidth; + const auto toHeight = Round(previewHeight * scaleHeight); + const auto toSkip = (to.height() - toHeight) / 2; + return std::make_pair( + QRect( + (fullWidth - useWidth) / 2, + 0, + useWidth, + fullHeight), + QMargins( + 0, + toSkip, + 0, + to.height() - toHeight - toSkip)); + } + } else { + if (previewWidth >= width) { + const auto takeHeight = previewHeight * width / previewWidth; + const auto useHeight = fullHeight * height / takeHeight; + return std::make_pair( + QRect( + 0, + (fullHeight - useHeight) / 2, + fullWidth, + useHeight), + QMargins(0, 0, 0, 0)); + } else { + const auto takeHeight = previewHeight; + const auto useHeight = fullHeight * height / takeHeight; + const auto toWidth = Round(previewWidth * scaleWidth); + const auto toSkip = (to.width() - toWidth) / 2; + return std::make_pair( + QRect( + 0, + (fullHeight - useHeight) / 2, + fullWidth, + useHeight), + QMargins( + toSkip, + 0, + to.width() - toWidth - toSkip, + 0)); + } + } + }(); + + p.drawImage(to.marginsRemoved(fillBlack), _fullPreview, from); + if (fillBlack.top() > 0) { + p.fillRect(to.x(), to.y(), to.width(), fillBlack.top(), st::imageBg); + } + if (fillBlack.bottom() > 0) { + p.fillRect( + to.x(), + to.y() + to.height() - fillBlack.bottom(), + to.width(), + fillBlack.bottom(), + st::imageBg); + } + if (fillBlack.left() > 0) { + p.fillRect( + to.x(), + to.y() + fillBlack.top(), + fillBlack.left(), + to.height() - fillBlack.top() - fillBlack.bottom(), + st::imageBg); + } + if (fillBlack.right() > 0) { + p.fillRect( + to.x() + to.width() - fillBlack.right(), + to.y() + fillBlack.top(), + fillBlack.right(), + to.height() - fillBlack.top() - fillBlack.bottom(), + st::imageBg); + } +} + +void AlbumThumb::paintPhoto(Painter &p, int left, int top, int outerWidth) { + const auto width = _photo.width() / cIntRetinaFactor(); + p.drawPixmapLeft( + left + (st::sendMediaPreviewSize - width) / 2, + top, + outerWidth, + _photo); +} + +void AlbumThumb::paintFile(Painter &p, int left, int top, int outerWidth) { + const auto textLeft = left + + st::sendMediaFileThumbSize + + st::sendMediaFileThumbSkip; + + p.drawPixmap(left, top, _fileThumb); + p.setFont(st::semiboldFont); + p.setPen(st::historyFileNameInFg); + p.drawTextLeft( + textLeft, + top + st::sendMediaFileNameTop, + outerWidth, + _name, + _nameWidth); + p.setFont(st::normalFont); + p.setPen(st::mediaInFg); + p.drawTextLeft( + textLeft, + top + st::sendMediaFileStatusTop, + outerWidth, + _status, + _statusWidth); +} + +bool AlbumThumb::containsPoint(QPoint position) const { + return _layout.geometry.contains(position); +} + +int AlbumThumb::distanceTo(QPoint position) const { + const auto delta = (_layout.geometry.center() - position); + return QPoint::dotProduct(delta, delta); +} + +bool AlbumThumb::isPointAfter(QPoint position) const { + return position.x() > _layout.geometry.center().x(); +} + +void AlbumThumb::moveInAlbum(QPoint to) { + _albumPosition = to; +} + +QPoint AlbumThumb::center() const { + auto realGeometry = _layout.geometry; + realGeometry.moveTopLeft(realGeometry.topLeft() + _albumPosition); + return realGeometry.center(); +} + +void AlbumThumb::suggestMove(float64 delta, base::lambda callback) { + if (_suggestedMove != delta) { + _suggestedMoveAnimation.start( + std::move(callback), + _suggestedMove, + delta, + kShrinkDuration); + _suggestedMove = delta; + } +} + +QRect AlbumThumb::countRealGeometry() const { + const auto addLeft = int(std::round( + _suggestedMoveAnimation.current(_suggestedMove) * _lastShrinkValue)); + const auto current = _layout.geometry; + const auto realTopLeft = current.topLeft() + + _albumPosition + + QPoint(addLeft, 0); + return { realTopLeft, current.size() }; +} + +QRect AlbumThumb::countCurrentGeometry(float64 progress) const { + const auto now = countRealGeometry(); + if (progress < 1.) { + return { + anim::interpolate(_fromGeometry.x(), now.x(), progress), + anim::interpolate(_fromGeometry.y(), now.y(), progress), + anim::interpolate(_fromGeometry.width(), now.width(), progress), + anim::interpolate(_fromGeometry.height(), now.height(), progress) + }; + } + return now; +} + +void AlbumThumb::finishAnimations() { + _suggestedMoveAnimation.finish(); +} + SingleMediaPreview *SingleMediaPreview::Create( QWidget *parent, not_null controller, @@ -438,18 +890,6 @@ base::lambda FieldPlaceholder(const Storage::PreparedList &list) { } // namespace -struct SendFilesBox::AlbumThumb { - Ui::GroupMediaLayout layout; - QPixmap albumImage; - QPixmap photo; - QPixmap fileThumb; - QString name; - QString status; - int nameWidth = 0; - int statusWidth = 0; - bool video = false; -}; - class SendFilesBox::AlbumPreview : public Ui::RpWidget { public: AlbumPreview( @@ -458,28 +898,53 @@ public: SendFilesWay way); void setSendWay(SendFilesWay way); + std::vector takeOrder(); protected: void paintEvent(QPaintEvent *e) override; + void mousePressEvent(QMouseEvent *e) override; + void mouseMoveEvent(QMouseEvent *e) override; + void mouseReleaseEvent(QMouseEvent *e) override; private: + int countLayoutHeight( + const std::vector &layout) const; + std::vector generateOrderedLayout() const; + std::vector defaultOrder() const; void prepareThumbs(); + void updateSizeAnimated(const std::vector &layout); void updateSize(); - AlbumThumb prepareThumb( - const Storage::PreparedFile &file, - const Ui::GroupMediaLayout &layout) const; void paintAlbum(Painter &p) const; void paintPhotos(Painter &p, QRect clip) const; void paintFiles(Painter &p, QRect clip) const; + int contentLeft() const; + int contentTop() const; + AlbumThumb *findThumb(QPoint position) const; + not_null findClosestThumb(QPoint position) const; + void updateSuggestedDrag(QPoint position); + int orderIndex(not_null thumb) const; + void cancelDrag(); + void finishDrag(); + const Storage::PreparedList &_list; SendFilesWay _sendWay = SendFilesWay::Files; - std::vector _thumbs; + std::vector _order; + std::vector> _thumbs; int _thumbsHeight = 0; int _photosHeight = 0; int _filesHeight = 0; + AlbumThumb *_draggedThumb = nullptr; + AlbumThumb *_suggestedThumb = nullptr; + AlbumThumb *_paintedAbove = nullptr; + QPoint _draggedStartPosition; + + mutable Animation _thumbsHeightAnimation; + mutable Animation _shrinkAnimation; + mutable Animation _finishDragAnimation; + }; SendFilesBox::AlbumPreview::AlbumPreview( @@ -489,139 +954,202 @@ SendFilesBox::AlbumPreview::AlbumPreview( : RpWidget(parent) , _list(list) , _sendWay(way) { + setMouseTracking(true); prepareThumbs(); updateSize(); } void SendFilesBox::AlbumPreview::setSendWay(SendFilesWay way) { - _sendWay = way; + if (_sendWay != way) { + cancelDrag(); + _sendWay = way; + } updateSize(); update(); } -void SendFilesBox::AlbumPreview::prepareThumbs() { +std::vector SendFilesBox::AlbumPreview::takeOrder() { + auto reordered = std::vector>(); + reordered.reserve(_thumbs.size()); + for (auto index : _order) { + reordered.push_back(std::move(_thumbs[index])); + } + _thumbs = std::move(reordered); + return std::exchange(_order, defaultOrder()); +} + +auto SendFilesBox::AlbumPreview::generateOrderedLayout() const +-> std::vector { auto sizes = ranges::view::all( - _list.files - ) | ranges::view::transform([](const Storage::PreparedFile &file) { - return file.preview.size() / cIntRetinaFactor(); + _order + ) | ranges::view::transform([&](int index) { + return _list.files[index].preview.size() / cIntRetinaFactor(); }) | ranges::to_vector; - const auto count = int(sizes.size()); - const auto layout = Ui::LayoutMediaGroup( + auto layout = Ui::LayoutMediaGroup( sizes, st::sendMediaPreviewSize, st::historyGroupWidthMin / 2, st::historyGroupSkip / 2); - Assert(layout.size() == count); + Assert(layout.size() == _order.size()); + return layout; +} +std::vector SendFilesBox::AlbumPreview::defaultOrder() const { + const auto count = int(_list.files.size()); + return ranges::view::ints(0, count) | ranges::to_vector; +} + +void SendFilesBox::AlbumPreview::prepareThumbs() { + _order = defaultOrder(); + + const auto count = int(_list.files.size()); + const auto layout = generateOrderedLayout(); _thumbs.reserve(count); for (auto i = 0; i != count; ++i) { - _thumbs.push_back(prepareThumb(_list.files[i], layout[i])); - const auto &geometry = layout[i].geometry; - accumulate_max(_thumbsHeight, geometry.y() + geometry.height()); + _thumbs.push_back(std::make_unique( + _list.files[i], + layout[i])); } + _thumbsHeight = countLayoutHeight(layout); _photosHeight = ranges::accumulate(ranges::view::all( _thumbs - ) | ranges::view::transform([](const AlbumThumb &thumb) { - return thumb.photo.height() / cIntRetinaFactor(); + ) | ranges::view::transform([](const auto &thumb) { + return thumb->photoHeight(); }), 0) + (count - 1) * st::sendMediaPreviewPhotoSkip; _filesHeight = count * st::sendMediaFileThumbSize + (count - 1) * st::sendMediaFileThumbSkip; } -SendFilesBox::AlbumThumb SendFilesBox::AlbumPreview::prepareThumb( - const Storage::PreparedFile &file, - const Ui::GroupMediaLayout &layout) const { - Expects(!file.preview.isNull()); +int SendFilesBox::AlbumPreview::contentLeft() const { + return (st::boxWideWidth - st::sendMediaPreviewSize) / 2; +} - const auto &preview = file.preview; - auto result = AlbumThumb(); - result.layout = layout; - result.video = (file.type == Storage::PreparedFile::AlbumType::Video); +int SendFilesBox::AlbumPreview::contentTop() const { + return 0; +} - const auto width = layout.geometry.width(); - const auto height = layout.geometry.height(); - const auto corners = Ui::GetCornersFromSides(layout.sides); - using Option = Images::Option; - const auto options = Option::Smooth - | Option::RoundedLarge - | ((corners & RectPart::TopLeft) ? Option::RoundedTopLeft : Option::None) - | ((corners & RectPart::TopRight) ? Option::RoundedTopRight : Option::None) - | ((corners & RectPart::BottomLeft) ? Option::RoundedBottomLeft : Option::None) - | ((corners & RectPart::BottomRight) ? Option::RoundedBottomRight : Option::None); - const auto pixSize = Ui::GetImageScaleSizeForGeometry( - { preview.width(), preview.height() }, - { width, height }); - const auto pixWidth = pixSize.width() * cIntRetinaFactor(); - const auto pixHeight = pixSize.height() * cIntRetinaFactor(); +AlbumThumb *SendFilesBox::AlbumPreview::findThumb(QPoint position) const { + position -= QPoint(contentLeft(), contentTop()); + const auto i = ranges::find_if(_thumbs, [&](const auto &thumb) { + return thumb->containsPoint(position); + }); + return (i == _thumbs.end()) ? nullptr : i->get(); +} - result.albumImage = App::pixmapFromImageInPlace(Images::prepare( - preview, - pixWidth, - pixHeight, - options, - width, - height)); - result.photo = App::pixmapFromImageInPlace(Images::prepare( - preview, - preview.width(), - preview.height(), - Option::RoundedLarge | Option::RoundedAll, - preview.width() / cIntRetinaFactor(), - preview.height() / cIntRetinaFactor())); +not_null SendFilesBox::AlbumPreview::findClosestThumb( + QPoint position) const { + Expects(_draggedThumb != nullptr); - const auto idealSize = st::sendMediaFileThumbSize * cIntRetinaFactor(); - const auto fileThumbSize = (preview.width() > preview.height()) - ? QSize(preview.width() * idealSize / preview.height(), idealSize) - : QSize(idealSize, preview.height() * idealSize / preview.width()); - result.fileThumb = App::pixmapFromImageInPlace(Images::prepare( - preview, - fileThumbSize.width(), - fileThumbSize.height(), - Option::RoundedSmall | Option::RoundedAll, - st::sendMediaFileThumbSize, - st::sendMediaFileThumbSize - )); - - const auto availableFileWidth = st::sendMediaPreviewSize - - st::sendMediaFileThumbSkip - - st::sendMediaFileThumbSize; - const auto filepath = file.path; - if (filepath.isEmpty()) { - result.name = filedialogDefaultName( - qsl("image"), - qsl(".png"), - QString(), - true); - result.status = qsl("%1x%2").arg(preview.width()).arg(preview.height()); - } else { - auto fileinfo = QFileInfo(filepath); - result.name = fileinfo.fileName(); - result.status = formatSizeText(fileinfo.size()); + if (const auto exact = findThumb(position)) { + return exact; } - result.nameWidth = st::semiboldFont->width(result.name); - if (result.nameWidth > availableFileWidth) { - result.name = st::semiboldFont->elided( - result.name, - Qt::ElideMiddle); - result.nameWidth = st::semiboldFont->width(result.name); + 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(); + } } - result.statusWidth = st::normalFont->width(result.status); - return result; } +int SendFilesBox::AlbumPreview::orderIndex( + not_null 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 SendFilesBox::AlbumPreview::cancelDrag() { + _thumbsHeightAnimation.finish(); + _finishDragAnimation.finish(); + _shrinkAnimation.finish(); + 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 SendFilesBox::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 { + _draggedThumb->moveInAlbum(QPoint()); + } +} + +int SendFilesBox::AlbumPreview::countLayoutHeight( + const std::vector &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 SendFilesBox::AlbumPreview::updateSizeAnimated( + const std::vector &layout) { + const auto newHeight = countLayoutHeight(layout); + if (newHeight != _thumbsHeight) { + _thumbsHeightAnimation.start( + [=] { updateSize(); }, + _thumbsHeight, + newHeight, + kDragDuration); + _thumbsHeight = newHeight; + } +} + void SendFilesBox::AlbumPreview::updateSize() { - const auto height = [&] { + const auto newHeight = [&] { switch (_sendWay) { - case SendFilesWay::Album: return _thumbsHeight; + case SendFilesWay::Album: + return int(std::round(_thumbsHeightAnimation.current( + _thumbsHeight))); case SendFilesWay::Photos: return _photosHeight; case SendFilesWay::Files: return _filesHeight; } Unexpected("Send way in SendFilesBox::AlbumPreview::updateSize"); }(); - resize(st::boxWideWidth, height); + if (height() != newHeight) { + resize(st::boxWideWidth, newHeight); + } } void SendFilesBox::AlbumPreview::paintEvent(QPaintEvent *e) { @@ -635,36 +1163,29 @@ void SendFilesBox::AlbumPreview::paintEvent(QPaintEvent *e) { } void SendFilesBox::AlbumPreview::paintAlbum(Painter &p) const { - const auto left = (st::boxWideWidth - st::sendMediaPreviewSize) / 2; - const auto top = 0; + const auto ms = getms(); + const auto shrink = _shrinkAnimation.current( + ms, + _draggedThumb ? 1. : 0.); + const auto moveProgress = _finishDragAnimation.current(ms, 1.); + const auto left = contentLeft(); + const auto top = contentTop(); for (const auto &thumb : _thumbs) { - const auto geometry = thumb.layout.geometry; - const auto x = left + geometry.x(); - const auto y = top + geometry.y(); - p.drawPixmap(x, y, thumb.albumImage); - - if (thumb.video) { - const auto inner = QRect( - x + (geometry.width() - st::msgFileSize) / 2, - y + (geometry.height() - st::msgFileSize) / 2, - st::msgFileSize, - st::msgFileSize); - { - PainterHighQualityEnabler hq(p); - p.setPen(Qt::NoPen); - p.setBrush(st::msgDateImgBg); - p.drawEllipse(inner); - } - st::historyFileThumbPlay.paintInCenter(p, inner); + if (thumb.get() != _paintedAbove) { + thumb->paintInAlbum(p, left, top, shrink, moveProgress, ms); } } + if (_paintedAbove) { + _paintedAbove->paintInAlbum(p, left, top, shrink, moveProgress, ms); + } } void SendFilesBox::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.photo.height() / cIntRetinaFactor(); + const auto bottom = top + thumb->photoHeight(); const auto guard = gsl::finally([&] { top = bottom + st::sendMediaPreviewPhotoSkip; }); @@ -673,10 +1194,7 @@ void SendFilesBox::AlbumPreview::paintPhotos(Painter &p, QRect clip) const { } else if (bottom <= clip.y()) { continue; } - p.drawPixmap( - left, - top, - thumb.photo); + thumb->paintPhoto(p, left, top, outerWidth); } } @@ -687,34 +1205,95 @@ void SendFilesBox::AlbumPreview::paintFiles(Painter &p, QRect clip) const { const auto from = floorclamp(clip.y(), fileHeight, 0, _thumbs.size()); const auto till = ceilclamp(bottom, fileHeight, 0, _thumbs.size()); const auto left = (st::boxWideWidth - st::sendMediaPreviewSize) / 2; - const auto textLeft = left - + st::sendMediaFileThumbSize - + st::sendMediaFileThumbSkip; + const auto outerWidth = width(); auto top = from * fileHeight; for (auto i = from; i != till; ++i) { - const auto &thumb = _thumbs[i]; - p.drawPixmap(left, top, thumb.fileThumb); - p.setFont(st::semiboldFont); - p.setPen(st::historyFileNameInFg); - p.drawTextLeft( - textLeft, - top + st::sendMediaFileNameTop, - width(), - thumb.name, - thumb.nameWidth); - p.setFont(st::normalFont); - p.setPen(st::mediaInFg); - p.drawTextLeft( - textLeft, - top + st::sendMediaFileStatusTop, - width(), - thumb.status, - thumb.statusWidth); + _thumbs[i]->paintFile(p, left, top, outerWidth); top += fileHeight; } } +void SendFilesBox::AlbumPreview::mousePressEvent(QMouseEvent *e) { + if (_finishDragAnimation.animating()) { + return; + } + const auto position = e->pos(); + cancelDrag(); + if (const auto thumb = findThumb(position)) { + _paintedAbove = _suggestedThumb = _draggedThumb = thumb; + _draggedStartPosition = position; + _shrinkAnimation.start([=] { update(); }, 0., 1., kShrinkDuration); + } +} + +void SendFilesBox::AlbumPreview::mouseMoveEvent(QMouseEvent *e) { + if (_draggedThumb) { + const auto position = e->pos(); + _draggedThumb->moveInAlbum(position - _draggedStartPosition); + updateSuggestedDrag(_draggedThumb->center()); + update(); + } else { + const auto cursor = findThumb(e->pos()) + ? style::cur_sizeall + : style::cur_default; + setCursor(cursor); + } +} + +void SendFilesBox::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 SendFilesBox::AlbumPreview::mouseReleaseEvent(QMouseEvent *e) { + if (_draggedThumb) { + finishDrag(); + _shrinkAnimation.start([=] { update(); }, 1., 0., kShrinkDuration); + _draggedThumb = nullptr; + _suggestedThumb = nullptr; + update(); + } +} + SendFilesBox::SendFilesBox( QWidget*, Storage::PreparedList &&list, @@ -900,12 +1479,32 @@ void SendFilesBox::setupSendWayControls() { : lng_send_files(lt_count, _list.files.size())); _sendWay->setChangedCallback([this](SendFilesWay value) { if (_albumPreview) { + applyAlbumOrder(); _albumPreview->setSendWay(value); } setInnerFocus(); }); } +void SendFilesBox::applyAlbumOrder() { + Expects(_albumPreview != nullptr); + + const auto order = _albumPreview->takeOrder(); + const auto isDefault = [&] { + for (auto i = 0, count = int(order.size()); i != count; ++i) { + if (order[i] != i) { + return false; + } + } + return true; + }(); + if (isDefault) { + return; + } + + _list = Storage::PreparedList::Reordered(std::move(_list), order); +} + void SendFilesBox::setupCaption() { if (!_caption) { return; @@ -1051,6 +1650,10 @@ void SendFilesBox::send(bool ctrlShiftEnter) { } } } + + if (_albumPreview) { + applyAlbumOrder(); + } _confirmed = true; if (_confirmedCallback) { auto caption = _caption diff --git a/Telegram/SourceFiles/boxes/send_files_box.h b/Telegram/SourceFiles/boxes/send_files_box.h index f16fbb99a..015ea5f7a 100644 --- a/Telegram/SourceFiles/boxes/send_files_box.h +++ b/Telegram/SourceFiles/boxes/send_files_box.h @@ -72,7 +72,6 @@ protected: private: class AlbumPreview; - struct AlbumThumb; void initSendWay(); void initPreview(rpl::producer desiredPreviewHeight); @@ -86,6 +85,7 @@ private: void prepareSingleFilePreview(); void prepareAlbumPreview(); + void applyAlbumOrder(); void send(bool ctrlShiftEnter = false); void captionResized(); diff --git a/Telegram/SourceFiles/info/media/info_media_list_widget.cpp b/Telegram/SourceFiles/info/media/info_media_list_widget.cpp index 328d60fc2..8867a5509 100644 --- a/Telegram/SourceFiles/info/media/info_media_list_widget.cpp +++ b/Telegram/SourceFiles/info/media/info_media_list_widget.cpp @@ -554,7 +554,7 @@ ListWidget::ListWidget( , _migrated(_controller->migrated()) , _type(_controller->section().mediaType()) , _slice(sliceKey(_universalAroundId)) { - setAttribute(Qt::WA_MouseTracking); + setMouseTracking(true); start(); } diff --git a/Telegram/SourceFiles/storage/storage_media_prepare.cpp b/Telegram/SourceFiles/storage/storage_media_prepare.cpp index 36627c4a4..d4df88d57 100644 --- a/Telegram/SourceFiles/storage/storage_media_prepare.cpp +++ b/Telegram/SourceFiles/storage/storage_media_prepare.cpp @@ -120,11 +120,13 @@ void PrepareAlbum(PreparedList &result, int previewWidth) { } if (waiting > 0) { semaphore.acquire(waiting); - const auto badIt = ranges::find( - result.files, - PreparedFile::AlbumType::None, - [](const PreparedFile &file) { return file.type; }); - result.albumIsPossible = (badIt == result.files.end()); + if (result.albumIsPossible) { + const auto badIt = ranges::find( + result.files, + PreparedFile::AlbumType::None, + [](const PreparedFile &file) { return file.type; }); + result.albumIsPossible = (badIt == result.files.end()); + } } } @@ -269,5 +271,21 @@ PreparedList PrepareMediaFromImage( return result; } +PreparedList PreparedList::Reordered( + PreparedList &&list, + std::vector order) { + Expects(list.error == PreparedList::Error::None); + Expects(list.files.size() == order.size()); + + auto result = PreparedList(list.error, list.errorData); + result.albumIsPossible = list.albumIsPossible; + result.allFilesForCompress = list.allFilesForCompress; + result.files.reserve(list.files.size()); + for (auto index : order) { + result.files.push_back(std::move(list.files[index])); + } + return result; +} + } // namespace Storage diff --git a/Telegram/SourceFiles/storage/storage_media_prepare.h b/Telegram/SourceFiles/storage/storage_media_prepare.h index 16daf75a8..79600814f 100644 --- a/Telegram/SourceFiles/storage/storage_media_prepare.h +++ b/Telegram/SourceFiles/storage/storage_media_prepare.h @@ -68,6 +68,9 @@ struct PreparedList { : error(error) , errorData(errorData) { } + static PreparedList Reordered( + PreparedList &&list, + std::vector order); Error error = Error::None; QString errorData; diff --git a/Telegram/SourceFiles/ui/grouped_layout.cpp b/Telegram/SourceFiles/ui/grouped_layout.cpp index 98af5d901..c4a405560 100644 --- a/Telegram/SourceFiles/ui/grouped_layout.cpp +++ b/Telegram/SourceFiles/ui/grouped_layout.cpp @@ -454,9 +454,11 @@ std::vector ComplexLayouter::CropRatios( return ranges::view::all( ratios ) | ranges::view::transform([&](float64 ratio) { + constexpr auto kMaxRatio = 2.75; + constexpr auto kMinRatio = 0.6667; return (averageRatio > 1.1) - ? snap(ratio, 1., 1.7) - : snap(ratio, 0.66667, 1.); + ? snap(ratio, 1., kMaxRatio) + : snap(ratio, kMinRatio, 1.); }) | ranges::to_vector; } diff --git a/Telegram/SourceFiles/ui/images.cpp b/Telegram/SourceFiles/ui/images.cpp index ad17a8afd..57f772d23 100644 --- a/Telegram/SourceFiles/ui/images.cpp +++ b/Telegram/SourceFiles/ui/images.cpp @@ -191,24 +191,16 @@ void prepareCircle(QImage &img) { p.drawPixmap(0, 0, mask); } -void prepareRound(QImage &image, ImageRoundRadius radius, RectParts corners) { - if (!static_cast(corners)) { - return; - } else if (radius == ImageRoundRadius::Ellipse) { - Assert((corners & RectPart::AllCorners) == RectPart::AllCorners); - prepareCircle(image); +void prepareRound( + QImage &image, + QImage *cornerMasks, + RectParts corners, + QRect target) { + if (target.isNull()) { + target = QRect(QPoint(), image.size()); + } else { + Assert(QRect(QPoint(), image.size()).contains(target)); } - Assert(!image.isNull()); - - image.setDevicePixelRatio(cRetinaFactor()); - image = std::move(image).convertToFormat(QImage::Format_ARGB32_Premultiplied); - Assert(!image.isNull()); - - auto masks = App::cornersMask(radius); - prepareRound(image, masks, corners); -} - -void prepareRound(QImage &image, QImage *cornerMasks, RectParts corners) { auto cornerWidth = cornerMasks[0].width(); auto cornerHeight = cornerMasks[0].height(); auto imageWidth = image.width(); @@ -222,10 +214,10 @@ void prepareRound(QImage &image, QImage *cornerMasks, RectParts corners) { Assert(image.bytesPerLine() == (imageIntsPerLine << 2)); auto ints = reinterpret_cast(image.bits()); - auto intsTopLeft = ints; - auto intsTopRight = ints + imageWidth - cornerWidth; - auto intsBottomLeft = ints + (imageHeight - cornerHeight) * imageWidth; - auto intsBottomRight = ints + (imageHeight - cornerHeight + 1) * imageWidth - cornerWidth; + auto intsTopLeft = ints + target.x() + target.y() * imageWidth; + auto intsTopRight = ints + target.x() + target.width() - cornerWidth + target.y() * imageWidth; + auto intsBottomLeft = ints + target.x() + (target.y() + target.height() - cornerHeight) * imageWidth; + auto intsBottomRight = ints + target.x() + target.width() - cornerWidth + (target.y() + target.height() - cornerHeight) * imageWidth; auto maskCorner = [imageWidth, imageHeight, imageIntsPerPixel, imageIntsPerLine](uint32 *imageInts, const QImage &mask) { auto maskWidth = mask.width(); auto maskHeight = mask.height(); @@ -254,6 +246,28 @@ void prepareRound(QImage &image, QImage *cornerMasks, RectParts corners) { if (corners & RectPart::BottomRight) maskCorner(intsBottomRight, cornerMasks[3]); } +void prepareRound( + QImage &image, + ImageRoundRadius radius, + RectParts corners, + QRect target) { + if (!static_cast(corners)) { + return; + } else if (radius == ImageRoundRadius::Ellipse) { + Assert((corners & RectPart::AllCorners) == RectPart::AllCorners); + Assert(target.isNull()); + prepareCircle(image); + } + Assert(!image.isNull()); + + image.setDevicePixelRatio(cRetinaFactor()); + image = std::move(image).convertToFormat(QImage::Format_ARGB32_Premultiplied); + Assert(!image.isNull()); + + auto masks = App::cornersMask(radius); + prepareRound(image, masks, corners, target); +} + QImage prepareColored(style::color add, QImage image) { auto format = image.format(); if (format != QImage::Format_RGB32 && format != QImage::Format_ARGB32_Premultiplied) { diff --git a/Telegram/SourceFiles/ui/images.h b/Telegram/SourceFiles/ui/images.h index 9abc9f7ce..e8d73fc3f 100644 --- a/Telegram/SourceFiles/ui/images.h +++ b/Telegram/SourceFiles/ui/images.h @@ -195,8 +195,7 @@ inline bool operator!=(const WebFileImageLocation &a, const WebFileImageLocation namespace Images { QImage prepareBlur(QImage image); -void prepareRound(QImage &image, ImageRoundRadius radius, RectParts corners = RectPart::AllCorners); -void prepareRound(QImage &image, QImage *cornerMasks, RectParts corners = RectPart::AllCorners); +void prepareRound(QImage &image, ImageRoundRadius radius, RectParts corners = RectPart::AllCorners, QRect target = QRect()); void prepareCircle(QImage &image); QImage prepareColored(style::color add, QImage image); QImage prepareOpaque(QImage image);