From 58d21ff916b4662fbdd4e9a30d60927fe26817ac Mon Sep 17 00:00:00 2001 From: John Preston Date: Fri, 22 Dec 2017 21:42:33 +0400 Subject: [PATCH] Add album support to SendFilesBox. --- Telegram/Resources/langs/lang.strings | 27 +- Telegram/SourceFiles/boxes/boxes.style | 6 + Telegram/SourceFiles/boxes/send_files_box.cpp | 1232 +++++++++++------ Telegram/SourceFiles/boxes/send_files_box.h | 119 +- .../SourceFiles/history/history_widget.cpp | 190 ++- Telegram/SourceFiles/history/history_widget.h | 15 +- .../storage/storage_media_prepare.cpp | 8 +- Telegram/SourceFiles/ui/images.h | 5 + 8 files changed, 959 insertions(+), 643 deletions(-) diff --git a/Telegram/Resources/langs/lang.strings b/Telegram/Resources/langs/lang.strings index 49ba8d5c1..854033fda 100644 --- a/Telegram/Resources/langs/lang.strings +++ b/Telegram/Resources/langs/lang.strings @@ -735,15 +735,15 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org "lng_failed_add_not_mutual" = "Sorry, if a person leaves a group, only a mutual contact can bring them back (they need to have your phone number, and you need theirs)."; "lng_failed_add_not_mutual_channel" = "Sorry, if a person leaves a channel, only a mutual contact can bring them back (they need to have your phone number, and you need theirs)."; -"lng_sure_delete_contact" = "Are you sure, you want to delete {contact} from your contact list?"; -"lng_sure_delete_history" = "Are you sure, you want to delete all message history with {contact}?\n\nThis action cannot be undone."; -"lng_sure_delete_group_history" = "Are you sure, you want to delete all message history in «{group}»?\n\nThis action cannot be undone."; -"lng_sure_delete_and_exit" = "Are you sure, you want to delete all message history and leave «{group}»?\n\nThis action cannot be undone."; -"lng_sure_leave_channel" = "Are you sure, you want to leave\nthis channel?"; -"lng_sure_delete_channel" = "Are you sure, you want to delete this channel? All members will be removed and all messages will be lost."; -"lng_sure_leave_group" = "Are you sure, you want to leave\nthis group?"; -"lng_sure_delete_group" = "Are you sure, you want to delete this group? All members will be removed and all messages will be lost."; -"lng_sure_delete_saved_messages" = "Are you sure, you want to delete all your saved messages?\n\nThis action cannot be undone."; +"lng_sure_delete_contact" = "Are you sure you want to delete {contact} from your contact list?"; +"lng_sure_delete_history" = "Are you sure you want to delete all message history with {contact}?\n\nThis action cannot be undone."; +"lng_sure_delete_group_history" = "Are you sure you want to delete all message history in «{group}»?\n\nThis action cannot be undone."; +"lng_sure_delete_and_exit" = "Are you sure you want to delete all message history and leave «{group}»?\n\nThis action cannot be undone."; +"lng_sure_leave_channel" = "Are you sure you want to leave\nthis channel?"; +"lng_sure_delete_channel" = "Are you sure you want to delete this channel? All members will be removed and all messages will be lost."; +"lng_sure_leave_group" = "Are you sure you want to leave\nthis group?"; +"lng_sure_delete_group" = "Are you sure you want to delete this group? All members will be removed and all messages will be lost."; +"lng_sure_delete_saved_messages" = "Are you sure you want to delete all your saved messages?\n\nThis action cannot be undone."; "lng_message_empty" = "Empty Message"; "lng_message_unsupported" = "This message is not supported by your version of Telegram Desktop. Please update to the last version in Settings or install it from {link}"; @@ -1092,18 +1092,23 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org "lng_context_forward_selected" = "Forward Selected"; "lng_context_delete_selected" = "Delete Selected"; "lng_context_clear_selection" = "Clear Selection"; -"lng_send_images_compress#one" = "Compress image"; -"lng_send_images_compress#other" = "Compress images"; "lng_send_image_empty" = "Could not send an empty file: {name}"; "lng_send_image_too_large" = "Could not send a file, because it is larger than 1500 MB: {name}"; "lng_send_images_selected#one" = "{count} image selected"; "lng_send_images_selected#other" = "{count} images selected"; "lng_send_photos#one" = "Send {count} photo"; "lng_send_photos#other" = "Send {count} photos"; +"lng_send_photos_videos#one" = "Send {count} photo and video file"; +"lng_send_photos_videos#other" = "Send {count} photos and video files"; +"lng_send_separate_photos" = "Send separate photos"; +"lng_send_separate_photos_videos" = "Send separate photos and video files"; "lng_send_files_selected#one" = "{count} file selected"; "lng_send_files_selected#other" = "{count} files selected"; "lng_send_files#one" = "Send {count} file"; "lng_send_files#other" = "Send {count} files"; +"lng_send_album" = "Send an album"; +"lng_send_photo" = "Send a photo"; +"lng_send_file" = "Send a file"; "lng_forward_choose" = "Choose recipient..."; "lng_forward_cant" = "Sorry, no way to forward here :("; diff --git a/Telegram/SourceFiles/boxes/boxes.style b/Telegram/SourceFiles/boxes/boxes.style index 3e8058791..a6b46fada 100644 --- a/Telegram/SourceFiles/boxes/boxes.style +++ b/Telegram/SourceFiles/boxes/boxes.style @@ -718,3 +718,9 @@ groupStickersField: InputField(contactsSearchField) { groupStickersSubTitleHeight: 36px; sendMediaPreviewSize: 308px; +sendMediaPreviewHeightMax: 1280; +sendMediaPreviewPhotoSkip: 10px; +sendMediaFileThumbSize: 64px; +sendMediaFileThumbSkip: 10px; +sendMediaFileNameTop: 7px; +sendMediaFileStatusTop: 37px; diff --git a/Telegram/SourceFiles/boxes/send_files_box.cpp b/Telegram/SourceFiles/boxes/send_files_box.cpp index f9acb945b..21da31c66 100644 --- a/Telegram/SourceFiles/boxes/send_files_box.cpp +++ b/Telegram/SourceFiles/boxes/send_files_box.cpp @@ -29,167 +29,206 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org #include "ui/widgets/checkbox.h" #include "ui/widgets/buttons.h" #include "ui/widgets/input_fields.h" +#include "ui/widgets/scroll_area.h" +#include "ui/wrap/fade_wrap.h" #include "ui/grouped_layout.h" -#include "styles/style_history.h" -#include "styles/style_boxes.h" #include "media/media_clip_reader.h" #include "window/window_controller.h" +#include "styles/style_history.h" +#include "styles/style_boxes.h" namespace { constexpr auto kMinPreviewWidth = 20; -} // namespace +class SingleMediaPreview : public Ui::RpWidget { +public: + static SingleMediaPreview *Create( + QWidget *parent, + not_null controller, + const Storage::PreparedFile &file); -SendFilesBox::SendFilesBox( - QWidget*, - Storage::PreparedList &&list, - CompressConfirm compressed) -: _list(std::move(list)) -, _compressConfirm(compressed) -, _caption( - this, - st::confirmCaptionArea, - langFactory(_list.files.size() > 1 - ? lng_photos_comment - : lng_photo_caption)) { - if (_list.files.size() == 1) { - prepareSingleFileLayout(); - } else if (_list.albumIsPossible) { + SingleMediaPreview( + QWidget *parent, + not_null controller, + QImage preview, + bool animated, + const QString &animatedPreviewPath); + bool canSendAsPhoto() const { + return _canSendAsPhoto; } -} -void SendFilesBox::prepareSingleFileLayout() { - Expects(_list.files.size() == 1); + rpl::producer desiredHeightValue() const override; - const auto &file = _list.files[0]; +protected: + void paintEvent(QPaintEvent *e) override; + +private: + void preparePreview( + QImage preview, + const QString &animatedPreviewPath); + void prepareAnimatedPreview(const QString &animatedPreviewPath); + void clipCallback(Media::Clip::Notification notification); + + not_null _controller; + bool _animated = false; + bool _canSendAsPhoto = false; + QPixmap _preview; + int _previewLeft = 0; + int _previewWidth = 0; + int _previewHeight = 0; + Media::Clip::ReaderPointer _gifPreview; + +}; + +class SingleFilePreview : public Ui::RpWidget { +public: + SingleFilePreview( + QWidget *parent, + const Storage::PreparedFile &file); + + rpl::producer desiredHeightValue() const override; + +protected: + void paintEvent(QPaintEvent *e) override; + +private: + void preparePreview(const Storage::PreparedFile &file); + void prepareThumb(const QImage &preview); + + QPixmap _fileThumb; + Text _nameText; + bool _fileIsAudio = false; + bool _fileIsImage = false; + QString _statusText; + int _statusWidth = 0; + +}; + +SingleMediaPreview *SingleMediaPreview::Create( + QWidget *parent, + not_null controller, + const Storage::PreparedFile &file) { auto preview = QImage(); + bool animated = false; + bool animationPreview = false; if (const auto image = base::get_if( &file.information->media)) { preview = image->data; - _animated = image->animated; + animated = animationPreview = image->animated; } else if (const auto video = base::get_if( &file.information->media)) { preview = video->thumbnail; - _animated = true; + animated = true; + animationPreview = video->isGifv; } - - if (!Storage::ValidateThumbDimensions(preview.width(), preview.height()) - || _animated) { - _compressConfirm = CompressConfirm::None; - } - - if (!preview.isNull()) { - if (!_animated && _compressConfirm == CompressConfirm::None) { - auto originalWidth = preview.width(); - auto originalHeight = preview.height(); - auto thumbWidth = st::msgFileThumbSize; - if (originalWidth > originalHeight) { - thumbWidth = (originalWidth * st::msgFileThumbSize) - / originalHeight; - } - auto options = Images::Option::Smooth - | Images::Option::RoundedSmall - | Images::Option::RoundedTopLeft - | Images::Option::RoundedTopRight - | Images::Option::RoundedBottomLeft - | Images::Option::RoundedBottomRight; - _fileThumb = Images::pixmap( - preview, - thumbWidth * cIntRetinaFactor(), - 0, - options, - st::msgFileThumbSize, - st::msgFileThumbSize); - } else { - auto maxW = 0; - auto maxH = 0; - if (_animated) { - auto limitW = st::sendMediaPreviewSize; - auto limitH = st::confirmMaxHeight; - maxW = qMax(preview.width(), 1); - maxH = qMax(preview.height(), 1); - if (maxW * limitH > maxH * limitW) { - if (maxW < limitW) { - maxH = maxH * limitW / maxW; - maxW = limitW; - } - } else { - if (maxH < limitH) { - maxW = maxW * limitH / maxH; - maxH = limitH; - } - } - preview = Images::prepare( - preview, - maxW * cIntRetinaFactor(), - maxH * cIntRetinaFactor(), - Images::Option::Smooth | Images::Option::Blurred, - maxW, - maxH); - } - auto originalWidth = preview.width(); - auto originalHeight = preview.height(); - if (!originalWidth || !originalHeight) { - originalWidth = originalHeight = 1; - } - _previewWidth = st::sendMediaPreviewSize; - if (preview.width() < _previewWidth) { - _previewWidth = qMax(preview.width(), kMinPreviewWidth); - } - auto maxthumbh = qMin(qRound(1.5 * _previewWidth), st::confirmMaxHeight); - _previewHeight = qRound(originalHeight * float64(_previewWidth) / originalWidth); - if (_previewHeight > maxthumbh) { - _previewWidth = qRound(_previewWidth * float64(maxthumbh) / _previewHeight); - accumulate_max(_previewWidth, kMinPreviewWidth); - _previewHeight = maxthumbh; - } - _previewLeft = (st::boxWideWidth - _previewWidth) / 2; - - preview = std::move(preview).scaled( - _previewWidth * cIntRetinaFactor(), - _previewHeight * cIntRetinaFactor(), - Qt::IgnoreAspectRatio, - Qt::SmoothTransformation); - preview = Images::prepareOpaque(std::move(preview)); - _preview = App::pixmapFromImageInPlace(std::move(preview)); - _preview.setDevicePixelRatio(cRetinaFactor()); - - prepareGifPreview(); - } - } - if (_preview.isNull()) { - prepareDocumentLayout(); + if (preview.isNull()) { + return nullptr; + } else if (!animated && !Storage::ValidateThumbDimensions( + preview.width(), + preview.height())) { + return nullptr; } + return Ui::CreateChild( + parent, + controller, + preview, + animated, + animationPreview ? file.path : QString()); } -void SendFilesBox::prepareGifPreview() { - using namespace Media::Clip; - auto createGifPreview = [this] { - const auto &information = _list.files.front().information; - if (!information) { - return false; +SingleMediaPreview::SingleMediaPreview( + QWidget *parent, + not_null controller, + QImage preview, + bool animated, + const QString &animatedPreviewPath) +: RpWidget(parent) +, _controller(controller) +, _animated(animated) { + Expects(!preview.isNull()); + + _canSendAsPhoto = !_animated && Storage::ValidateThumbDimensions( + preview.width(), + preview.height()); + + preparePreview(preview, animatedPreviewPath); +} + +void SingleMediaPreview::preparePreview( + QImage preview, + const QString &animatedPreviewPath) { + auto maxW = 0; + auto maxH = 0; + if (_animated) { + auto limitW = st::sendMediaPreviewSize; + auto limitH = st::confirmMaxHeight; + maxW = qMax(preview.width(), 1); + maxH = qMax(preview.height(), 1); + if (maxW * limitH > maxH * limitW) { + if (maxW < limitW) { + maxH = maxH * limitW / maxW; + maxW = limitW; + } + } else { + if (maxH < limitH) { + maxW = maxW * limitH / maxH; + maxH = limitH; + } } - if (const auto video = base::get_if( - &information->media)) { - return video->isGifv; - } - // Plain old .gif animation. - return _animated; - }; - if (createGifPreview()) { - const auto callback = [this](Notification notification) { + preview = Images::prepare( + preview, + maxW * cIntRetinaFactor(), + maxH * cIntRetinaFactor(), + Images::Option::Smooth | Images::Option::Blurred, + maxW, + maxH); + } + auto originalWidth = preview.width(); + auto originalHeight = preview.height(); + if (!originalWidth || !originalHeight) { + originalWidth = originalHeight = 1; + } + _previewWidth = st::sendMediaPreviewSize; + if (preview.width() < _previewWidth) { + _previewWidth = qMax(preview.width(), kMinPreviewWidth); + } + auto maxthumbh = qMin(qRound(1.5 * _previewWidth), st::confirmMaxHeight); + _previewHeight = qRound(originalHeight * float64(_previewWidth) / originalWidth); + if (_previewHeight > maxthumbh) { + _previewWidth = qRound(_previewWidth * float64(maxthumbh) / _previewHeight); + accumulate_max(_previewWidth, kMinPreviewWidth); + _previewHeight = maxthumbh; + } + _previewLeft = (st::boxWideWidth - _previewWidth) / 2; + + preview = std::move(preview).scaled( + _previewWidth * cIntRetinaFactor(), + _previewHeight * cIntRetinaFactor(), + Qt::IgnoreAspectRatio, + Qt::SmoothTransformation); + preview = Images::prepareOpaque(std::move(preview)); + _preview = App::pixmapFromImageInPlace(std::move(preview)); + _preview.setDevicePixelRatio(cRetinaFactor()); + + prepareAnimatedPreview(animatedPreviewPath); +} + +void SingleMediaPreview::prepareAnimatedPreview( + const QString &animatedPreviewPath) { + if (!animatedPreviewPath.isEmpty()) { + auto callback = [this](Media::Clip::Notification notification) { clipCallback(notification); }; _gifPreview = Media::Clip::MakeReader( - _list.files.front().path, - callback); + animatedPreviewPath, + std::move(callback)); if (_gifPreview) _gifPreview->setAutoplay(); } } -void SendFilesBox::clipCallback(Media::Clip::Notification notification) { +void SingleMediaPreview::clipCallback(Media::Clip::Notification notification) { using namespace Media::Clip; switch (notification) { case NotificationReinit: { @@ -213,16 +252,91 @@ void SendFilesBox::clipCallback(Media::Clip::Notification notification) { } } -void SendFilesBox::prepareDocumentLayout() { - const auto &file = _list.files.front(); +void SingleMediaPreview::paintEvent(QPaintEvent *e) { + Painter p(this); + + if (_previewLeft > st::boxPhotoPadding.left()) { + p.fillRect(st::boxPhotoPadding.left(), st::boxPhotoPadding.top(), _previewLeft - st::boxPhotoPadding.left(), _previewHeight, st::confirmBg); + } + if (_previewLeft + _previewWidth < width() - st::boxPhotoPadding.right()) { + p.fillRect(_previewLeft + _previewWidth, st::boxPhotoPadding.top(), width() - st::boxPhotoPadding.right() - _previewLeft - _previewWidth, _previewHeight, st::confirmBg); + } + if (_gifPreview && _gifPreview->started()) { + auto s = QSize(_previewWidth, _previewHeight); + auto paused = _controller->isGifPausedAtLeastFor(Window::GifPauseReason::Layer); + auto frame = _gifPreview->current(s.width(), s.height(), s.width(), s.height(), ImageRoundRadius::None, RectPart::None, paused ? 0 : getms()); + p.drawPixmap(_previewLeft, st::boxPhotoPadding.top(), frame); + } else { + p.drawPixmap(_previewLeft, st::boxPhotoPadding.top(), _preview); + } + if (_animated && !_gifPreview) { + auto inner = QRect(_previewLeft + (_previewWidth - st::msgFileSize) / 2, st::boxPhotoPadding.top() + (_previewHeight - st::msgFileSize) / 2, st::msgFileSize, st::msgFileSize); + p.setPen(Qt::NoPen); + p.setBrush(st::msgDateImgBg); + + { + PainterHighQualityEnabler hq(p); + p.drawEllipse(inner); + } + + auto icon = &st::historyFileInPlay; + icon->paintInCenter(p, inner); + } +} + +rpl::producer SingleMediaPreview::desiredHeightValue() const { + return rpl::single(st::boxPhotoPadding.top() + _previewHeight); +} + +SingleFilePreview::SingleFilePreview( + QWidget *parent, + const Storage::PreparedFile &file) +: RpWidget(parent) { + preparePreview(file); +} + +void SingleFilePreview::prepareThumb(const QImage &preview) { + if (preview.isNull()) { + return; + } + + auto originalWidth = preview.width(); + auto originalHeight = preview.height(); + auto thumbWidth = st::msgFileThumbSize; + if (originalWidth > originalHeight) { + thumbWidth = (originalWidth * st::msgFileThumbSize) + / originalHeight; + } + auto options = Images::Option::Smooth + | Images::Option::RoundedSmall + | Images::Option::RoundedTopLeft + | Images::Option::RoundedTopRight + | Images::Option::RoundedBottomLeft + | Images::Option::RoundedBottomRight; + _fileThumb = Images::pixmap( + preview, + thumbWidth * cIntRetinaFactor(), + 0, + options, + st::msgFileThumbSize, + st::msgFileThumbSize); +} + +void SingleFilePreview::preparePreview(const Storage::PreparedFile &file) { + auto preview = QImage(); + if (const auto image = base::get_if( + &file.information->media)) { + preview = image->data; + } else if (const auto video = base::get_if( + &file.information->media)) { + preview = video->thumbnail; + } + prepareThumb(preview); const auto filepath = file.path; if (filepath.isEmpty()) { - const auto data = base::get_if( - &file.information->media); - const auto image = data ? data->data : QImage(); auto filename = filedialogDefaultName(qsl("image"), qsl(".png"), QString(), true); _nameText.setText(st::semiboldTextStyle, filename, _textNameOptions); - _statusText = qsl("%1x%2").arg(image.width()).arg(image.height()); + _statusText = qsl("%1x%2").arg(preview.width()).arg(preview.height()); _statusWidth = qMax(_nameText.maxWidth(), st::normalFont->width(_statusText)); _fileIsImage = true; } else { @@ -256,292 +370,136 @@ void SendFilesBox::prepareDocumentLayout() { } } -void SendFilesBox::prepare() { - Expects(controller() != nullptr); - - if (_list.files.size() > 1) { - updateTitleText(); - } - - _send = addButton(langFactory(lng_send_button), [this] { send(); }); - addButton(langFactory(lng_cancel), [this] { closeBox(); }); - - if (_compressConfirm != CompressConfirm::None) { - auto compressed = (_compressConfirm == CompressConfirm::Auto) ? cCompressPastedImage() : (_compressConfirm == CompressConfirm::Yes); - auto text = lng_send_images_compress(lt_count, _list.files.size()); - _compressed.create(this, text, compressed, st::defaultBoxCheckbox); - subscribe(_compressed->checkedChanged, [this](bool checked) { - compressedChange(); - }); - } - if (_caption) { - _caption->setMaxLength(MaxPhotoCaption); - _caption->setCtrlEnterSubmit(Ui::CtrlEnterSubmit::Both); - connect(_caption, &Ui::InputArea::resized, this, [this] { - captionResized(); - }); - connect(_caption, &Ui::InputArea::submitted, this, [this]( - bool ctrlShiftEnter) { - send(ctrlShiftEnter); - }); - connect(_caption, &Ui::InputArea::cancelled, this, [this] { - closeBox(); - }); - } - subscribe(boxClosing, [this] { - if (!_confirmed && _cancelledCallback) { - _cancelledCallback(); - } - }); - _send->setText(getSendButtonText()); - updateButtonsGeometry(); - updateBoxSize(); -} - -base::lambda SendFilesBox::getSendButtonText() const { - if (_compressed && _compressed->checked()) { - return [count = _list.files.size()] { - return lng_send_photos(lt_count, count); - }; - } - return [count = _list.files.size()] { - return lng_send_files(lt_count, count); - }; -} - -void SendFilesBox::compressedChange() { - setInnerFocus(); - _send->setText(getSendButtonText()); - updateButtonsGeometry(); - updateControlsGeometry(); -} - -void SendFilesBox::captionResized() { - updateBoxSize(); - updateControlsGeometry(); - update(); -} - -void SendFilesBox::updateTitleText() { - _titleText = (_compressConfirm == CompressConfirm::None) - ? lng_send_files_selected(lt_count, _list.files.size()) - : lng_send_images_selected(lt_count, _list.files.size()); - update(); -} - -void SendFilesBox::updateBoxSize() { - auto newHeight = _titleText.isEmpty() ? 0 : st::boxTitleHeight; - if (!_preview.isNull()) { - newHeight += st::boxPhotoPadding.top() + _previewHeight; - } else if (!_fileThumb.isNull()) { - newHeight += st::boxPhotoPadding.top() + st::msgFileThumbPadding.top() + st::msgFileThumbSize + st::msgFileThumbPadding.bottom(); - } else if (_list.files.size() > 1) { - newHeight += 0; - } else { - newHeight += st::boxPhotoPadding.top() + st::msgFilePadding.top() + st::msgFileSize + st::msgFilePadding.bottom(); - } - if (_compressed) { - newHeight += st::boxPhotoCompressedSkip + _compressed->heightNoMargins(); - } - if (_caption) { - newHeight += st::boxPhotoCaptionSkip + _caption->height(); - } - setDimensions(st::boxWideWidth, newHeight); -} - -void SendFilesBox::keyPressEvent(QKeyEvent *e) { - if (e->key() == Qt::Key_Enter || e->key() == Qt::Key_Return) { - const auto modifiers = e->modifiers(); - const auto ctrl = modifiers.testFlag(Qt::ControlModifier) - || modifiers.testFlag(Qt::MetaModifier); - const auto shift = modifiers.testFlag(Qt::ShiftModifier); - send(ctrl && shift); - } else { - BoxContent::keyPressEvent(e); - } -} - -void SendFilesBox::paintEvent(QPaintEvent *e) { - BoxContent::paintEvent(e); - +void SingleFilePreview::paintEvent(QPaintEvent *e) { Painter p(this); - if (!_titleText.isEmpty()) { - p.setFont(st::boxPhotoTitleFont); - p.setPen(st::boxTitleFg); - p.drawTextLeft(st::boxPhotoTitlePosition.x(), st::boxPhotoTitlePosition.y(), width(), _titleText); - } - - if (!_preview.isNull()) { - if (_previewLeft > st::boxPhotoPadding.left()) { - p.fillRect(st::boxPhotoPadding.left(), st::boxPhotoPadding.top(), _previewLeft - st::boxPhotoPadding.left(), _previewHeight, st::confirmBg); - } - if (_previewLeft + _previewWidth < width() - st::boxPhotoPadding.right()) { - p.fillRect(_previewLeft + _previewWidth, st::boxPhotoPadding.top(), width() - st::boxPhotoPadding.right() - _previewLeft - _previewWidth, _previewHeight, st::confirmBg); - } - if (_gifPreview && _gifPreview->started()) { - auto s = QSize(_previewWidth, _previewHeight); - auto paused = controller()->isGifPausedAtLeastFor(Window::GifPauseReason::Layer); - auto frame = _gifPreview->current(s.width(), s.height(), s.width(), s.height(), ImageRoundRadius::None, RectPart::None, paused ? 0 : getms()); - p.drawPixmap(_previewLeft, st::boxPhotoPadding.top(), frame); - } else { - p.drawPixmap(_previewLeft, st::boxPhotoPadding.top(), _preview); - } - if (_animated && !_gifPreview) { - auto inner = QRect(_previewLeft + (_previewWidth - st::msgFileSize) / 2, st::boxPhotoPadding.top() + (_previewHeight - st::msgFileSize) / 2, st::msgFileSize, st::msgFileSize); - p.setPen(Qt::NoPen); - p.setBrush(st::msgDateImgBg); - - { - PainterHighQualityEnabler hq(p); - p.drawEllipse(inner); - } - - auto icon = &st::historyFileInPlay; - icon->paintInCenter(p, inner); - } - } else if (_list.files.size() < 2) { - auto w = width() - st::boxPhotoPadding.left() - st::boxPhotoPadding.right(); - auto h = _fileThumb.isNull() ? (st::msgFilePadding.top() + st::msgFileSize + st::msgFilePadding.bottom()) : (st::msgFileThumbPadding.top() + st::msgFileThumbSize + st::msgFileThumbPadding.bottom()); - auto nameleft = 0, nametop = 0, nameright = 0, statustop = 0, linktop = 0; - if (_fileThumb.isNull()) { - nameleft = st::msgFilePadding.left() + st::msgFileSize + st::msgFilePadding.right(); - nametop = st::msgFileNameTop; - nameright = st::msgFilePadding.left(); - statustop = st::msgFileStatusTop; - } else { - nameleft = st::msgFileThumbPadding.left() + st::msgFileThumbSize + st::msgFileThumbPadding.right(); - nametop = st::msgFileThumbNameTop; - nameright = st::msgFileThumbPadding.left(); - statustop = st::msgFileThumbStatusTop; - linktop = st::msgFileThumbLinkTop; - } - auto namewidth = w - nameleft - (_fileThumb.isNull() ? st::msgFilePadding.left() : st::msgFileThumbPadding.left()); - int32 x = (width() - w) / 2, y = st::boxPhotoPadding.top(); - - App::roundRect(p, x, y, w, h, st::msgOutBg, MessageOutCorners, &st::msgOutShadow); - - if (_fileThumb.isNull()) { - QRect inner(rtlrect(x + st::msgFilePadding.left(), y + st::msgFilePadding.top(), st::msgFileSize, st::msgFileSize, width())); - p.setPen(Qt::NoPen); - p.setBrush(st::msgFileOutBg); - - { - PainterHighQualityEnabler hq(p); - p.drawEllipse(inner); - } - - auto &icon = _fileIsAudio ? st::historyFileOutPlay : _fileIsImage ? st::historyFileOutImage : st::historyFileOutDocument; - icon.paintInCenter(p, inner); - } else { - QRect rthumb(rtlrect(x + st::msgFileThumbPadding.left(), y + st::msgFileThumbPadding.top(), st::msgFileThumbSize, st::msgFileThumbSize, width())); - p.drawPixmap(rthumb.topLeft(), _fileThumb); - } - p.setFont(st::semiboldFont); - p.setPen(st::historyFileNameOutFg); - _nameText.drawLeftElided(p, x + nameleft, y + nametop, namewidth, width()); - - auto &status = st::mediaOutFg; - p.setFont(st::normalFont); - p.setPen(status); - p.drawTextLeft(x + nameleft, y + statustop, width(), _statusText); - } -} - -void SendFilesBox::resizeEvent(QResizeEvent *e) { - BoxContent::resizeEvent(e); - updateControlsGeometry(); -} - -void SendFilesBox::updateControlsGeometry() { - auto bottom = height(); - if (_caption) { - _caption->resize(st::sendMediaPreviewSize, _caption->height()); - _caption->moveToLeft(st::boxPhotoPadding.left(), bottom - _caption->height()); - bottom -= st::boxPhotoCaptionSkip + _caption->height(); - } - if (_compressed) { - _compressed->moveToLeft(st::boxPhotoPadding.left(), bottom - _compressed->heightNoMargins()); - bottom -= st::boxPhotoCompressedSkip + _compressed->heightNoMargins(); - } -} - -void SendFilesBox::setInnerFocus() { - if (!_caption || _caption->isHidden()) { - setFocus(); + auto w = width() - st::boxPhotoPadding.left() - st::boxPhotoPadding.right(); + auto h = _fileThumb.isNull() ? (st::msgFilePadding.top() + st::msgFileSize + st::msgFilePadding.bottom()) : (st::msgFileThumbPadding.top() + st::msgFileThumbSize + st::msgFileThumbPadding.bottom()); + auto nameleft = 0, nametop = 0, nameright = 0, statustop = 0, linktop = 0; + if (_fileThumb.isNull()) { + nameleft = st::msgFilePadding.left() + st::msgFileSize + st::msgFilePadding.right(); + nametop = st::msgFileNameTop; + nameright = st::msgFilePadding.left(); + statustop = st::msgFileStatusTop; } else { - _caption->setFocusFast(); + nameleft = st::msgFileThumbPadding.left() + st::msgFileThumbSize + st::msgFileThumbPadding.right(); + nametop = st::msgFileThumbNameTop; + nameright = st::msgFileThumbPadding.left(); + statustop = st::msgFileThumbStatusTop; + linktop = st::msgFileThumbLinkTop; } + auto namewidth = w - nameleft - (_fileThumb.isNull() ? st::msgFilePadding.left() : st::msgFileThumbPadding.left()); + int32 x = (width() - w) / 2, y = st::boxPhotoPadding.top(); + + App::roundRect(p, x, y, w, h, st::msgOutBg, MessageOutCorners, &st::msgOutShadow); + + if (_fileThumb.isNull()) { + QRect inner(rtlrect(x + st::msgFilePadding.left(), y + st::msgFilePadding.top(), st::msgFileSize, st::msgFileSize, width())); + p.setPen(Qt::NoPen); + p.setBrush(st::msgFileOutBg); + + { + PainterHighQualityEnabler hq(p); + p.drawEllipse(inner); + } + + auto &icon = _fileIsAudio + ? st::historyFileOutPlay + : _fileIsImage + ? st::historyFileOutImage + : st::historyFileOutDocument; + icon.paintInCenter(p, inner); + } else { + QRect rthumb(rtlrect(x + st::msgFileThumbPadding.left(), y + st::msgFileThumbPadding.top(), st::msgFileThumbSize, st::msgFileThumbSize, width())); + p.drawPixmap(rthumb.topLeft(), _fileThumb); + } + p.setFont(st::semiboldFont); + p.setPen(st::historyFileNameOutFg); + _nameText.drawLeftElided(p, x + nameleft, y + nametop, namewidth, width()); + + auto &status = st::mediaOutFg; + p.setFont(st::normalFont); + p.setPen(status); + p.drawTextLeft(x + nameleft, y + statustop, width(), _statusText); } -void SendFilesBox::send(bool ctrlShiftEnter) { - if (_compressed && _compressConfirm == CompressConfirm::Auto && _compressed->checked() != cCompressPastedImage()) { - cSetCompressPastedImage(_compressed->checked()); - Local::writeUserSettings(); - } - _confirmed = true; - if (_confirmedCallback) { - auto compressed = _compressed ? _compressed->checked() : false; - auto caption = _caption ? TextUtilities::PrepareForSending(_caption->getLastText(), TextUtilities::PrepareTextOption::CheckLinks) : QString(); - _confirmedCallback( - std::move(_list), - compressed, - caption, - ctrlShiftEnter); - } - closeBox(); +rpl::producer SingleFilePreview::desiredHeightValue() const { + auto h = _fileThumb.isNull() + ? (st::msgFilePadding.top() + st::msgFileSize + st::msgFilePadding.bottom()) + : (st::msgFileThumbPadding.top() + st::msgFileThumbSize + st::msgFileThumbPadding.bottom()); + return rpl::single(st::boxPhotoPadding.top() + h + st::msgShadow); } -SendFilesBox::~SendFilesBox() = default; +base::lambda FieldPlaceholder(const Storage::PreparedList &list) { + return langFactory(list.files.size() > 1 + ? lng_photos_comment + : lng_photo_caption); +} -struct SendAlbumBox::Thumb { +} // namespace + +struct SendFilesBox::AlbumThumb { Ui::GroupMediaLayout layout; - QPixmap image; + QPixmap albumImage; + QPixmap photo; + QPixmap fileThumb; + QString name; + QString status; + int nameWidth = 0; + int statusWidth = 0; + bool video = false; }; -SendAlbumBox::SendAlbumBox(QWidget*, Storage::PreparedList &&list) -: _list(std::move(list)) -, _caption( - this, - st::confirmCaptionArea, - langFactory(_list.files.size() > 1 - ? lng_photos_comment - : lng_photo_caption)) { -} +class SendFilesBox::AlbumPreview : public Ui::RpWidget { +public: + AlbumPreview( + QWidget *parent, + const Storage::PreparedList &list, + SendWay way); -void SendAlbumBox::prepare() { - Expects(controller() != nullptr); + void setSendWay(SendWay way); +protected: + void paintEvent(QPaintEvent *e) override; + +private: + void prepareThumbs(); + 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; + + const Storage::PreparedList &_list; + SendWay _sendWay = SendWay::Files; + std::vector _thumbs; + int _thumbsHeight = 0; + int _photosHeight = 0; + int _filesHeight = 0; + +}; + +SendFilesBox::AlbumPreview::AlbumPreview( + QWidget *parent, + const Storage::PreparedList &list, + SendWay way) +: RpWidget(parent) +, _list(list) +, _sendWay(way) { prepareThumbs(); - - addButton(langFactory(lng_send_button), [this] { send(); }); - addButton(langFactory(lng_cancel), [this] { closeBox(); }); - - if (_caption) { - _caption->setMaxLength(MaxPhotoCaption); - _caption->setCtrlEnterSubmit(Ui::CtrlEnterSubmit::Both); - connect(_caption, &Ui::InputArea::resized, this, [this] { - captionResized(); - }); - connect(_caption, &Ui::InputArea::submitted, this, [this]( - bool ctrlShiftEnter) { - send(ctrlShiftEnter); - }); - connect(_caption, &Ui::InputArea::cancelled, this, [this] { - closeBox(); - }); - } - subscribe(boxClosing, [this] { - if (!_confirmed && _cancelledCallback) { - _cancelledCallback(); - } - }); - - updateButtonsGeometry(); - updateBoxSize(); + updateSize(); } -void SendAlbumBox::prepareThumbs() { +void SendFilesBox::AlbumPreview::setSendWay(SendWay way) { + _sendWay = way; + updateSize(); + update(); +} + +void SendFilesBox::AlbumPreview::prepareThumbs() { auto sizes = ranges::view::all( _list.files ) | ranges::view::transform([](const Storage::PreparedFile &file) { @@ -558,17 +516,29 @@ void SendAlbumBox::prepareThumbs() { _thumbs.reserve(count); for (auto i = 0; i != count; ++i) { - _thumbs.push_back(prepareThumb(_list.files[i].preview, layout[i])); + _thumbs.push_back(prepareThumb(_list.files[i], layout[i])); const auto &geometry = layout[i].geometry; accumulate_max(_thumbsHeight, geometry.y() + geometry.height()); } + _photosHeight = ranges::accumulate(ranges::view::all( + _thumbs + ) | ranges::view::transform([](const AlbumThumb &thumb) { + return thumb.photo.height() / cIntRetinaFactor(); + }), 0) + (count - 1) * st::sendMediaPreviewPhotoSkip; + + _filesHeight = count * st::sendMediaFileThumbSize + + (count - 1) * st::sendMediaFileThumbSkip; } -SendAlbumBox::Thumb SendAlbumBox::prepareThumb( - const QImage &preview, +SendFilesBox::AlbumThumb SendFilesBox::AlbumPreview::prepareThumb( + const Storage::PreparedFile &file, const Ui::GroupMediaLayout &layout) const { - auto result = Thumb(); + Expects(!file.preview.isNull()); + + const auto &preview = file.preview; + auto result = AlbumThumb(); result.layout = layout; + result.video = (file.type == Storage::PreparedFile::AlbumType::Video); const auto width = layout.geometry.width(); const auto height = layout.geometry.height(); @@ -586,31 +556,402 @@ SendAlbumBox::Thumb SendAlbumBox::prepareThumb( const auto pixWidth = pixSize.width() * cIntRetinaFactor(); const auto pixHeight = pixSize.height() * cIntRetinaFactor(); - result.image = App::pixmapFromImageInPlace(Images::prepare( + 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())); + + 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()); + } + 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); + } + result.statusWidth = st::normalFont->width(result.status); + return result; } -void SendAlbumBox::captionResized() { +void SendFilesBox::AlbumPreview::updateSize() { + const auto height = [&] { + switch (_sendWay) { + case SendWay::Album: return _thumbsHeight; + case SendWay::Photos: return _photosHeight; + case SendWay::Files: return _filesHeight; + } + Unexpected("Send way in SendFilesBox::AlbumPreview::updateSize"); + }(); + resize(st::boxWideWidth, height); +} + +void SendFilesBox::AlbumPreview::paintEvent(QPaintEvent *e) { + Painter p(this); + + switch (_sendWay) { + case SendWay::Album: paintAlbum(p); break; + case SendWay::Photos: paintPhotos(p, e->rect()); break; + case SendWay::Files: paintFiles(p, e->rect()); break; + } +} + +void SendFilesBox::AlbumPreview::paintAlbum(Painter &p) const { + const auto left = (st::boxWideWidth - st::sendMediaPreviewSize) / 2; + const auto top = 0; + 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); + } + } +} + +void SendFilesBox::AlbumPreview::paintPhotos(Painter &p, QRect clip) const { + const auto left = (st::boxWideWidth - st::sendMediaPreviewSize) / 2; + auto top = 0; + for (const auto &thumb : _thumbs) { + const auto bottom = top + thumb.photo.height() / cIntRetinaFactor(); + const auto guard = gsl::finally([&] { + top = bottom + st::sendMediaPreviewPhotoSkip; + }); + if (top >= clip.y() + clip.height()) { + break; + } else if (bottom <= clip.y()) { + continue; + } + p.drawPixmap( + left, + top, + thumb.photo); + } +} + +void SendFilesBox::AlbumPreview::paintFiles(Painter &p, QRect clip) const { + const auto fileHeight = st::sendMediaFileThumbSize + + st::sendMediaFileThumbSkip; + const auto bottom = clip.y() + clip.height(); + 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; + + 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); + top += fileHeight; + } +} + +SendFilesBox::SendFilesBox( + QWidget*, + Storage::PreparedList &&list, + CompressConfirm compressed) +: _list(std::move(list)) +, _compressConfirm(compressed) +, _caption(this, st::confirmCaptionArea, FieldPlaceholder(_list)) { +} + +void SendFilesBox::initPreview(rpl::producer desiredPreviewHeight) { + setupControls(); + + updateBoxSize(); + + using namespace rpl::mappers; + rpl::combine( + std::move(desiredPreviewHeight), + _footerHeight.value(), + _titleHeight + _1 + _2 + ) | rpl::start_with_next([this](int height) { + setDimensions( + st::boxWideWidth, + std::min(st::sendMediaPreviewHeightMax, height)); + }, lifetime()); + + if (_preview) { + _preview->show(); + } +} + +void SendFilesBox::prepareSingleFilePreview() { + Expects(_list.files.size() == 1); + + const auto &file = _list.files[0]; + const auto media = SingleMediaPreview::Create(this, controller(), file); + if (media) { + if (!media->canSendAsPhoto()) { + _compressConfirm = CompressConfirm::None; + } + _preview = media; + initPreview(media->desiredHeightValue()); + } else { + const auto preview = Ui::CreateChild(this, file); + _compressConfirm = CompressConfirm::None; + _preview = preview; + initPreview(preview->desiredHeightValue()); + } +} + +void SendFilesBox::prepareAlbumPreview() { + Expects(_sendWay != nullptr); + + const auto wrap = Ui::CreateChild( + this, + st::boxLayerScroll); + _albumPreview = wrap->setOwnedWidget(object_ptr( + this, + _list, + _sendWay->value())); + _preview = wrap; + _albumPreview->show(); + setupShadows(wrap, _albumPreview); + + initPreview(_albumPreview->desiredHeightValue()); +} + +void SendFilesBox::setupShadows( + not_null wrap, + not_null content) { + using namespace rpl::mappers; + + const auto topShadow = Ui::CreateChild(this); + const auto bottomShadow = Ui::CreateChild(this); + wrap->geometryValue( + ) | rpl::start_with_next([=](const QRect &geometry) { + topShadow->resizeToWidth(geometry.width()); + topShadow->move( + geometry.x(), + geometry.y()); + bottomShadow->resizeToWidth(geometry.width()); + bottomShadow->move( + geometry.x(), + geometry.y() + geometry.height() - st::lineWidth); + }, topShadow->lifetime()); + topShadow->toggleOn(wrap->scrollTopValue() | rpl::map(_1 > 0)); + bottomShadow->toggleOn(rpl::combine( + wrap->scrollTopValue(), + wrap->heightValue(), + content->heightValue(), + _1 + _2 < _3)); +} + +void SendFilesBox::prepare() { + Expects(controller() != nullptr); + + _send = addButton(langFactory(lng_send_button), [this] { send(); }); + addButton(langFactory(lng_cancel), [this] { closeBox(); }); + initSendWay(); + if (_list.files.size() == 1) { + prepareSingleFilePreview(); + } else { + if (_list.albumIsPossible) { + prepareAlbumPreview(); + } else { + auto desiredPreviewHeight = rpl::single(0); + initPreview(std::move(desiredPreviewHeight)); + } + } + + subscribe(boxClosing, [this] { + if (!_confirmed && _cancelledCallback) { + _cancelledCallback(); + } + }); +} + +void SendFilesBox::initSendWay() { + const auto value = (_compressConfirm == CompressConfirm::None) + ? SendWay::Files + : (_compressConfirm == CompressConfirm::Auto) + ? (cCompressPastedImage() + ? (_list.albumIsPossible ? SendWay::Album : SendWay::Photos) + : SendWay::Files) + : (_compressConfirm == CompressConfirm::Yes + ? (_list.albumIsPossible ? SendWay::Album : SendWay::Photos) + : SendWay::Files); + _sendWay = std::make_shared>(value); +} + +void SendFilesBox::setupControls() { + _albumVideosCount = ranges::v3::count( + _list.files, + Storage::PreparedFile::AlbumType::Video, + [](const Storage::PreparedFile &file) { return file.type; }); + _albumPhotosCount = _list.albumIsPossible + ? (_list.files.size() - _albumVideosCount) + : 0; + setupTitleText(); + setupSendWayControls(); + setupCaption(); +} + +void SendFilesBox::setupSendWayControls() { + if (_compressConfirm == CompressConfirm::None) { + return; + } + const auto addRadio = [&]( + object_ptr> &button, + SendWay value, + const QString &text) { + const auto &style = st::defaultBoxCheckbox; + button.create(this, _sendWay, value, text, style); + }; + if (_list.albumIsPossible) { + addRadio(_sendAlbum, SendWay::Album, lang(lng_send_album)); + } + if (!_list.albumIsPossible || _albumPhotosCount > 0) { + addRadio(_sendPhotos, SendWay::Photos, (_list.files.size() == 1) + ? lang(lng_send_photo) + : (_albumVideosCount > 0) + ? (_list.albumIsPossible + ? lang(lng_send_separate_photos_videos) + : lng_send_photos_videos(lt_count, _list.files.size())) + : (_list.albumIsPossible + ? lang(lng_send_separate_photos) + : lng_send_photos(lt_count, _list.files.size()))); + } + addRadio(_sendFiles, SendWay::Files, (_list.files.size() == 1) + ? lang(lng_send_file) + : lng_send_files(lt_count, _list.files.size())); + _sendWay->setChangedCallback([this](SendWay value) { + if (_albumPreview) { + _albumPreview->setSendWay(value); + } + setInnerFocus(); + }); +} + +void SendFilesBox::setupCaption() { + if (!_caption) { + return; + } + + _caption->setMaxLength(MaxPhotoCaption); + _caption->setCtrlEnterSubmit(Ui::CtrlEnterSubmit::Both); + connect(_caption, &Ui::InputArea::resized, this, [this] { + captionResized(); + }); + connect(_caption, &Ui::InputArea::submitted, this, [this]( + bool ctrlShiftEnter) { + send(ctrlShiftEnter); + }); + connect(_caption, &Ui::InputArea::cancelled, this, [this] { + closeBox(); + }); +} + +void SendFilesBox::captionResized() { updateBoxSize(); updateControlsGeometry(); update(); } -void SendAlbumBox::updateBoxSize() { - auto newHeight = st::boxPhotoPadding.top() + _thumbsHeight; - if (_caption) { - newHeight += st::boxPhotoCaptionSkip + _caption->height(); +void SendFilesBox::setupTitleText() { + if (_list.files.size() > 1) { + const auto onlyImages = (_compressConfirm != CompressConfirm::None) + && (_albumVideosCount == 0); + _titleText = onlyImages + ? lng_send_images_selected(lt_count, _list.files.size()) + : lng_send_files_selected(lt_count, _list.files.size()); + _titleHeight = st::boxTitleHeight; + } else { + _titleText = QString(); + _titleHeight = 0; } - setDimensions(st::boxWideWidth, newHeight); } -void SendAlbumBox::keyPressEvent(QKeyEvent *e) { +void SendFilesBox::updateBoxSize() { + auto footerHeight = 0; + if (_caption) { + footerHeight += st::boxPhotoCaptionSkip + _caption->height(); + } + const auto pointers = { + _sendAlbum.data(), + _sendPhotos.data(), + _sendFiles.data() + }; + for (auto pointer : pointers) { + if (pointer) { + footerHeight += st::boxPhotoCompressedSkip + + pointer->heightNoMargins(); + } + } + _footerHeight = footerHeight; +} + +void SendFilesBox::keyPressEvent(QKeyEvent *e) { if (e->key() == Qt::Key_Enter || e->key() == Qt::Key_Return) { const auto modifiers = e->modifiers(); const auto ctrl = modifiers.testFlag(Qt::ControlModifier) @@ -622,36 +963,56 @@ void SendAlbumBox::keyPressEvent(QKeyEvent *e) { } } -void SendAlbumBox::paintEvent(QPaintEvent *e) { +void SendFilesBox::paintEvent(QPaintEvent *e) { BoxContent::paintEvent(e); - Painter p(this); + if (!_titleText.isEmpty()) { + Painter p(this); - const auto left = (st::boxWideWidth - st::sendMediaPreviewSize) / 2; - const auto top = st::boxPhotoPadding.top(); - for (const auto &thumb : _thumbs) { - p.drawPixmap( - left + thumb.layout.geometry.x(), - top + thumb.layout.geometry.y(), - thumb.image); + p.setFont(st::boxPhotoTitleFont); + p.setPen(st::boxTitleFg); + p.drawTextLeft( + st::boxPhotoTitlePosition.x(), + st::boxPhotoTitlePosition.y(), + width(), + _titleText); } } -void SendAlbumBox::resizeEvent(QResizeEvent *e) { +void SendFilesBox::resizeEvent(QResizeEvent *e) { BoxContent::resizeEvent(e); updateControlsGeometry(); } -void SendAlbumBox::updateControlsGeometry() { +void SendFilesBox::updateControlsGeometry() { auto bottom = height(); if (_caption) { _caption->resize(st::sendMediaPreviewSize, _caption->height()); - _caption->moveToLeft(st::boxPhotoPadding.left(), bottom - _caption->height()); + _caption->moveToLeft( + st::boxPhotoPadding.left(), + bottom - _caption->height()); bottom -= st::boxPhotoCaptionSkip + _caption->height(); } + const auto pointers = { + _sendAlbum.data(), + _sendPhotos.data(), + _sendFiles.data() + }; + for (auto pointer : base::reversed(pointers)) { + if (pointer) { + pointer->moveToLeft( + st::boxPhotoPadding.left(), + bottom - pointer->heightNoMargins()); + bottom -= st::boxPhotoCompressedSkip + pointer->heightNoMargins(); + } + } + if (_preview) { + _preview->resize(width(), bottom - _titleHeight); + _preview->move(0, _titleHeight); + } } -void SendAlbumBox::setInnerFocus() { +void SendFilesBox::setInnerFocus() { if (!_caption || _caption->isHidden()) { setFocus(); } else { @@ -659,9 +1020,15 @@ void SendAlbumBox::setInnerFocus() { } } -void SendAlbumBox::send(bool ctrlShiftEnter) { +void SendFilesBox::send(bool ctrlShiftEnter) { + // TODO settings + //if (_compressed && _compressConfirm == CompressConfirm::Auto && _compressed->checked() != cCompressPastedImage()) { + // cSetCompressPastedImage(_compressed->checked()); + // Local::writeUserSettings(); + //} _confirmed = true; if (_confirmedCallback) { + const auto way = _sendWay ? _sendWay->value() : SendWay::Files; auto caption = _caption ? TextUtilities::PrepareForSending( _caption->getLastText(), @@ -669,10 +1036,11 @@ void SendAlbumBox::send(bool ctrlShiftEnter) { : QString(); _confirmedCallback( std::move(_list), - caption, + way, + std::move(caption), ctrlShiftEnter); } closeBox(); } -SendAlbumBox::~SendAlbumBox() = default; +SendFilesBox::~SendFilesBox() = default; diff --git a/Telegram/SourceFiles/boxes/send_files_box.h b/Telegram/SourceFiles/boxes/send_files_box.h index 7a784da1e..edca3b190 100644 --- a/Telegram/SourceFiles/boxes/send_files_box.h +++ b/Telegram/SourceFiles/boxes/send_files_box.h @@ -20,12 +20,16 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org */ #pragma once +#include #include "boxes/abstract_box.h" #include "storage/localimageloader.h" #include "storage/storage_media_prepare.h" namespace Ui { -class Checkbox; +template +class Radioenum; +template +class RadioenumGroup; class RoundButton; class InputArea; struct GroupMediaLayout; @@ -33,6 +37,12 @@ struct GroupMediaLayout; class SendFilesBox : public BoxContent { public: + enum class SendWay { + Album, + Photos, + Files, + }; + SendFilesBox( QWidget*, Storage::PreparedList &&list, @@ -41,7 +51,7 @@ public: void setConfirmedCallback( base::lambda callback) { _confirmedCallback = std::move(callback); @@ -61,102 +71,57 @@ protected: void resizeEvent(QResizeEvent *e) override; private: - void prepareSingleFileLayout(); - void prepareDocumentLayout(); - void prepareGifPreview(); - void clipCallback(Media::Clip::Notification notification); + class AlbumPreview; + struct AlbumThumb; + + void initSendWay(); + void initPreview(rpl::producer desiredPreviewHeight); + + void setupControls(); + void setupSendWayControls(); + void setupCaption(); + void setupShadows( + not_null wrap, + not_null content); + + void prepareSingleFilePreview(); + void prepareAlbumPreview(); void send(bool ctrlShiftEnter = false); void captionResized(); - void compressedChange(); - void updateTitleText(); + void setupTitleText(); void updateBoxSize(); void updateControlsGeometry(); - base::lambda getSendButtonText() const; QString _titleText; + int _titleHeight = 0; + Storage::PreparedList _list; CompressConfirm _compressConfirm = CompressConfirm::None; - bool _animated = false; - - QPixmap _preview; - int _previewLeft = 0; - int _previewWidth = 0; - int _previewHeight = 0; - Media::Clip::ReaderPointer _gifPreview; - - QPixmap _fileThumb; - Text _nameText; - bool _fileIsAudio = false; - bool _fileIsImage = false; - QString _statusText; - int _statusWidth = 0; base::lambda _confirmedCallback; base::lambda _cancelledCallback; bool _confirmed = false; object_ptr _caption = { nullptr }; - object_ptr _compressed = { nullptr }; + object_ptr> _sendAlbum = { nullptr }; + object_ptr> _sendPhotos = { nullptr }; + object_ptr> _sendFiles = { nullptr }; + std::shared_ptr> _sendWay; + + rpl::variable _footerHeight = 0; + + QWidget *_preview = nullptr; + AlbumPreview *_albumPreview = nullptr; + int _albumVideosCount = 0; + int _albumPhotosCount = 0; QPointer _send; }; - -class SendAlbumBox : public BoxContent { -public: - SendAlbumBox(QWidget*, Storage::PreparedList &&list); - - void setConfirmedCallback( - base::lambda callback) { - _confirmedCallback = std::move(callback); - } - void setCancelledCallback(base::lambda callback) { - _cancelledCallback = std::move(callback); - } - - ~SendAlbumBox(); - -protected: - void prepare() override; - void setInnerFocus() override; - - void keyPressEvent(QKeyEvent *e) override; - void paintEvent(QPaintEvent *e) override; - void resizeEvent(QResizeEvent *e) override; - -private: - struct Thumb; - - void prepareThumbs(); - Thumb prepareThumb( - const QImage &preview, - const Ui::GroupMediaLayout &layout) const; - - void send(bool ctrlShiftEnter = false); - void captionResized(); - - void updateBoxSize(); - void updateControlsGeometry(); - - Storage::PreparedList _list; - - std::vector _thumbs; - int _thumbsHeight = 0; - - base::lambda _confirmedCallback; - base::lambda _cancelledCallback; - bool _confirmed = false; - - object_ptr _caption = { nullptr }; - -}; diff --git a/Telegram/SourceFiles/history/history_widget.cpp b/Telegram/SourceFiles/history/history_widget.cpp index 07b6386b5..05665f3f9 100644 --- a/Telegram/SourceFiles/history/history_widget.cpp +++ b/Telegram/SourceFiles/history/history_widget.cpp @@ -106,6 +106,17 @@ ApiWrap::RequestMessageDataCallback replyEditMessageDataCallback() { }; } +void ActivateWindowDelayed(not_null controller) { + const auto window = controller->window(); + const auto weak = make_weak(window.get()); + window->activateWindow(); + crl::on_main([=] { + if (weak) { + weak->activateWindow(); + } + }); +} + } // namespace ReportSpamPanel::ReportSpamPanel(QWidget *parent) : TWidget(parent), @@ -542,11 +553,11 @@ HistoryWidget::HistoryWidget(QWidget *parent, not_null cont _attachDragDocument->setDroppedCallback([this](const QMimeData *data) { confirmSendingFiles(data, CompressConfirm::No); - this->controller()->window()->activateWindow(); + ActivateWindowDelayed(this->controller()); }); _attachDragPhoto->setDroppedCallback([this](const QMimeData *data) { confirmSendingFiles(data, CompressConfirm::Yes); - this->controller()->window()->activateWindow(); + ActivateWindowDelayed(this->controller()); }); connect(&_updateEditTimeLeftDisplay, SIGNAL(timeout()), this, SLOT(updateField())); @@ -1275,7 +1286,7 @@ void HistoryWidget::onRecordDone( qint32 samples) { if (!canWriteMessage() || result.isEmpty()) return; - App::wnd()->activateWindow(); + ActivateWindowDelayed(controller()); const auto duration = samples / Media::Player::kDefaultFrequency; auto options = ApiWrap::SendOptions(_history); options.replyTo = replyToId(); @@ -3940,48 +3951,15 @@ void HistoryWidget::updateFieldPlaceholder() { updateSendButtonType(); } -template -bool HistoryWidget::showSendFilesBox( - object_ptr box, - const QString &insertTextOnCancel, - SendCallback callback) { - App::wnd()->activateWindow(); - - const auto confirmedCallback = [=, sendCallback = std::move(callback)]( - Storage::PreparedList &&list, - bool compressed, - const QString &caption, - bool ctrlShiftEnter) { - if (!canWriteMessage()) return; - - sendCallback( - std::move(list), - compressed, - caption, - replyToId()); - }; - box->setConfirmedCallback( - base::lambda_guarded(this, std::move(confirmedCallback))); - - if (!insertTextOnCancel.isEmpty()) { - box->setCancelledCallback(base::lambda_guarded(this, [=] { - _field->textCursor().insertText(insertTextOnCancel); - })); - } - - Ui::show(std::move(box)); - return true; -} - bool HistoryWidget::showSendingFilesError( const Storage::PreparedList &list) const { - App::wnd()->activateWindow(); const auto text = [&] { if (const auto megagroup = _peer ? _peer->asMegagroup() : nullptr) { if (megagroup->restricted(ChannelRestriction::f_send_media)) { return lang(lng_restricted_send_media); } - } else if (!canWriteMessage()) { + } + if (!canWriteMessage()) { return lang(lng_forward_send_files_cant); } using Error = Storage::PreparedList::Error; @@ -4015,70 +3993,70 @@ bool HistoryWidget::confirmSendingFiles(const QMimeData *data) { bool HistoryWidget::confirmSendingFiles( const QList &files, - CompressConfirm compressed) { + CompressConfirm compressed, + const QString &insertTextOnCancel) { return confirmSendingFiles( Storage::PrepareMediaList(files, st::sendMediaPreviewSize), - compressed); + compressed, + insertTextOnCancel); } bool HistoryWidget::confirmSendingFiles( const QStringList &files, - CompressConfirm compressed) { + CompressConfirm compressed, + const QString &insertTextOnCancel) { return confirmSendingFiles( Storage::PrepareMediaList(files, st::sendMediaPreviewSize), - compressed); + compressed, + insertTextOnCancel); } bool HistoryWidget::confirmSendingFiles( Storage::PreparedList &&list, - CompressConfirm compressed) { + CompressConfirm compressed, + const QString &insertTextOnCancel) { if (showSendingFilesError(list)) { return false; } - if (list.albumIsPossible) { - auto box = Ui::show(Box(std::move(list))); - const auto confirmedCallback = [=]( - Storage::PreparedList &&list, - const QString &caption, - bool ctrlShiftEnter) { - if (!canWriteMessage()) return; - uploadFilesAfterConfirmation( - std::move(list), - SendMediaType::Photo, - caption, - replyToId(), - std::make_shared()); - }; - box->setConfirmedCallback( - base::lambda_guarded(this, std::move(confirmedCallback))); - return true; - } else { - const auto insertTextOnCancel = QString(); - auto sendCallback = [this]( - Storage::PreparedList &&list, - bool compressed, - const QString &caption, - MsgId replyTo) { - const auto type = compressed - ? SendMediaType::Photo - : SendMediaType::File; - uploadFilesAfterConfirmation( - std::move(list), - type, - caption, - replyTo); - }; - const auto noCompressOption = (list.files.size() > 1) - && !list.allFilesForCompress; - const auto boxCompressConfirm = noCompressOption - ? CompressConfirm::None - : compressed; - return showSendFilesBox( - Box(std::move(list), boxCompressConfirm), - insertTextOnCancel, - std::move(sendCallback)); + const auto noCompressOption = (list.files.size() > 1) + && !list.allFilesForCompress + && !list.albumIsPossible; + const auto boxCompressConfirm = noCompressOption + ? CompressConfirm::None + : compressed; + + auto box = Box(std::move(list), boxCompressConfirm); + box->setConfirmedCallback(base::lambda_guarded(this, [=]( + Storage::PreparedList &&list, + SendFilesBox::SendWay way, + const QString &caption, + bool ctrlShiftEnter) { + if (showSendingFilesError(list)) { + return; + } + const auto type = (way == SendFilesBox::SendWay::Files) + ? SendMediaType::File + : SendMediaType::Photo; + const auto album = (way == SendFilesBox::SendWay::Album) + ? std::make_shared() + : nullptr; + uploadFilesAfterConfirmation( + std::move(list), + type, + caption, + replyToId(), + album); + })); + if (!insertTextOnCancel.isEmpty()) { + box->setCancelledCallback(base::lambda_guarded(this, [=] { + _field->textCursor().insertText(insertTextOnCancel); + })); } + + ActivateWindowDelayed(controller()); + Ui::show(std::move(box)); + return true; } bool HistoryWidget::confirmSendingFiles( @@ -4086,31 +4064,18 @@ bool HistoryWidget::confirmSendingFiles( QByteArray &&content, CompressConfirm compressed, const QString &insertTextOnCancel) { - if (!canWriteMessage() || image.isNull()) return false; + if (image.isNull()) { + return false; + } - App::wnd()->activateWindow(); - auto sendCallback = [this]( - Storage::PreparedList &&list, - bool compressed, - const QString &caption, - MsgId replyTo) { - const auto type = compressed - ? SendMediaType::Photo - : SendMediaType::File; - uploadFilesAfterConfirmation( - std::move(list), - type, - caption, - replyTo); - }; auto list = Storage::PrepareMediaFromImage( std::move(image), std::move(content), st::sendMediaPreviewSize); - return showSendFilesBox( - Box(std::move(list), compressed), - insertTextOnCancel, - std::move(sendCallback)); + return confirmSendingFiles( + std::move(list), + compressed, + insertTextOnCancel); } bool HistoryWidget::confirmSendingFiles( @@ -4121,15 +4086,14 @@ bool HistoryWidget::confirmSendingFiles( return false; } - auto urls = data->urls(); - if (!urls.isEmpty()) { - for_const (auto &url, urls) { - if (url.isLocalFile()) { - confirmSendingFiles(urls, compressed); - return true; - } + const auto urls = data->urls(); + for (const auto &url : urls) { + if (url.isLocalFile()) { + confirmSendingFiles(urls, compressed, insertTextOnCancel); + return true; } } + if (data->hasImage()) { auto image = qvariant_cast(data->imageData()); if (!image.isNull()) { @@ -4147,7 +4111,7 @@ bool HistoryWidget::confirmSendingFiles( void HistoryWidget::uploadFiles( Storage::PreparedList &&list, SendMediaType type) { - if (!canWriteMessage()) return; + ActivateWindowDelayed(controller()); auto caption = QString(); uploadFilesAfterConfirmation( diff --git a/Telegram/SourceFiles/history/history_widget.h b/Telegram/SourceFiles/history/history_widget.h index fec7f5c33..deecb33de 100644 --- a/Telegram/SourceFiles/history/history_widget.h +++ b/Telegram/SourceFiles/history/history_widget.h @@ -499,10 +499,12 @@ private: bool confirmSendingFiles( const QList &files, - CompressConfirm compressed); + CompressConfirm compressed, + const QString &insertTextOnCancel = QString()); bool confirmSendingFiles( const QStringList &files, - CompressConfirm compressed); + CompressConfirm compressed, + const QString &insertTextOnCancel = QString()); bool confirmSendingFiles( QImage &&image, QByteArray &&content, @@ -514,15 +516,10 @@ private: const QString &insertTextOnCancel = QString()); bool confirmSendingFiles( Storage::PreparedList &&list, - CompressConfirm compressed); + CompressConfirm compressed, + const QString &insertTextOnCancel = QString()); bool showSendingFilesError(const Storage::PreparedList &list) const; - template - bool showSendFilesBox( - object_ptr box, - const QString &insertTextOnCancel, - SendCallback callback); - void uploadFiles(Storage::PreparedList &&list, SendMediaType type); void uploadFile(const QByteArray &fileContent, SendMediaType type); diff --git a/Telegram/SourceFiles/storage/storage_media_prepare.cpp b/Telegram/SourceFiles/storage/storage_media_prepare.cpp index 39ddf142e..36627c4a4 100644 --- a/Telegram/SourceFiles/storage/storage_media_prepare.cpp +++ b/Telegram/SourceFiles/storage/storage_media_prepare.cpp @@ -83,7 +83,8 @@ bool PrepareAlbumMediaIsWaiting( &file.information->media)) { if (ValidPhotoForAlbum(*image)) { file.preview = image->data.scaledToWidth( - previewWidth * cIntRetinaFactor(), + std::min(previewWidth, convertScale(image->data.width())) + * cIntRetinaFactor(), Qt::SmoothTransformation); file.preview.setDevicePixelRatio(cRetinaFactor()); file.type = PreparedFile::AlbumType::Photo; @@ -119,6 +120,11 @@ 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()); } } diff --git a/Telegram/SourceFiles/ui/images.h b/Telegram/SourceFiles/ui/images.h index 64147f49e..9abc9f7ce 100644 --- a/Telegram/SourceFiles/ui/images.h +++ b/Telegram/SourceFiles/ui/images.h @@ -212,6 +212,11 @@ enum class Option { RoundedTopRight = (1 << 6), RoundedBottomLeft = (1 << 7), RoundedBottomRight = (1 << 8), + RoundedAll = (None + | RoundedTopLeft + | RoundedTopRight + | RoundedBottomLeft + | RoundedBottomRight), Colored = (1 << 9), TransparentBackground = (1 << 10), };