| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087 |
- /*
- This file is part of Telegram Desktop,
- the official desktop application for the Telegram messaging service.
- For license and copyright information please follow this link:
- https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
- */
- #include "boxes/edit_caption_box.h"
- #include "api/api_editing.h"
- #include "api/api_text_entities.h"
- #include "apiwrap.h"
- #include "base/event_filter.h"
- #include "boxes/premium_limits_box.h"
- #include "boxes/premium_preview_box.h"
- #include "chat_helpers/emoji_suggestions_widget.h"
- #include "chat_helpers/field_autocomplete.h"
- #include "chat_helpers/message_field.h"
- #include "chat_helpers/tabbed_panel.h"
- #include "chat_helpers/tabbed_selector.h"
- #include "core/application.h"
- #include "core/core_settings.h"
- #include "core/file_utilities.h"
- #include "core/mime_type.h"
- #include "data/data_document.h"
- #include "data/data_photo_media.h"
- #include "data/data_session.h"
- #include "data/data_user.h"
- #include "data/data_premium_limits.h"
- #include "data/stickers/data_stickers.h"
- #include "data/stickers/data_custom_emoji.h"
- #include "editor/editor_layer_widget.h"
- #include "editor/photo_editor.h"
- #include "editor/photo_editor_layer_widget.h"
- #include "history/history_drag_area.h"
- #include "history/history_item.h"
- #include "history/history.h"
- #include "lang/lang_keys.h"
- #include "main/main_session.h"
- #include "main/main_session_settings.h"
- #include "mainwidget.h" // controller->content() -> QWidget*
- #include "menu/menu_send.h"
- #include "mtproto/mtproto_config.h"
- #include "platform/platform_specific.h"
- #include "storage/localimageloader.h" // SendMediaType
- #include "storage/storage_media_prepare.h"
- #include "ui/boxes/confirm_box.h"
- #include "ui/chat/attach/attach_item_single_file_preview.h"
- #include "ui/chat/attach/attach_item_single_media_preview.h"
- #include "ui/chat/attach/attach_single_file_preview.h"
- #include "ui/chat/attach/attach_single_media_preview.h"
- #include "ui/controls/emoji_button.h"
- #include "ui/effects/scroll_content_shadow.h"
- #include "ui/image/image.h"
- #include "ui/toast/toast.h"
- #include "ui/painter.h"
- #include "ui/ui_utility.h"
- #include "ui/widgets/checkbox.h"
- #include "ui/widgets/fields/input_field.h"
- #include "ui/widgets/scroll_area.h"
- #include "ui/wrap/slide_wrap.h"
- #include "ui/wrap/vertical_layout.h"
- #include "window/window_session_controller.h"
- #include "styles/style_boxes.h"
- #include "styles/style_chat.h"
- #include "styles/style_chat_helpers.h"
- #include "styles/style_layers.h"
- #include <QtCore/QMimeData>
- namespace {
- constexpr auto kChangesDebounceTimeout = crl::time(1000);
- [[nodiscard]] Ui::PreparedList ListFromMimeData(
- not_null<const QMimeData*> data,
- bool premium) {
- using Error = Ui::PreparedList::Error;
- const auto list = Core::ReadMimeUrls(data);
- auto result = !list.isEmpty()
- ? Storage::PrepareMediaList(
- list.mid(0, 1), // When we edit media, we need only 1 file.
- st::sendMediaPreviewSize,
- premium)
- : Ui::PreparedList(Error::EmptyFile, QString());
- if (result.error == Error::None) {
- return result;
- } else if (auto read = Core::ReadMimeImage(data)) {
- return Storage::PrepareMediaFromImage(
- std::move(read.image),
- std::move(read.content),
- st::sendMediaPreviewSize);
- }
- return result;
- }
- [[nodiscard]] Ui::AlbumType ComputeAlbumType(not_null<HistoryItem*> item) {
- if (item->groupId().empty()) {
- return Ui::AlbumType();
- }
- const auto media = item->media();
- if (media->photo()) {
- return Ui::AlbumType::PhotoVideo;
- } else if (const auto document = media->document()) {
- if (document->isVideoFile()) {
- return Ui::AlbumType::PhotoVideo;
- } else if (document->isSong()) {
- return Ui::AlbumType::Music;
- } else {
- return Ui::AlbumType::File;
- }
- }
- return Ui::AlbumType();
- }
- [[nodiscard]] bool CanBeCompressed(Ui::AlbumType type) {
- return (type == Ui::AlbumType::None)
- || (type == Ui::AlbumType::PhotoVideo);
- }
- void ChooseReplacement(
- not_null<Window::SessionController*> controller,
- Ui::AlbumType type,
- Fn<void(Ui::PreparedList&&)> chosen) {
- const auto weak = base::make_weak(controller);
- const auto callback = [=](FileDialog::OpenResult &&result) {
- const auto strong = weak.get();
- if (!strong) {
- return;
- }
- const auto showError = [=](tr::phrase<> t) {
- if (const auto strong = weak.get()) {
- strong->showToast(t(tr::now));
- }
- };
- const auto checkResult = [=](const Ui::PreparedList &list) {
- if (list.files.size() != 1) {
- return false;
- }
- const auto &file = list.files.front();
- const auto mime = file.information->filemime;
- if (Core::IsMimeSticker(mime)) {
- showError(tr::lng_edit_media_invalid_file);
- return false;
- } else if (type != Ui::AlbumType::None
- && !file.canBeInAlbumType(type)) {
- showError(tr::lng_edit_media_album_error);
- return false;
- }
- return true;
- };
- const auto premium = strong->session().premium();
- auto list = Storage::PreparedFileFromFilesDialog(
- std::move(result),
- checkResult,
- showError,
- st::sendMediaPreviewSize,
- premium);
- if (list) {
- chosen(std::move(*list));
- }
- };
- const auto filters = (type == Ui::AlbumType::PhotoVideo)
- ? FileDialog::PhotoVideoFilesFilter()
- : FileDialog::AllFilesFilter();
- FileDialog::GetOpenPath(
- controller->content().get(),
- tr::lng_choose_file(tr::now),
- filters,
- crl::guard(controller, callback));
- }
- void EditPhotoImage(
- not_null<Window::SessionController*> controller,
- std::shared_ptr<Data::PhotoMedia> media,
- bool spoilered,
- Fn<void(Ui::PreparedList)> done) {
- const auto large = media
- ? media->image(Data::PhotoSize::Large)
- : nullptr;
- const auto parent = controller->content();
- const auto previewWidth = st::sendMediaPreviewSize;
- auto callback = [=](const Editor::PhotoModifications &mods) {
- if (!mods) {
- return;
- }
- const auto large = media->image(Data::PhotoSize::Large);
- if (!large) {
- return;
- }
- auto copy = large->original();
- auto list = Storage::PrepareMediaFromImage(
- std::move(copy),
- QByteArray(),
- previewWidth);
- using ImageInfo = Ui::PreparedFileInformation::Image;
- auto &file = list.files.front();
- file.spoiler = spoilered;
- const auto image = std::get_if<ImageInfo>(&file.information->media);
- image->modifications = mods;
- const auto sideLimit = PhotoSideLimit();
- Storage::UpdateImageDetails(file, previewWidth, sideLimit);
- done(std::move(list));
- };
- if (!large) {
- return;
- }
- const auto fileImage = std::make_shared<Image>(*large);
- auto editor = base::make_unique_q<Editor::PhotoEditor>(
- parent,
- &controller->window(),
- fileImage,
- Editor::PhotoModifications());
- const auto raw = editor.get();
- auto layer = std::make_unique<Editor::LayerWidget>(
- parent,
- std::move(editor));
- Editor::InitEditorLayer(layer.get(), raw, std::move(callback));
- controller->showLayer(std::move(layer), Ui::LayerOption::KeepOther);
- }
- } // namespace
- EditCaptionBox::EditCaptionBox(
- QWidget*,
- not_null<Window::SessionController*> controller,
- not_null<HistoryItem*> item,
- TextWithTags &&text,
- bool spoilered,
- bool invertCaption,
- Ui::PreparedList &&list,
- Fn<void()> saved)
- : _controller(controller)
- , _historyItem(item)
- , _isAllowedEditMedia(item->allowsEditMedia())
- , _albumType(ComputeAlbumType(item))
- , _controls(base::make_unique_q<Ui::VerticalLayout>(this))
- , _scroll(base::make_unique_q<Ui::ScrollArea>(this, st::boxScroll))
- , _field(base::make_unique_q<Ui::InputField>(
- this,
- st::defaultComposeFiles.caption,
- Ui::InputField::Mode::MultiLine,
- tr::lng_photo_caption()))
- , _emojiToggle(base::make_unique_q<Ui::EmojiButton>(
- this,
- st::defaultComposeFiles.emoji))
- , _initialText(std::move(text))
- , _initialList(std::move(list))
- , _saved(std::move(saved)) {
- Expects(!_initialList.files.empty());
- Expects(item->allowsEditMedia());
- _mediaEditManager.start(item, spoilered, invertCaption);
- _controller->session().data().itemRemoved(
- _historyItem->fullId()
- ) | rpl::start_with_next([=] {
- closeBox();
- }, lifetime());
- }
- EditCaptionBox::~EditCaptionBox() = default;
- void EditCaptionBox::StartMediaReplace(
- not_null<Window::SessionController*> controller,
- FullMsgId itemId,
- TextWithTags text,
- bool spoilered,
- bool invertCaption,
- Fn<void()> saved) {
- const auto session = &controller->session();
- const auto item = session->data().message(itemId);
- if (!item) {
- return;
- }
- const auto show = [=](Ui::PreparedList &&list) mutable {
- controller->show(Box<EditCaptionBox>(
- controller,
- item,
- std::move(text),
- spoilered,
- invertCaption,
- std::move(list),
- std::move(saved)));
- };
- ChooseReplacement(
- controller,
- ComputeAlbumType(item),
- crl::guard(controller, show));
- }
- void EditCaptionBox::StartMediaReplace(
- not_null<Window::SessionController*> controller,
- FullMsgId itemId,
- Ui::PreparedList &&list,
- TextWithTags text,
- bool spoilered,
- bool invertCaption,
- Fn<void()> saved) {
- const auto session = &controller->session();
- const auto item = session->data().message(itemId);
- if (!item) {
- return;
- }
- const auto type = ComputeAlbumType(item);
- const auto showError = [=](tr::phrase<> t) {
- controller->showToast(t(tr::now));
- };
- const auto checkResult = [=](const Ui::PreparedList &list) {
- if (list.files.size() != 1) {
- return false;
- }
- const auto &file = list.files.front();
- const auto mime = file.information->filemime;
- if (Core::IsMimeSticker(mime)) {
- showError(tr::lng_edit_media_invalid_file);
- return false;
- } else if (type != Ui::AlbumType::None
- && !file.canBeInAlbumType(type)) {
- showError(tr::lng_edit_media_album_error);
- return false;
- }
- return true;
- };
- if (list.error != Ui::PreparedList::Error::None) {
- showError(tr::lng_send_media_invalid_files);
- } else if (checkResult(list)) {
- controller->show(Box<EditCaptionBox>(
- controller,
- item,
- std::move(text),
- spoilered,
- invertCaption,
- std::move(list),
- std::move(saved)));
- }
- }
- void EditCaptionBox::StartPhotoEdit(
- not_null<Window::SessionController*> controller,
- std::shared_ptr<Data::PhotoMedia> media,
- FullMsgId itemId,
- TextWithTags text,
- bool spoilered,
- bool invertCaption,
- Fn<void()> saved) {
- const auto session = &controller->session();
- const auto item = session->data().message(itemId);
- if (!item) {
- return;
- }
- EditPhotoImage(controller, media, spoilered, [=](
- Ui::PreparedList &&list) mutable {
- const auto item = session->data().message(itemId);
- if (!item) {
- return;
- }
- controller->show(Box<EditCaptionBox>(
- controller,
- item,
- std::move(text),
- spoilered,
- invertCaption,
- std::move(list),
- std::move(saved)));
- });
- }
- void EditCaptionBox::showFinished() {
- if (const auto raw = _autocomplete.get()) {
- InvokeQueued(raw, [=] {
- raw->raise();
- });
- }
- }
- void EditCaptionBox::prepare() {
- const auto button = addButton(tr::lng_settings_save(), [=] { save(); });
- addButton(tr::lng_cancel(), [=] { closeBox(); });
- const auto details = crl::guard(this, [=] {
- auto result = SendMenu::Details();
- const auto allWithSpoilers = ranges::all_of(
- _preparedList.files,
- &Ui::PreparedFile::spoiler);
- result.spoiler = !_preparedList.hasSpoilerMenu(!_asFile)
- ? SendMenu::SpoilerState::None
- : allWithSpoilers
- ? SendMenu::SpoilerState::Enabled
- : SendMenu::SpoilerState::Possible;
- const auto canMoveCaption = _preparedList.canMoveCaption(
- false,
- !_asFile
- ) && _field && HasSendText(_field);
- result.caption = !canMoveCaption
- ? SendMenu::CaptionState::None
- : _mediaEditManager.invertCaption()
- ? SendMenu::CaptionState::Above
- : SendMenu::CaptionState::Below;
- return result;
- });
- const auto callback = [=](SendMenu::Action action, const auto &) {
- _mediaEditManager.apply(action);
- rebuildPreview();
- };
- SendMenu::SetupMenuAndShortcuts(
- button,
- nullptr,
- details,
- crl::guard(this, callback));
- updateBoxSize();
- setupField();
- setupEmojiPanel();
- setInitialText();
- if (!setPreparedList(std::move(_initialList))) {
- crl::on_main(this, [=] { closeBox(); });
- return;
- }
- setupEditEventHandler();
- SetupShadowsToScrollContent(this, _scroll, _contentHeight.events());
- setupControls();
- setupPhotoEditorEventHandler();
- setupDragArea();
- captionResized();
- }
- void EditCaptionBox::rebuildPreview() {
- const auto gifPaused = [controller = _controller] {
- return controller->isGifPausedAtLeastFor(
- Window::GifPauseReason::Layer);
- };
- applyChanges();
- if (_preparedList.files.empty()) {
- const auto media = _historyItem->media();
- const auto photo = media->photo();
- const auto document = media->document();
- _isPhoto = (photo != nullptr);
- if (photo || document->isVideoFile() || document->isAnimation()) {
- const auto media = Ui::CreateChild<Ui::ItemSingleMediaPreview>(
- this,
- st::defaultComposeControls,
- gifPaused,
- _historyItem,
- Ui::AttachControls::Type::EditOnly);
- _photoMedia = media->sharedPhotoMedia();
- _content.reset(media);
- } else {
- _content.reset(Ui::CreateChild<Ui::ItemSingleFilePreview>(
- this,
- st::defaultComposeControls,
- _historyItem,
- Ui::AttachControls::Type::EditOnly));
- }
- } else {
- const auto &file = _preparedList.files.front();
- const auto isVideoFile = file.isVideoFile();
- const auto media = Ui::SingleMediaPreview::Create(
- this,
- st::defaultComposeControls,
- gifPaused,
- file,
- [=](Ui::AttachActionType type) {
- return (type != Ui::AttachActionType::EditCover)
- || isVideoFile;
- },
- Ui::AttachControls::Type::EditOnly);
- _isPhoto = (media && media->isPhoto());
- const auto withCheckbox = _isPhoto && CanBeCompressed(_albumType);
- if (media && (!withCheckbox || !_asFile)) {
- media->spoileredChanges(
- ) | rpl::start_with_next([=](bool spoilered) {
- _mediaEditManager.apply({ .type = spoilered
- ? SendMenu::ActionType::SpoilerOn
- : SendMenu::ActionType::SpoilerOff
- });
- }, media->lifetime());
- _content.reset(media);
- } else {
- _content.reset(Ui::CreateChild<Ui::SingleFilePreview>(
- this,
- st::defaultComposeControls,
- file,
- Ui::AttachControls::Type::EditOnly));
- }
- }
- Assert(_content != nullptr);
- rpl::combine(
- _content->heightValue(),
- _footerHeight.value(),
- rpl::single(st::boxPhotoPadding.top()),
- rpl::mappers::_1 + rpl::mappers::_2 + rpl::mappers::_3
- ) | rpl::start_with_next([=](int height) {
- setDimensions(
- st::boxWideWidth,
- std::min(st::sendMediaPreviewHeightMax, height),
- true);
- }, _content->lifetime());
- _content->editRequests(
- ) | rpl::start_to_stream(_editMediaClicks, _content->lifetime());
- _content->modifyRequests(
- ) | rpl::start_to_stream(_photoEditorOpens, _content->lifetime());
- _content->heightValue(
- ) | rpl::start_to_stream(_contentHeight, _content->lifetime());
- _scroll->setOwnedWidget(
- object_ptr<Ui::RpWidget>::fromRaw(_content.get()));
- _previewRebuilds.fire({});
- captionResized();
- }
- void EditCaptionBox::setupField() {
- const auto peer = _historyItem->history()->peer;
- const auto allow = [=](not_null<DocumentData*> emoji) {
- return Data::AllowEmojiWithoutPremium(peer, emoji);
- };
- InitMessageFieldHandlers(
- _controller,
- _field.get(),
- Window::GifPauseReason::Layer,
- allow);
- setupFieldAutocomplete();
- Ui::Emoji::SuggestionsController::Init(
- getDelegate()->outerContainer(),
- _field,
- &_controller->session(),
- { .suggestCustomEmoji = true, .allowCustomWithoutPremium = allow });
- _field->setSubmitSettings(
- Core::App().settings().sendSubmitWay());
- _field->setMaxHeight(st::defaultComposeFiles.caption.heightMax);
- _field->submits(
- ) | rpl::start_with_next([=] { save(); }, _field->lifetime());
- _field->cancelled(
- ) | rpl::start_with_next([=] {
- closeBox();
- }, _field->lifetime());
- _field->heightChanges(
- ) | rpl::start_with_next([=] {
- captionResized();
- }, _field->lifetime());
- _field->setMimeDataHook([=](
- not_null<const QMimeData*> data,
- Ui::InputField::MimeAction action) {
- if (action == Ui::InputField::MimeAction::Check) {
- if (!data->hasText() && !_isAllowedEditMedia) {
- return false;
- } else if (Storage::ValidateEditMediaDragData(data, _albumType)) {
- return true;
- }
- return data->hasText();
- } else if (action == Ui::InputField::MimeAction::Insert) {
- return fileFromClipboard(data);
- }
- Unexpected("Action in MimeData hook.");
- });
- }
- void EditCaptionBox::setupFieldAutocomplete() {
- const auto parent = getDelegate()->outerContainer();
- ChatHelpers::InitFieldAutocomplete(_autocomplete, {
- .parent = parent,
- .show = _controller->uiShow(),
- .field = _field.get(),
- .peer = _historyItem->history()->peer,
- .features = [=] {
- auto result = ChatHelpers::ComposeFeatures();
- result.autocompleteCommands = false;
- result.suggestStickersByEmoji = false;
- return result;
- },
- });
- const auto raw = _autocomplete.get();
- const auto scheduled = std::make_shared<bool>();
- const auto recountPostponed = [=] {
- if (*scheduled) {
- return;
- }
- *scheduled = true;
- Ui::PostponeCall(raw, [=] {
- *scheduled = false;
- auto field = Ui::MapFrom(parent, this, _field->geometry());
- _autocomplete->setBoundings(QRect(
- field.x() - _field->x(),
- st::defaultBox.margin.top(),
- width(),
- (field.y()
- + st::defaultComposeFiles.caption.textMargins.top()
- + st::defaultComposeFiles.caption.placeholderShift
- + st::defaultComposeFiles.caption.placeholderFont->height
- - st::defaultBox.margin.top())));
- });
- };
- for (auto w = (QWidget*)_field.get(); w; w = w->parentWidget()) {
- base::install_event_filter(raw, w, [=](not_null<QEvent*> e) {
- if (e->type() == QEvent::Move || e->type() == QEvent::Resize) {
- recountPostponed();
- }
- return base::EventFilterResult::Continue;
- });
- if (w == parent) {
- break;
- }
- }
- }
- void EditCaptionBox::setInitialText() {
- _field->setTextWithTags(
- _initialText,
- Ui::InputField::HistoryAction::Clear);
- auto cursor = _field->textCursor();
- cursor.movePosition(QTextCursor::End);
- _field->setTextCursor(cursor);
- _checkChangedTimer.setCallback([=] {
- if (_field->getTextWithAppliedMarkdown() == _initialText
- && _preparedList.files.empty()) {
- setCloseByOutsideClick(true);
- }
- });
- _field->changes(
- ) | rpl::start_with_next([=] {
- _checkChangedTimer.callOnce(kChangesDebounceTimeout);
- setCloseByOutsideClick(false);
- }, _field->lifetime());
- }
- void EditCaptionBox::setupControls() {
- auto hintLabelToggleOn = _previewRebuilds.events_starting_with(
- {}
- ) | rpl::map([=] {
- return _controller->session().settings().photoEditorHintShown()
- ? (_isPhoto && !_asFile)
- : false;
- });
- _controls->add(object_ptr<Ui::SlideWrap<Ui::FlatLabel>>(
- this,
- object_ptr<Ui::FlatLabel>(
- this,
- tr::lng_edit_photo_editor_hint(tr::now),
- st::editMediaHintLabel),
- st::editMediaLabelMargins)
- )->toggleOn(std::move(hintLabelToggleOn), anim::type::instant);
- _controls->add(object_ptr<Ui::SlideWrap<Ui::Checkbox>>(
- this,
- object_ptr<Ui::Checkbox>(
- this,
- tr::lng_send_compressed_one(tr::now),
- true,
- st::defaultBoxCheckbox),
- st::editMediaCheckboxMargins)
- )->toggleOn(
- _previewRebuilds.events_starting_with({}) | rpl::map([=] {
- return _isPhoto
- && CanBeCompressed(_albumType)
- && !_preparedList.files.empty();
- }),
- anim::type::instant
- )->entity()->checkedChanges(
- ) | rpl::start_with_next([&](bool checked) {
- applyChanges();
- _asFile = !checked;
- rebuildPreview();
- }, _controls->lifetime());
- _controls->resizeToWidth(st::sendMediaPreviewSize);
- }
- void EditCaptionBox::setupEditEventHandler() {
- _editMediaClicks.events(
- ) | rpl::start_with_next([=] {
- ChooseReplacement(_controller, _albumType, crl::guard(this, [=](
- Ui::PreparedList &&list) {
- setPreparedList(std::move(list));
- }));
- }, lifetime());
- }
- void EditCaptionBox::setupPhotoEditorEventHandler() {
- const auto openedOnce = lifetime().make_state<bool>(false);
- _photoEditorOpens.events(
- ) | rpl::start_with_next([=, controller = _controller] {
- if (_preparedList.files.empty()
- && (!_photoMedia
- || !_photoMedia->image(Data::PhotoSize::Large))) {
- return;
- } else if (!*openedOnce) {
- *openedOnce = true;
- controller->session().settings().incrementPhotoEditorHintShown();
- controller->session().saveSettings();
- }
- if (!_error.isEmpty()) {
- _error = QString();
- update();
- }
- if (!_preparedList.files.empty()) {
- Editor::OpenWithPreparedFile(
- this,
- controller->uiShow(),
- &_preparedList.files.front(),
- st::sendMediaPreviewSize,
- [=](bool ok) { if (ok) rebuildPreview(); });
- } else {
- EditPhotoImage(_controller, _photoMedia, hasSpoiler(), [=](
- Ui::PreparedList &&list) {
- setPreparedList(std::move(list));
- });
- }
- }, lifetime());
- }
- void EditCaptionBox::setupDragArea() {
- auto enterFilter = [=](not_null<const QMimeData*> data) {
- return !_isAllowedEditMedia
- ? false
- : Storage::ValidateEditMediaDragData(data, _albumType);
- };
- // Avoid both drag areas appearing at one time.
- auto computeState = [=](const QMimeData *data) {
- using DragState = Storage::MimeDataState;
- const auto state = Storage::ComputeMimeDataState(data);
- return (state == DragState::PhotoFiles || state == DragState::Image)
- ? (_asFile ? DragState::Files : DragState::Image)
- : state;
- };
- const auto areas = DragArea::SetupDragAreaToContainer(
- this,
- std::move(enterFilter),
- [=](bool f) { _field->setAcceptDrops(f); },
- nullptr,
- std::move(computeState));
- const auto droppedCallback = [=](bool compress) {
- return [=](const QMimeData *data) {
- fileFromClipboard(data);
- Window::ActivateWindow(_controller);
- };
- };
- areas.document->setDroppedCallback(droppedCallback(false));
- areas.photo->setDroppedCallback(droppedCallback(true));
- }
- void EditCaptionBox::setupEmojiPanel() {
- const auto container = getDelegate()->outerContainer();
- using Selector = ChatHelpers::TabbedSelector;
- _emojiPanel = base::make_unique_q<ChatHelpers::TabbedPanel>(
- container,
- _controller,
- object_ptr<Selector>(
- nullptr,
- _controller->uiShow(),
- Window::GifPauseReason::Layer,
- Selector::Mode::EmojiOnly));
- _emojiPanel->setDesiredHeightValues(
- 1.,
- st::emojiPanMinHeight / 2,
- st::emojiPanMinHeight);
- _emojiPanel->hide();
- _emojiPanel->selector()->setCurrentPeer(_historyItem->history()->peer);
- _emojiPanel->selector()->emojiChosen(
- ) | rpl::start_with_next([=](ChatHelpers::EmojiChosen data) {
- Ui::InsertEmojiAtCursor(_field->textCursor(), data.emoji);
- }, lifetime());
- _emojiPanel->selector()->customEmojiChosen(
- ) | rpl::start_with_next([=](ChatHelpers::FileChosen data) {
- const auto info = data.document->sticker();
- if (info
- && info->setType == Data::StickersType::Emoji
- && !_controller->session().premium()) {
- ShowPremiumPreviewBox(
- _controller,
- PremiumFeature::AnimatedEmoji);
- } else {
- Data::InsertCustomEmoji(_field.get(), data.document);
- }
- }, lifetime());
- const auto filterCallback = [=](not_null<QEvent*> event) {
- emojiFilterForGeometry(event);
- return base::EventFilterResult::Continue;
- };
- _emojiFilter.reset(base::install_event_filter(container, filterCallback));
- _emojiToggle->installEventFilter(_emojiPanel);
- _emojiToggle->addClickHandler([=] {
- _emojiPanel->toggleAnimated();
- });
- }
- void EditCaptionBox::emojiFilterForGeometry(not_null<QEvent*> event) {
- const auto type = event->type();
- if (type == QEvent::Move || type == QEvent::Resize) {
- // updateEmojiPanelGeometry uses not only container geometry, but
- // also container children geometries that will be updated later.
- crl::on_main(this, [=] { updateEmojiPanelGeometry(); });
- }
- }
- void EditCaptionBox::updateEmojiPanelGeometry() {
- const auto parent = _emojiPanel->parentWidget();
- const auto global = _emojiToggle->mapToGlobal({ 0, 0 });
- const auto local = parent->mapFromGlobal(global);
- _emojiPanel->moveBottomRight(
- local.y(),
- local.x() + _emojiToggle->width() * 3);
- }
- bool EditCaptionBox::fileFromClipboard(not_null<const QMimeData*> data) {
- const auto premium = _controller->session().premium();
- return setPreparedList(ListFromMimeData(data, premium));
- }
- bool EditCaptionBox::setPreparedList(Ui::PreparedList &&list) {
- if (!_isAllowedEditMedia) {
- return false;
- }
- using Error = Ui::PreparedList::Error;
- if (list.error != Error::None || list.files.empty()) {
- return false;
- }
- auto file = &list.files.front();
- const auto invalidForAlbum = (_albumType != Ui::AlbumType::None)
- && !file->canBeInAlbumType(_albumType);
- if (_albumType == Ui::AlbumType::PhotoVideo) {
- using Video = Ui::PreparedFileInformation::Video;
- if (const auto video = std::get_if<Video>(
- &file->information->media)) {
- video->isGifv = false;
- }
- }
- if (invalidForAlbum) {
- showToast(tr::lng_edit_media_album_error(tr::now));
- return false;
- }
- const auto wasSpoiler = hasSpoiler();
- _preparedList = std::move(list);
- _preparedList.files.front().spoiler = wasSpoiler;
- setCloseByOutsideClick(false);
- rebuildPreview();
- return true;
- }
- bool EditCaptionBox::hasSpoiler() const {
- return _mediaEditManager.spoilered();
- }
- void EditCaptionBox::captionResized() {
- updateBoxSize();
- resizeEvent(0);
- updateEmojiPanelGeometry();
- update();
- }
- void EditCaptionBox::updateBoxSize() {
- auto footerHeight = 0;
- footerHeight += st::normalFont->height + errorTopSkip();
- if (_field) {
- footerHeight += st::boxPhotoCaptionSkip + _field->height();
- }
- if (_controls && !_controls->isHidden()) {
- footerHeight += _controls->heightNoMargins();
- }
- _footerHeight = footerHeight;
- }
- int EditCaptionBox::errorTopSkip() const {
- return (st::defaultBox.buttonPadding.top() / 2);
- }
- void EditCaptionBox::paintEvent(QPaintEvent *e) {
- BoxContent::paintEvent(e);
- Painter p(this);
- if (!_error.isEmpty()) {
- p.setFont(st::normalFont);
- p.setPen(st::boxTextFgError);
- p.drawTextLeft(
- _field->x(),
- _field->y() + _field->height() + errorTopSkip(),
- width(),
- _error);
- }
- }
- void EditCaptionBox::resizeEvent(QResizeEvent *e) {
- BoxContent::resizeEvent(e);
- const auto errorHeight = st::normalFont->height + errorTopSkip();
- auto bottom = height();
- {
- const auto resultScrollHeight = bottom
- - _field->height()
- - st::boxPhotoCaptionSkip
- - (_controls->isHidden() ? 0 : _controls->heightNoMargins())
- - st::boxPhotoPadding.top()
- - errorHeight;
- const auto minThumbH = st::sendBoxAlbumGroupSize.height()
- + st::sendBoxAlbumGroupSkipTop * 2;
- const auto diff = resultScrollHeight - minThumbH;
- if (diff < 0) {
- bottom -= diff;
- }
- }
- bottom -= errorHeight;
- _field->resize(st::sendMediaPreviewSize, _field->height());
- _field->moveToLeft(
- st::boxPhotoPadding.left(),
- bottom - _field->height());
- bottom -= st::boxPhotoCaptionSkip + _field->height();
- _emojiToggle->moveToLeft(
- (st::boxPhotoPadding.left()
- + st::sendMediaPreviewSize
- - _emojiToggle->width()),
- _field->y() + st::boxAttachEmojiTop);
- _emojiToggle->update();
- if (!_controls->isHidden()) {
- _controls->resizeToWidth(width());
- _controls->moveToLeft(
- st::boxPhotoPadding.left(),
- bottom - _controls->heightNoMargins());
- bottom -= _controls->heightNoMargins();
- }
- _scroll->resize(width(), bottom - st::boxPhotoPadding.top());
- _scroll->move(0, st::boxPhotoPadding.top());
- if (_content) {
- _content->resize(_scroll->width(), _content->height());
- }
- }
- void EditCaptionBox::setInnerFocus() {
- _field->setFocusFast();
- }
- bool EditCaptionBox::validateLength(const QString &text) const {
- const auto session = &_controller->session();
- const auto limit = Data::PremiumLimits(session).captionLengthCurrent();
- const auto remove = int(text.size()) - limit;
- if (remove <= 0) {
- return true;
- }
- _controller->show(
- Box(CaptionLimitReachedBox, session, remove, nullptr));
- return false;
- }
- void EditCaptionBox::applyChanges() {
- if (!_preparedList.files.empty()) {
- _preparedList.files.front().spoiler = _mediaEditManager.spoilered();
- }
- }
- void EditCaptionBox::save() {
- if (_saveRequestId) {
- return;
- }
- const auto item = _controller->session().data().message(
- _historyItem->fullId());
- if (!item) {
- _error = tr::lng_edit_deleted(tr::now);
- update();
- return;
- }
- const auto textWithTags = _field->getTextWithAppliedMarkdown();
- if (!validateLength(textWithTags.text)) {
- return;
- }
- const auto sending = TextWithEntities{
- textWithTags.text,
- TextUtilities::ConvertTextTagsToEntities(textWithTags.tags)
- };
- auto options = Api::SendOptions();
- options.scheduled = item->isScheduled() ? item->date() : 0;
- options.shortcutId = item->shortcutId();
- options.invertCaption = _mediaEditManager.invertCaption();
- if (!_preparedList.files.empty()) {
- if ((_albumType != Ui::AlbumType::None)
- && !_preparedList.files.front().canBeInAlbumType(
- _albumType)) {
- _error = tr::lng_edit_media_album_error(tr::now);
- update();
- return;
- }
- auto action = Api::SendAction(item->history(), options);
- action.replaceMediaOf = item->fullId().msg;
- Storage::ApplyModifications(_preparedList);
- if (!_preparedList.files.empty()) {
- _preparedList.files.front().spoiler = false;
- applyChanges();
- }
- _controller->session().api().editMedia(
- std::move(_preparedList),
- (_isPhoto && !_asFile && CanBeCompressed(_albumType))
- ? SendMediaType::Photo
- : SendMediaType::File,
- _field->getTextWithAppliedMarkdown(),
- action);
- closeAfterSave();
- return;
- }
- const auto done = crl::guard(this, [=] {
- _saveRequestId = 0;
- closeAfterSave();
- });
- const auto fail = crl::guard(this, [=](const QString &error) {
- _saveRequestId = 0;
- if (ranges::contains(Api::kDefaultEditMessagesErrors, error)) {
- _error = tr::lng_edit_error(tr::now);
- update();
- } else if (error == u"MESSAGE_NOT_MODIFIED"_q) {
- closeAfterSave();
- } else if (error == u"MESSAGE_EMPTY"_q) {
- _field->setFocus();
- _field->showError();
- update();
- } else {
- _error = tr::lng_edit_error(tr::now);
- update();
- }
- });
- lifetime().add([=] {
- if (_saveRequestId) {
- auto &session = _controller->session();
- session.api().request(base::take(_saveRequestId)).cancel();
- }
- });
- _saveRequestId = Api::EditCaption(item, sending, options, done, fail);
- }
- void EditCaptionBox::closeAfterSave() {
- const auto weak = MakeWeak(this);
- if (_saved) {
- _saved();
- }
- if (weak) {
- closeBox();
- }
- }
- void EditCaptionBox::keyPressEvent(QKeyEvent *e) {
- const auto ctrl = e->modifiers().testFlag(Qt::ControlModifier);
- if ((e->key() == Qt::Key_E) && ctrl) {
- _photoEditorOpens.fire({});
- } else if ((e->key() == Qt::Key_O) && ctrl) {
- _editMediaClicks.fire({});
- } else {
- e->ignore();
- }
- }
|