| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957 |
- /*
- 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/send_files_box.h"
- #include "lang/lang_keys.h"
- #include "storage/localstorage.h"
- #include "storage/storage_media_prepare.h"
- #include "iv/iv_instance.h"
- #include "mainwidget.h"
- #include "main/main_app_config.h"
- #include "main/main_session.h"
- #include "main/main_session_settings.h"
- #include "mtproto/mtproto_config.h"
- #include "chat_helpers/message_field.h"
- #include "menu/menu_send.h"
- #include "chat_helpers/emoji_suggestions_widget.h"
- #include "chat_helpers/field_autocomplete.h"
- #include "chat_helpers/tabbed_panel.h"
- #include "chat_helpers/tabbed_selector.h"
- #include "editor/photo_editor_layer_widget.h"
- #include "history/history_drag_area.h"
- #include "history/view/controls/history_view_characters_limit.h"
- #include "history/view/history_view_schedule_box.h"
- #include "core/mime_type.h"
- #include "core/ui_integration.h"
- #include "base/event_filter.h"
- #include "base/call_delayed.h"
- #include "boxes/premium_limits_box.h"
- #include "boxes/premium_preview_box.h"
- #include "boxes/send_credits_box.h"
- #include "ui/effects/scroll_content_shadow.h"
- #include "ui/widgets/fields/number_input.h"
- #include "ui/widgets/checkbox.h"
- #include "ui/widgets/scroll_area.h"
- #include "ui/widgets/popup_menu.h"
- #include "ui/chat/attach/attach_album_preview.h"
- #include "ui/chat/attach/attach_single_file_preview.h"
- #include "ui/chat/attach/attach_single_media_preview.h"
- #include "ui/grouped_layout.h"
- #include "ui/text/text_utilities.h"
- #include "ui/toast/toast.h"
- #include "ui/controls/emoji_button.h"
- #include "ui/painter.h"
- #include "ui/vertical_list.h"
- #include "ui/ui_utility.h"
- #include "lottie/lottie_single_player.h"
- #include "data/data_channel.h"
- #include "data/data_document.h"
- #include "data/data_user.h"
- #include "data/data_peer_values.h" // Data::AmPremiumValue.
- #include "data/data_premium_limits.h"
- #include "data/stickers/data_stickers.h"
- #include "data/stickers/data_custom_emoji.h"
- #include "window/window_session_controller.h"
- #include "core/application.h"
- #include "core/core_settings.h"
- #include "styles/style_boxes.h"
- #include "styles/style_chat_helpers.h"
- #include "styles/style_layers.h"
- #include <QtCore/QMimeData>
- namespace {
- constexpr auto kMaxMessageLength = 4096;
- using Ui::SendFilesWay;
- [[nodiscard]] inline bool CanAddUrls(const QList<QUrl> &urls) {
- return !urls.isEmpty() && ranges::all_of(urls, &QUrl::isLocalFile);
- }
- [[nodiscard]] bool CanAddFiles(not_null<const QMimeData*> data) {
- return data->hasImage() || CanAddUrls(Core::ReadMimeUrls(data));
- }
- void FileDialogCallback(
- FileDialog::OpenResult &&result,
- Fn<bool(const Ui::PreparedList&)> checkResult,
- Fn<void(Ui::PreparedList)> callback,
- bool premium,
- std::shared_ptr<Ui::Show> show) {
- auto showError = [=](tr::phrase<> text) {
- show->showToast(text(tr::now));
- };
- auto list = Storage::PreparedFileFromFilesDialog(
- std::move(result),
- checkResult,
- showError,
- st::sendMediaPreviewSize,
- premium);
- if (!list) {
- return;
- }
- callback(std::move(*list));
- }
- rpl::producer<QString> FieldPlaceholder(
- const Ui::PreparedList &list,
- SendFilesWay way) {
- return list.canAddCaption(
- way.groupFiles() && way.sendImagesAsPhotos(),
- way.sendImagesAsPhotos())
- ? tr::lng_photo_caption()
- : tr::lng_photos_comment();
- }
- void EditPriceBox(
- not_null<Ui::GenericBox*> box,
- not_null<Main::Session*> session,
- uint64 price,
- Fn<void(uint64)> apply) {
- box->setTitle(tr::lng_paid_title());
- AddSubsectionTitle(
- box->verticalLayout(),
- tr::lng_paid_enter_cost(),
- (st::boxRowPadding - QMargins(
- st::defaultSubsectionTitlePadding.left(),
- 0,
- st::defaultSubsectionTitlePadding.right(),
- 0)));
- const auto limit = session->appConfig().get<int>(
- u"stars_paid_post_amount_max"_q,
- 10'000);
- const auto wrap = box->addRow(object_ptr<Ui::FixedHeightWidget>(
- box,
- st::editTagField.heightMin));
- auto owned = object_ptr<Ui::NumberInput>(
- wrap,
- st::editTagField,
- tr::lng_paid_cost_placeholder(),
- price ? QString::number(price) : QString(),
- limit);
- const auto field = owned.data();
- wrap->widthValue() | rpl::start_with_next([=](int width) {
- field->move(0, 0);
- field->resize(width, field->height());
- wrap->resize(width, field->height());
- }, wrap->lifetime());
- field->selectAll();
- box->setFocusCallback([=] {
- field->setFocusFast();
- });
- const auto about = box->addRow(
- object_ptr<Ui::FlatLabel>(
- box,
- tr::lng_paid_about(
- lt_link,
- tr::lng_paid_about_link() | Ui::Text::ToLink(),
- Ui::Text::WithEntities),
- st::paidAmountAbout),
- st::boxRowPadding + QMargins(0, st::sendMediaRowSkip, 0, 0));
- about->setClickHandlerFilter([=](const auto &...) {
- Core::App().iv().openWithIvPreferred(
- session,
- tr::lng_paid_about_link_url(tr::now));
- return false;
- });
- field->paintRequest() | rpl::start_with_next([=](QRect clip) {
- auto p = QPainter(field);
- st::paidStarIcon.paint(p, 0, st::paidStarIconTop, field->width());
- }, field->lifetime());
- const auto save = [=] {
- const auto now = field->getLastText().toULongLong();
- if (now > limit) {
- field->showError();
- return;
- }
- const auto weak = Ui::MakeWeak(box);
- apply(now);
- if (const auto strong = weak.data()) {
- strong->closeBox();
- }
- };
- QObject::connect(field, &Ui::NumberInput::submitted, box, save);
- box->addButton(tr::lng_settings_save(), save);
- box->addButton(tr::lng_cancel(), [=] {
- box->closeBox();
- });
- }
- } // namespace
- SendFilesLimits DefaultLimitsForPeer(not_null<PeerData*> peer) {
- using Flag = SendFilesAllow;
- using Restriction = ChatRestriction;
- const auto allowByRestriction = [&](Restriction check, Flag allow) {
- return Data::RestrictionError(peer, check) ? Flag() : allow;
- };
- return Flag()
- | (peer->slowmodeApplied() ? Flag::OnlyOne : Flag())
- | (Data::AllowEmojiWithoutPremium(peer)
- ? Flag::EmojiWithoutPremium
- : Flag())
- | allowByRestriction(Restriction::SendPhotos, Flag::Photos)
- | allowByRestriction(Restriction::SendVideos, Flag::Videos)
- | allowByRestriction(Restriction::SendMusic, Flag::Music)
- | allowByRestriction(Restriction::SendFiles, Flag::Files)
- | allowByRestriction(Restriction::SendStickers, Flag::Stickers)
- | allowByRestriction(Restriction::SendGifs, Flag::Gifs)
- | allowByRestriction(Restriction::SendOther, Flag::Texts);
- }
- SendFilesCheck DefaultCheckForPeer(
- not_null<Window::SessionController*> controller,
- not_null<PeerData*> peer) {
- return DefaultCheckForPeer(controller->uiShow(), peer);
- }
- SendFilesCheck DefaultCheckForPeer(
- std::shared_ptr<ChatHelpers::Show> show,
- not_null<PeerData*> peer) {
- return [=](
- const Ui::PreparedFile &file,
- bool compress,
- bool silent) {
- const auto error = Data::FileRestrictionError(peer, file, compress);
- if (error && !silent) {
- Data::ShowSendErrorToast(show, peer, error);
- }
- return !error.has_value();
- };
- }
- SendFilesBox::Block::Block(
- not_null<QWidget*> parent,
- const style::ComposeControls &st,
- not_null<std::vector<Ui::PreparedFile>*> items,
- int from,
- int till,
- Fn<bool()> gifPaused,
- SendFilesWay way,
- Fn<bool(const Ui::PreparedFile &, Ui::AttachActionType)> actionAllowed)
- : _items(items)
- , _from(from)
- , _till(till) {
- Expects(from >= 0);
- Expects(till > from);
- Expects(till <= items->size());
- const auto count = till - from;
- const auto my = gsl::make_span(*items).subspan(from, count);
- const auto &first = my.front();
- _isAlbum = (my.size() > 1);
- if (_isAlbum) {
- const auto preview = Ui::CreateChild<Ui::AlbumPreview>(
- parent.get(),
- st,
- my,
- way,
- [=](int index, Ui::AttachActionType type) {
- return actionAllowed((*_items)[from + index], type);
- });
- _preview.reset(preview);
- } else {
- const auto media = Ui::SingleMediaPreview::Create(
- parent,
- st,
- gifPaused,
- first,
- [=](Ui::AttachActionType type) {
- return actionAllowed((*_items)[from], type);
- });
- if (media) {
- _isSingleMedia = true;
- _preview.reset(media);
- } else {
- _preview.reset(Ui::CreateChild<Ui::SingleFilePreview>(
- parent.get(),
- st,
- first));
- }
- }
- _preview->show();
- }
- int SendFilesBox::Block::fromIndex() const {
- return _from;
- }
- int SendFilesBox::Block::tillIndex() const {
- return _till;
- }
- object_ptr<Ui::RpWidget> SendFilesBox::Block::takeWidget() {
- return object_ptr<Ui::RpWidget>::fromRaw(_preview.get());
- }
- rpl::producer<int> SendFilesBox::Block::itemDeleteRequest() const {
- using namespace rpl::mappers;
- const auto preview = _preview.get();
- const auto from = _from;
- if (_isAlbum) {
- const auto album = static_cast<Ui::AlbumPreview*>(_preview.get());
- return album->thumbDeleted() | rpl::map(_1 + from);
- } else if (_isSingleMedia) {
- const auto media = static_cast<Ui::SingleMediaPreview*>(preview);
- return media->deleteRequests() | rpl::map([from] { return from; });
- } else {
- const auto single = static_cast<Ui::SingleFilePreview*>(preview);
- return single->deleteRequests() | rpl::map([from] { return from; });
- }
- }
- rpl::producer<int> SendFilesBox::Block::itemReplaceRequest() const {
- using namespace rpl::mappers;
- const auto preview = _preview.get();
- const auto from = _from;
- if (_isAlbum) {
- const auto album = static_cast<Ui::AlbumPreview*>(preview);
- return album->thumbChanged() | rpl::map(_1 + from);
- } else if (_isSingleMedia) {
- const auto media = static_cast<Ui::SingleMediaPreview*>(preview);
- return media->editRequests() | rpl::map([from] { return from; });
- } else {
- const auto single = static_cast<Ui::SingleFilePreview*>(preview);
- return single->editRequests() | rpl::map([from] { return from; });
- }
- }
- rpl::producer<int> SendFilesBox::Block::itemModifyRequest() const {
- using namespace rpl::mappers;
- const auto preview = _preview.get();
- const auto from = _from;
- if (_isAlbum) {
- const auto album = static_cast<Ui::AlbumPreview*>(preview);
- return album->thumbModified() | rpl::map(_1 + from);
- } else if (_isSingleMedia) {
- const auto media = static_cast<Ui::SingleMediaPreview*>(preview);
- return media->modifyRequests() | rpl::map_to(from);
- } else {
- return rpl::never<int>();
- }
- }
- rpl::producer<int> SendFilesBox::Block::itemEditCoverRequest() const {
- using namespace rpl::mappers;
- const auto preview = _preview.get();
- const auto from = _from;
- if (_isAlbum) {
- const auto album = static_cast<Ui::AlbumPreview*>(preview);
- return album->thumbEditCoverRequested() | rpl::map(_1 + from);
- } else if (_isSingleMedia) {
- const auto media = static_cast<Ui::SingleMediaPreview*>(preview);
- return media->editCoverRequests() | rpl::map_to(from);
- } else {
- return rpl::never<int>();
- }
- }
- rpl::producer<int> SendFilesBox::Block::itemClearCoverRequest() const {
- using namespace rpl::mappers;
- const auto preview = _preview.get();
- const auto from = _from;
- if (_isAlbum) {
- const auto album = static_cast<Ui::AlbumPreview*>(preview);
- return album->thumbClearCoverRequested() | rpl::map(_1 + from);
- } else if (_isSingleMedia) {
- const auto media = static_cast<Ui::SingleMediaPreview*>(preview);
- return media->clearCoverRequests() | rpl::map_to(from);
- } else {
- return rpl::never<int>();
- }
- }
- rpl::producer<> SendFilesBox::Block::orderUpdated() const {
- if (_isAlbum) {
- const auto album = static_cast<Ui::AlbumPreview*>(_preview.get());
- return album->orderUpdated();
- }
- return rpl::never<>();
- }
- void SendFilesBox::Block::setSendWay(Ui::SendFilesWay way) {
- if (!_isAlbum) {
- if (_isSingleMedia) {
- const auto media = static_cast<Ui::SingleMediaPreview*>(
- _preview.get());
- media->setSendWay(way);
- }
- return;
- }
- applyChanges();
- const auto album = static_cast<Ui::AlbumPreview*>(_preview.get());
- album->setSendWay(way);
- }
- void SendFilesBox::Block::toggleSpoilers(bool enabled) {
- if (_isAlbum) {
- const auto album = static_cast<Ui::AlbumPreview*>(_preview.get());
- album->toggleSpoilers(enabled);
- } else if (_isSingleMedia) {
- const auto media = static_cast<Ui::SingleMediaPreview*>(
- _preview.get());
- media->setSpoiler(enabled);
- }
- }
- void SendFilesBox::Block::applyChanges() {
- if (!_isAlbum) {
- if (_isSingleMedia) {
- const auto media = static_cast<Ui::SingleMediaPreview*>(
- _preview.get());
- if (media->canHaveSpoiler()) {
- (*_items)[_from].spoiler = media->hasSpoiler();
- }
- }
- return;
- }
- const auto album = static_cast<Ui::AlbumPreview*>(_preview.get());
- const auto order = album->takeOrder();
- const auto guard = gsl::finally([&] {
- const auto spoilered = album->collectSpoileredIndices();
- for (auto i = 0, count = int(order.size()); i != count; ++i) {
- if (album->canHaveSpoiler(i)) {
- (*_items)[_from + i].spoiler = spoilered.contains(i);
- }
- }
- });
- const auto isIdentity = [&] {
- for (auto i = 0, count = int(order.size()); i != count; ++i) {
- if (order[i] != i) {
- return false;
- }
- }
- return true;
- }();
- if (isIdentity) {
- return;
- }
- auto elements = std::vector<Ui::PreparedFile>();
- elements.reserve(order.size());
- for (const auto index : order) {
- elements.push_back(std::move((*_items)[_from + index]));
- }
- for (auto i = 0, count = int(order.size()); i != count; ++i) {
- (*_items)[_from + i] = std::move(elements[i]);
- }
- }
- QImage SendFilesBox::Block::generatePriceTagBackground() const {
- const auto preview = _preview.get();
- if (_isAlbum) {
- const auto album = static_cast<Ui::AlbumPreview*>(preview);
- return album->generatePriceTagBackground();
- } else if (_isSingleMedia) {
- const auto media = static_cast<Ui::SingleMediaPreview*>(preview);
- return media->generatePriceTagBackground();
- }
- return QImage();
- }
- SendFilesBox::SendFilesBox(
- QWidget*,
- not_null<Window::SessionController*> controller,
- Ui::PreparedList &&list,
- const TextWithTags &caption,
- not_null<PeerData*> toPeer,
- Api::SendType sendType,
- SendMenu::Details sendMenuDetails)
- : SendFilesBox(nullptr, {
- .show = controller->uiShow(),
- .list = std::move(list),
- .caption = caption,
- .captionToPeer = toPeer,
- .limits = DefaultLimitsForPeer(toPeer),
- .check = DefaultCheckForPeer(controller, toPeer),
- .sendType = sendType,
- .sendMenuDetails = [=] { return sendMenuDetails; },
- }) {
- }
- SendFilesBox::SendFilesBox(QWidget*, SendFilesBoxDescriptor &&descriptor)
- : _show(std::move(descriptor.show))
- , _st(descriptor.stOverride
- ? *descriptor.stOverride
- : st::defaultComposeControls)
- , _sendType(descriptor.sendType)
- , _titleHeight(st::boxTitleHeight)
- , _list(std::move(descriptor.list))
- , _limits(descriptor.limits)
- , _sendMenuDetails(prepareSendMenuDetails(descriptor))
- , _sendMenuCallback(prepareSendMenuCallback())
- , _captionToPeer(descriptor.captionToPeer)
- , _check(std::move(descriptor.check))
- , _confirmedCallback(std::move(descriptor.confirmed))
- , _cancelledCallback(std::move(descriptor.cancelled))
- , _caption(this, _st.files.caption, Ui::InputField::Mode::MultiLine)
- , _prefilledCaptionText(std::move(descriptor.caption))
- , _scroll(this, st::boxScroll)
- , _inner(
- _scroll->setOwnedWidget(
- object_ptr<Ui::VerticalLayout>(_scroll.data()))) {
- enqueueNextPrepare();
- }
- Fn<SendMenu::Details()> SendFilesBox::prepareSendMenuDetails(
- const SendFilesBoxDescriptor &descriptor) {
- auto initial = descriptor.sendMenuDetails;
- return crl::guard(this, [=] {
- auto result = initial ? initial() : SendMenu::Details();
- result.spoiler = !hasSpoilerMenu()
- ? SendMenu::SpoilerState::None
- : allWithSpoilers()
- ? SendMenu::SpoilerState::Enabled
- : SendMenu::SpoilerState::Possible;
- const auto way = _sendWay.current();
- const auto canMoveCaption = _list.canMoveCaption(
- way.groupFiles() && way.sendImagesAsPhotos(),
- way.sendImagesAsPhotos()
- ) && _caption && HasSendText(_caption);
- result.caption = !canMoveCaption
- ? SendMenu::CaptionState::None
- : _invertCaption
- ? SendMenu::CaptionState::Above
- : SendMenu::CaptionState::Below;
- result.price = canChangePrice()
- ? _price.current()
- : std::optional<uint64>();
- return result;
- });
- }
- auto SendFilesBox::prepareSendMenuCallback()
- -> Fn<void(MenuAction, MenuDetails)> {
- return crl::guard(this, [=](MenuAction action, MenuDetails details) {
- using Type = SendMenu::ActionType;
- switch (action.type) {
- case Type::CaptionDown: _invertCaption = false; break;
- case Type::CaptionUp: _invertCaption = true; break;
- case Type::SpoilerOn: toggleSpoilers(true); break;
- case Type::SpoilerOff: toggleSpoilers(false); break;
- case Type::ChangePrice: changePrice(); break;
- default:
- SendMenu::DefaultCallback(
- _show,
- sendCallback())(
- action,
- details);
- break;
- }
- });
- }
- void SendFilesBox::initPreview() {
- using namespace rpl::mappers;
- refreshControls(true);
- updateBoxSize();
- _dimensionsLifetime.destroy();
- _inner->resizeToWidth(st::boxWideWidth);
- rpl::combine(
- _inner->heightValue(),
- _footerHeight.value(),
- _titleHeight.value(),
- _1 + _2 + _3
- ) | rpl::start_with_next([=](int height) {
- setDimensions(
- st::boxWideWidth,
- std::min(st::sendMediaPreviewHeightMax, height),
- true);
- }, _dimensionsLifetime);
- }
- void SendFilesBox::enqueueNextPrepare() {
- if (_preparing) {
- return;
- }
- while (!_list.filesToProcess.empty()
- && _list.filesToProcess.front().information) {
- auto file = std::move(_list.filesToProcess.front());
- _list.filesToProcess.pop_front();
- addFile(std::move(file));
- }
- if (_list.filesToProcess.empty()) {
- return;
- }
- auto file = std::move(_list.filesToProcess.front());
- _list.filesToProcess.pop_front();
- const auto weak = Ui::MakeWeak(this);
- _preparing = true;
- const auto sideLimit = PhotoSideLimit(); // Get on main thread.
- crl::async([weak, sideLimit, file = std::move(file)]() mutable {
- Storage::PrepareDetails(file, st::sendMediaPreviewSize, sideLimit);
- crl::on_main([weak, file = std::move(file)]() mutable {
- if (weak) {
- weak->addPreparedAsyncFile(std::move(file));
- }
- });
- });
- }
- void SendFilesBox::prepare() {
- initSendWay();
- setupCaption();
- setupSendWayControls();
- preparePreview();
- initPreview();
- SetupShadowsToScrollContent(this, _scroll, _inner->heightValue());
- setCloseByOutsideClick(false);
- boxClosing() | rpl::start_with_next([=] {
- if (!_confirmed && _cancelledCallback) {
- _cancelledCallback();
- }
- }, lifetime());
- setupDragArea();
- }
- void SendFilesBox::setupDragArea() {
- // 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)
- ? (_sendWay.current().sendImagesAsPhotos()
- ? DragState::Image
- : DragState::Files)
- : state;
- };
- const auto areas = DragArea::SetupDragAreaToContainer(
- this,
- CanAddFiles,
- [=](bool f) { _caption->setAcceptDrops(f); },
- [=] { updateControlsGeometry(); },
- std::move(computeState));
- const auto droppedCallback = [=](bool compress) {
- return [=](const QMimeData *data) {
- addFiles(data);
- _show->activate();
- };
- };
- areas.document->setDroppedCallback(droppedCallback(false));
- areas.photo->setDroppedCallback(droppedCallback(true));
- }
- void SendFilesBox::refreshAllAfterChanges(int fromItem, Fn<void()> perform) {
- auto fromBlock = 0;
- for (auto count = int(_blocks.size()); fromBlock != count; ++fromBlock) {
- if (_blocks[fromBlock].tillIndex() >= fromItem) {
- break;
- }
- }
- for (auto index = fromBlock; index < _blocks.size(); ++index) {
- _blocks[index].applyChanges();
- }
- if (perform) {
- perform();
- }
- generatePreviewFrom(fromBlock);
- {
- auto sendWay = _sendWay.current();
- sendWay.setHasCompressedStickers(_list.hasSticker());
- if (_limits & SendFilesAllow::OnlyOne) {
- if (_list.files.size() > 1) {
- sendWay.setGroupFiles(true);
- }
- }
- _sendWay = sendWay;
- }
- _inner->resizeToWidth(st::boxWideWidth);
- refreshControls();
- captionResized();
- }
- void SendFilesBox::openDialogToAddFileToAlbum() {
- const auto show = uiShow();
- const auto checkResult = [=](const Ui::PreparedList &list) {
- if (!(_limits & SendFilesAllow::OnlyOne)) {
- return true;
- } else if (!_list.canBeSentInSlowmodeWith(list)) {
- showToast(tr::lng_slowmode_no_many(tr::now));
- return false;
- }
- return true;
- };
- const auto callback = [=](FileDialog::OpenResult &&result) {
- const auto premium = _show->session().premium();
- FileDialogCallback(
- std::move(result),
- checkResult,
- [=](Ui::PreparedList list) { addFiles(std::move(list)); },
- premium,
- show);
- };
- FileDialog::GetOpenPaths(
- this,
- tr::lng_choose_file(tr::now),
- FileDialog::AllOrImagesFilter(),
- crl::guard(this, callback));
- }
- void SendFilesBox::refreshMessagesCount() {
- const auto way = _sendWay.current();
- const auto withCaption = _list.canAddCaption(
- way.groupFiles() && way.sendImagesAsPhotos(),
- way.sendImagesAsPhotos());
- const auto withComment = !withCaption
- && _caption
- && !_caption->isHidden()
- && !_caption->getTextWithTags().text.isEmpty();
- _messagesCount = _list.files.size() + (withComment ? 1 : 0);
- }
- void SendFilesBox::refreshButtons() {
- clearButtons();
- _send = addButton(
- (_sendType == Api::SendType::Normal
- ? tr::lng_send_button()
- : tr::lng_create_group_next()),
- [=] { send({}); });
- refreshMessagesCount();
- const auto perMessage = _captionToPeer
- ? _captionToPeer->starsPerMessageChecked()
- : 0;
- if (perMessage > 0) {
- _send->setText(PaidSendButtonText(_messagesCount.value(
- ) | rpl::map(rpl::mappers::_1 * perMessage)));
- }
- if (_sendType == Api::SendType::Normal) {
- SendMenu::SetupMenuAndShortcuts(
- _send,
- _show,
- _sendMenuDetails,
- _sendMenuCallback);
- }
- addButton(tr::lng_cancel(), [=] { closeBox(); });
- _addFile = addLeftButton(
- tr::lng_stickers_featured_add(),
- base::fn_delayed(st::historyAttach.ripple.hideDuration, this, [=] {
- openDialogToAddFileToAlbum();
- }));
- addMenuButton();
- }
- bool SendFilesBox::hasSendMenu(const MenuDetails &details) const {
- return (details.type != SendMenu::Type::Disabled)
- || (details.spoiler != SendMenu::SpoilerState::None)
- || (details.caption != SendMenu::CaptionState::None);
- }
- bool SendFilesBox::hasSpoilerMenu() const {
- return !hasPrice()
- && _list.hasSpoilerMenu(_sendWay.current().sendImagesAsPhotos());
- }
- bool SendFilesBox::canChangePrice() const {
- const auto way = _sendWay.current();
- const auto broadcast = _captionToPeer
- ? _captionToPeer->asBroadcast()
- : nullptr;
- return broadcast
- && broadcast->canPostPaidMedia()
- && _list.canChangePrice(
- way.groupFiles() && way.sendImagesAsPhotos(),
- way.sendImagesAsPhotos());
- }
- void SendFilesBox::applyBlockChanges() {
- for (auto &block : _blocks) {
- block.applyChanges();
- }
- }
- bool SendFilesBox::allWithSpoilers() {
- applyBlockChanges();
- return ranges::all_of(_list.files, &Ui::PreparedFile::spoiler);
- }
- void SendFilesBox::toggleSpoilers(bool enabled) {
- for (auto &file : _list.files) {
- file.spoiler = enabled;
- }
- for (auto &block : _blocks) {
- block.toggleSpoilers(enabled);
- }
- }
- void SendFilesBox::changePrice() {
- const auto weak = Ui::MakeWeak(this);
- const auto session = &_show->session();
- const auto now = _price.current();
- _show->show(Box(EditPriceBox, session, now, [=](uint64 price) {
- if (weak && price != now) {
- _price = price;
- refreshPriceTag();
- }
- }));
- }
- bool SendFilesBox::hasPrice() const {
- return canChangePrice() && _price.current() > 0;
- }
- void SendFilesBox::refreshPriceTag() {
- const auto resetSpoilers = hasPrice() || _priceTag;
- if (resetSpoilers) {
- for (auto &file : _list.files) {
- file.spoiler = false;
- }
- for (auto &block : _blocks) {
- block.toggleSpoilers(hasPrice());
- }
- }
- if (!hasPrice()) {
- _priceTag = nullptr;
- _priceTagBg = QImage();
- } else if (!_priceTag) {
- _priceTag = std::make_unique<Ui::RpWidget>(_inner.data());
- const auto raw = _priceTag.get();
- raw->show();
- raw->paintRequest() | rpl::start_with_next([=] {
- if (_priceTagBg.isNull()) {
- _priceTagBg = preparePriceTagBg(raw->size());
- }
- QPainter(raw).drawImage(0, 0, _priceTagBg);
- }, raw->lifetime());
- const auto session = &_show->session();
- auto price = _price.value() | rpl::map([=](uint64 amount) {
- auto result = Ui::Text::Colorized(Ui::CreditsEmoji(session));
- result.append(Lang::FormatCountDecimal(amount));
- return result;
- });
- auto text = tr::lng_paid_price(
- lt_price,
- std::move(price),
- Ui::Text::WithEntities);
- const auto label = Ui::CreateChild<Ui::FlatLabel>(
- raw,
- QString(),
- st::paidTagLabel);
- std::move(text) | rpl::start_with_next([=](TextWithEntities &&text) {
- label->setMarkedText(text, Core::TextContext({
- .session = session,
- }));
- }, label->lifetime());
- label->show();
- label->sizeValue() | rpl::start_with_next([=](QSize size) {
- const auto inner = QRect(QPoint(), size);
- const auto rect = inner.marginsAdded(st::paidTagPadding);
- raw->resize(rect.size());
- label->move(-rect.topLeft());
- }, label->lifetime());
- _inner->sizeValue() | rpl::start_with_next([=](QSize size) {
- raw->move(
- (size.width() - raw->width()) / 2,
- (size.height() - raw->height()) / 2);
- }, raw->lifetime());
- } else {
- _priceTag->raise();
- _priceTag->update();
- _priceTagBg = QImage();
- }
- }
- QImage SendFilesBox::preparePriceTagBg(QSize size) const {
- const auto ratio = style::DevicePixelRatio();
- const auto outer = _blocks.empty()
- ? size
- : _inner->widgetAt(0)->geometry().size();
- auto bg = _blocks.empty()
- ? QImage()
- : _blocks.front().generatePriceTagBackground();
- if (bg.isNull()) {
- bg = QImage(ratio, ratio, QImage::Format_ARGB32_Premultiplied);
- bg.fill(Qt::black);
- }
- auto result = QImage(size * ratio, QImage::Format_ARGB32_Premultiplied);
- result.setDevicePixelRatio(ratio);
- result.fill(Qt::black);
- auto p = QPainter(&result);
- auto hq = PainterHighQualityEnabler(p);
- p.drawImage(
- QRect(
- (size.width() - outer.width()) / 2,
- (size.height() - outer.height()) / 2,
- outer.width(),
- outer.height()),
- bg);
- p.fillRect(QRect(QPoint(), size), st::msgDateImgBg);
- p.end();
- const auto radius = std::min(size.width(), size.height()) / 2;
- return Images::Round(std::move(result), Images::CornersMask(radius));
- }
- void SendFilesBox::addMenuButton() {
- const auto details = _sendMenuDetails();
- if (!hasSendMenu(details)) {
- return;
- }
- const auto top = addTopButton(_st.files.menu);
- top->setClickedCallback([=] {
- const auto &tabbed = _st.tabbed;
- _menu = base::make_unique_q<Ui::PopupMenu>(top, tabbed.menu);
- const auto position = QCursor::pos();
- SendMenu::FillSendMenu(
- _menu.get(),
- _show,
- _sendMenuDetails(),
- _sendMenuCallback,
- &_st.tabbed.icons,
- position);
- _menu->popup(position);
- return true;
- });
- }
- void SendFilesBox::initSendWay() {
- _sendWay = [&] {
- auto result = Core::App().settings().sendFilesWay();
- result.setHasCompressedStickers(_list.hasSticker());
- if ((_limits & SendFilesAllow::OnlyOne)
- && (_list.files.size() > 1)) {
- result.setGroupFiles(true);
- }
- if (_list.overrideSendImagesAsPhotos == false) {
- if (!(_limits & SendFilesAllow::OnlyOne)
- || !_list.hasSticker()) {
- result.setSendImagesAsPhotos(false);
- }
- return result;
- } else if (_list.overrideSendImagesAsPhotos == true) {
- result.setSendImagesAsPhotos(true);
- const auto silent = true;
- if (!checkWithWay(result, silent)) {
- result.setSendImagesAsPhotos(false);
- }
- return result;
- }
- const auto silent = true;
- if (!checkWithWay(result, silent)) {
- result.setSendImagesAsPhotos(!result.sendImagesAsPhotos());
- }
- return result;
- }();
- _sendWay.changes(
- ) | rpl::start_with_next([=](SendFilesWay value) {
- const auto hidden = [&] {
- return !_caption || _caption->isHidden();
- };
- const auto was = hidden();
- updateCaptionPlaceholder();
- updateEmojiPanelGeometry();
- for (auto &block : _blocks) {
- block.setSendWay(value);
- }
- refreshButtons();
- refreshPriceTag();
- if (was != hidden()) {
- updateBoxSize();
- updateControlsGeometry();
- }
- setInnerFocus();
- }, lifetime());
- }
- void SendFilesBox::updateCaptionPlaceholder() {
- if (!_caption) {
- return;
- }
- const auto way = _sendWay.current();
- if (!_list.canAddCaption(
- way.groupFiles() && way.sendImagesAsPhotos(),
- way.sendImagesAsPhotos())
- && ((_limits & SendFilesAllow::OnlyOne)
- || !(_limits & SendFilesAllow::Texts))) {
- _caption->hide();
- if (_emojiToggle) {
- _emojiToggle->hide();
- }
- } else {
- _caption->setPlaceholder(FieldPlaceholder(_list, way));
- _caption->show();
- if (_emojiToggle) {
- _emojiToggle->show();
- }
- }
- }
- void SendFilesBox::preparePreview() {
- generatePreviewFrom(0);
- }
- void SendFilesBox::generatePreviewFrom(int fromBlock) {
- Expects(fromBlock <= _blocks.size());
- using Type = Ui::PreparedFile::Type;
- _blocks.erase(_blocks.begin() + fromBlock, _blocks.end());
- const auto fromItem = _blocks.empty() ? 0 : _blocks.back().tillIndex();
- Assert(fromItem <= _list.files.size());
- auto albumStart = -1;
- for (auto i = fromItem, till = int(_list.files.size()); i != till; ++i) {
- const auto type = _list.files[i].type;
- if (albumStart >= 0) {
- const auto albumCount = (i - albumStart);
- if ((type == Type::File)
- || (type == Type::None)
- || (type == Type::Music)
- || (albumCount == Ui::MaxAlbumItems())) {
- pushBlock(std::exchange(albumStart, -1), i);
- } else {
- continue;
- }
- }
- if (type != Type::File
- && type != Type::Music
- && type != Type::None) {
- if (albumStart < 0) {
- albumStart = i;
- }
- continue;
- }
- pushBlock(i, i + 1);
- }
- if (albumStart >= 0) {
- pushBlock(albumStart, _list.files.size());
- }
- }
- void SendFilesBox::pushBlock(int from, int till) {
- const auto gifPaused = [show = _show] {
- return show->paused(Window::GifPauseReason::Layer);
- };
- _blocks.emplace_back(
- _inner.data(),
- _st,
- &_list.files,
- from,
- till,
- gifPaused,
- _sendWay.current(),
- [=](const Ui::PreparedFile &file, Ui::AttachActionType type) {
- return (type == Ui::AttachActionType::ToggleSpoiler)
- ? !hasPrice()
- : (type == Ui::AttachActionType::EditCover)
- ? (file.isVideoFile()
- && _captionToPeer
- && (_captionToPeer->isBroadcast()
- || _captionToPeer->isSelf()))
- : (file.videoCover != nullptr);
- });
- auto &block = _blocks.back();
- const auto widget = _inner->add(
- block.takeWidget(),
- QMargins(0, _inner->count() ? st::sendMediaRowSkip : 0, 0, 0));
- block.itemDeleteRequest(
- ) | rpl::filter([=] {
- return !_removingIndex;
- }) | rpl::start_with_next([=](int index) {
- applyBlockChanges();
- _removingIndex = index;
- crl::on_main(this, [=] {
- const auto index = base::take(_removingIndex).value_or(-1);
- if (index < 0 || index >= _list.files.size()) {
- return;
- }
- // Just close the box if it is the only one.
- if (_list.files.size() == 1) {
- closeBox();
- return;
- }
- refreshAllAfterChanges(index, [&] {
- _list.files.erase(_list.files.begin() + index);
- });
- });
- }, widget->lifetime());
- const auto show = uiShow();
- block.itemReplaceRequest(
- ) | rpl::start_with_next([=](int index) {
- applyBlockChanges();
- const auto replace = [=](Ui::PreparedList list) {
- if (list.files.empty()) {
- return;
- }
- refreshAllAfterChanges(from, [&] {
- _list.files[index] = std::move(list.files.front());
- });
- };
- const auto checkSlowmode = [=](const Ui::PreparedList &list) {
- if (list.files.empty() || !(_limits & SendFilesAllow::OnlyOne)) {
- return true;
- }
- auto removing = std::move(_list.files[index]);
- std::swap(_list.files[index], _list.files.back());
- _list.files.pop_back();
- const auto result = _list.canBeSentInSlowmodeWith(list);
- _list.files.push_back(std::move(removing));
- std::swap(_list.files[index], _list.files.back());
- if (!result) {
- show->showToast(tr::lng_slowmode_no_many(tr::now));
- return false;
- }
- return true;
- };
- const auto checkRights = [=](const Ui::PreparedList &list) {
- if (list.files.empty()) {
- return true;
- }
- auto removing = std::move(_list.files[index]);
- std::swap(_list.files[index], _list.files.back());
- _list.files.pop_back();
- auto way = _sendWay.current();
- const auto has = _list.hasSticker()
- || list.files.front().isSticker();
- way.setHasCompressedStickers(has);
- if (_limits & SendFilesAllow::OnlyOne) {
- way.setGroupFiles(true);
- }
- const auto silent = true;
- if (!checkWith(list, way, silent)
- && (!(_limits & SendFilesAllow::OnlyOne) || !has)) {
- way.setSendImagesAsPhotos(!way.sendImagesAsPhotos());
- }
- const auto result = checkWith(list, way);
- _list.files.push_back(std::move(removing));
- std::swap(_list.files[index], _list.files.back());
- if (!result) {
- return false;
- }
- _sendWay = way;
- return true;
- };
- const auto checkResult = [=](const Ui::PreparedList &list) {
- return checkSlowmode(list) && checkRights(list);
- };
- const auto callback = [=](FileDialog::OpenResult &&result) {
- const auto premium = _show->session().premium();
- FileDialogCallback(
- std::move(result),
- checkResult,
- replace,
- premium,
- show);
- };
- FileDialog::GetOpenPath(
- this,
- tr::lng_choose_file(tr::now),
- FileDialog::AllOrImagesFilter(),
- crl::guard(this, callback));
- }, widget->lifetime());
- const auto openedOnce = widget->lifetime().make_state<bool>(false);
- block.itemModifyRequest(
- ) | rpl::start_with_next([=, show = _show](int index) {
- applyBlockChanges();
- if (!(*openedOnce)) {
- show->session().settings().incrementPhotoEditorHintShown();
- show->session().saveSettings();
- }
- *openedOnce = true;
- Editor::OpenWithPreparedFile(
- this,
- show,
- &_list.files[index],
- st::sendMediaPreviewSize,
- [=](bool ok) { if (ok) refreshAllAfterChanges(from); });
- }, widget->lifetime());
- block.itemEditCoverRequest(
- ) | rpl::start_with_next([=, show = _show](int index) {
- applyBlockChanges();
- const auto replace = [=](Ui::PreparedList list) {
- if (list.files.empty()) {
- return;
- }
- auto &entry = _list.files[index];
- const auto video = entry.information
- ? std::get_if<Ui::PreparedFileInformation::Video>(
- &entry.information->media)
- : nullptr;
- if (!video) {
- return;
- }
- auto old = std::shared_ptr<Ui::PreparedFile>(
- std::move(entry.videoCover));
- entry.videoCover = std::make_unique<Ui::PreparedFile>(
- std::move(list.files.front()));
- Editor::OpenWithPreparedFile(
- this,
- show,
- entry.videoCover.get(),
- st::sendMediaPreviewSize,
- crl::guard(this, [=](bool ok) {
- if (!ok) {
- _list.files[index].videoCover = old
- ? std::make_unique<Ui::PreparedFile>(
- std::move(*old))
- : nullptr;
- }
- refreshAllAfterChanges(from);
- }),
- video->thumbnail.size());
- };
- const auto checkResult = [=](const Ui::PreparedList &list) {
- if (list.files.empty()) {
- return true;
- }
- if (list.files.front().type != Ui::PreparedFile::Type::Photo) {
- show->showToast(tr::lng_choose_cover_bad(tr::now));
- return false;
- }
- return true;
- };
- const auto callback = [=](FileDialog::OpenResult &&result) {
- const auto premium = _show->session().premium();
- FileDialogCallback(
- std::move(result),
- checkResult,
- replace,
- premium,
- show);
- };
- FileDialog::GetOpenPath(
- this,
- tr::lng_choose_cover(tr::now),
- FileDialog::ImagesFilter(),
- crl::guard(this, callback));
- }, widget->lifetime());
- block.itemClearCoverRequest(
- ) | rpl::start_with_next([=](int index) {
- applyBlockChanges();
- refreshAllAfterChanges(from, [&] {
- auto &entry = _list.files[index];
- entry.videoCover = nullptr;
- });
- }, widget->lifetime());
- block.orderUpdated() | rpl::start_with_next([=]{
- if (_priceTag) {
- _priceTagBg = QImage();
- _priceTag->update();
- }
- }, widget->lifetime());
- }
- void SendFilesBox::refreshControls(bool initial) {
- refreshButtons();
- refreshPriceTag();
- refreshTitleText();
- updateSendWayControls();
- updateCaptionPlaceholder();
- }
- void SendFilesBox::setupSendWayControls() {
- const auto groupFilesFirst = _sendWay.current().groupFiles();
- const auto asPhotosFirst = _sendWay.current().sendImagesAsPhotos();
- _groupFiles.create(
- this,
- tr::lng_send_grouped(tr::now),
- groupFilesFirst,
- _st.files.checkbox,
- _st.files.check);
- _sendImagesAsPhotos.create(
- this,
- tr::lng_send_compressed(tr::now),
- _sendWay.current().sendImagesAsPhotos(),
- _st.files.checkbox,
- _st.files.check);
- _sendWay.changes(
- ) | rpl::start_with_next([=](SendFilesWay value) {
- _groupFiles->setChecked(value.groupFiles());
- _sendImagesAsPhotos->setChecked(value.sendImagesAsPhotos());
- }, lifetime());
- _groupFiles->checkedChanges(
- ) | rpl::start_with_next([=](bool checked) {
- auto sendWay = _sendWay.current();
- if (sendWay.groupFiles() == checked) {
- return;
- }
- sendWay.setGroupFiles(checked);
- if (checkWithWay(sendWay)) {
- _sendWay = sendWay;
- } else {
- Ui::PostponeCall(_groupFiles.data(), [=] {
- _groupFiles->setChecked(!checked);
- });
- }
- }, lifetime());
- _sendImagesAsPhotos->checkedChanges(
- ) | rpl::start_with_next([=](bool checked) {
- auto sendWay = _sendWay.current();
- if (sendWay.sendImagesAsPhotos() == checked) {
- return;
- }
- sendWay.setSendImagesAsPhotos(checked);
- if (checkWithWay(sendWay)) {
- _sendWay = sendWay;
- } else {
- Ui::PostponeCall(_sendImagesAsPhotos.data(), [=] {
- _sendImagesAsPhotos->setChecked(!checked);
- });
- }
- }, lifetime());
- _wayRemember.create(
- this,
- tr::lng_remember(tr::now),
- false,
- _st.files.checkbox,
- _st.files.check);
- _wayRemember->hide();
- rpl::combine(
- _groupFiles->checkedValue(),
- _sendImagesAsPhotos->checkedValue()
- ) | rpl::start_with_next([=](bool groupFiles, bool asPhoto) {
- _wayRemember->setVisible(
- (groupFiles != groupFilesFirst) || (asPhoto != asPhotosFirst));
- captionResized();
- }, lifetime());
- _hintLabel.create(
- this,
- tr::lng_edit_photo_editor_hint(tr::now),
- st::editMediaHintLabel);
- }
- bool SendFilesBox::checkWithWay(Ui::SendFilesWay way, bool silent) const {
- return checkWith({}, way, silent);
- }
- bool SendFilesBox::checkWith(
- const Ui::PreparedList &added,
- Ui::SendFilesWay way,
- bool silent) const {
- if (!_check) {
- return true;
- }
- const auto compress = way.sendImagesAsPhotos();
- auto &already = _list.files;
- for (const auto &file : ranges::views::concat(already, added.files)) {
- if (!_check(file, compress, silent)) {
- return false;
- }
- }
- return true;
- }
- void SendFilesBox::updateSendWayControls() {
- const auto onlyOne = (_limits & SendFilesAllow::OnlyOne);
- _groupFiles->setVisible(_list.hasGroupOption(onlyOne));
- _sendImagesAsPhotos->setVisible(
- _list.hasSendImagesAsPhotosOption(onlyOne));
- _sendImagesAsPhotos->setText((_list.files.size() > 1)
- ? tr::lng_send_compressed(tr::now)
- : tr::lng_send_compressed_one(tr::now));
- _hintLabel->setVisible(
- _show->session().settings().photoEditorHintShown()
- ? _list.canHaveEditorHintLabel()
- : false);
- }
- void SendFilesBox::setupCaption() {
- const auto allow = [=](not_null<DocumentData*> emoji) {
- return _captionToPeer
- ? Data::AllowEmojiWithoutPremium(_captionToPeer, emoji)
- : (_limits & SendFilesAllow::EmojiWithoutPremium);
- };
- const auto show = _show;
- InitMessageFieldHandlers({
- .session = &show->session(),
- .show = show,
- .field = _caption.data(),
- .customEmojiPaused = [=] {
- return show->paused(Window::GifPauseReason::Layer);
- },
- .allowPremiumEmoji = allow,
- .fieldStyle = &_st.files.caption,
- });
- setupCaptionAutocomplete();
- Ui::Emoji::SuggestionsController::Init(
- getDelegate()->outerContainer(),
- _caption,
- &_show->session(),
- {
- .suggestCustomEmoji = true,
- .allowCustomWithoutPremium = allow,
- .st = &_st.suggestions,
- });
- if (!_prefilledCaptionText.text.isEmpty()) {
- _caption->setTextWithTags(
- _prefilledCaptionText,
- Ui::InputField::HistoryAction::Clear);
- auto cursor = _caption->textCursor();
- cursor.movePosition(QTextCursor::End);
- _caption->setTextCursor(cursor);
- }
- _caption->setSubmitSettings(
- Core::App().settings().sendSubmitWay());
- _caption->setMaxLength(kMaxMessageLength);
- _caption->heightChanges(
- ) | rpl::start_with_next([=] {
- captionResized();
- }, _caption->lifetime());
- _caption->submits(
- ) | rpl::start_with_next([=](Qt::KeyboardModifiers modifiers) {
- const auto ctrlShiftEnter = modifiers.testFlag(Qt::ShiftModifier)
- && (modifiers.testFlag(Qt::ControlModifier)
- || modifiers.testFlag(Qt::MetaModifier));
- send({}, ctrlShiftEnter);
- }, _caption->lifetime());
- _caption->cancelled(
- ) | rpl::start_with_next([=] {
- closeBox();
- }, _caption->lifetime());
- _caption->setMimeDataHook([=](
- not_null<const QMimeData*> data,
- Ui::InputField::MimeAction action) {
- if (action == Ui::InputField::MimeAction::Check) {
- return CanAddFiles(data);
- } else if (action == Ui::InputField::MimeAction::Insert) {
- return addFiles(data);
- }
- Unexpected("action in MimeData hook.");
- });
- updateCaptionPlaceholder();
- setupEmojiPanel();
- rpl::single(rpl::empty_value()) | rpl::then(
- _caption->changes()
- ) | rpl::start_with_next([=] {
- checkCharsLimitation();
- refreshMessagesCount();
- }, _caption->lifetime());
- }
- void SendFilesBox::setupCaptionAutocomplete() {
- if (!_captionToPeer || !_caption) {
- return;
- }
- const auto parent = getDelegate()->outerContainer();
- ChatHelpers::InitFieldAutocomplete(_autocomplete, {
- .parent = parent,
- .show = _show,
- .field = _caption.data(),
- .peer = _captionToPeer,
- .features = [=] {
- auto result = ChatHelpers::ComposeFeatures();
- result.autocompleteCommands = false;
- result.suggestStickersByEmoji = false;
- return result;
- },
- .sendMenuDetails = _sendMenuDetails,
- });
- 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, _caption->geometry());
- _autocomplete->setBoundings(QRect(
- field.x() - _caption->x(),
- st::defaultBox.margin.top(),
- width(),
- (field.y()
- + _st.files.caption.textMargins.top()
- + _st.files.caption.placeholderShift
- + _st.files.caption.placeholderFont->height
- - st::defaultBox.margin.top())));
- });
- };
- for (auto w = (QWidget*)_caption.data(); 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 SendFilesBox::checkCharsLimitation() {
- const auto limits = Data::PremiumLimits(&_show->session());
- const auto caption = (_caption && !_caption->isHidden())
- ? _caption->getTextWithAppliedMarkdown()
- : TextWithTags();
- const auto remove = caption.text.size() - limits.captionLengthCurrent();
- if ((remove > 0) && _emojiToggle) {
- if (!_charsLimitation) {
- _charsLimitation = base::make_unique_q<CharactersLimitLabel>(
- this,
- _emojiToggle.data(),
- style::al_top);
- _charsLimitation->show();
- Data::AmPremiumValue(
- &_show->session()
- ) | rpl::start_with_next([=] {
- checkCharsLimitation();
- }, _charsLimitation->lifetime());
- }
- _charsLimitation->setLeft(remove);
- } else {
- if (_charsLimitation) {
- _charsLimitation = nullptr;
- }
- }
- }
- void SendFilesBox::setupEmojiPanel() {
- Expects(_caption != nullptr);
- const auto container = getDelegate()->outerContainer();
- using Selector = ChatHelpers::TabbedSelector;
- _emojiPanel = base::make_unique_q<ChatHelpers::TabbedPanel>(
- container,
- ChatHelpers::TabbedPanelDescriptor{
- .ownedSelector = object_ptr<Selector>(
- nullptr,
- ChatHelpers::TabbedSelectorDescriptor{
- .show = _show,
- .st = _st.tabbed,
- .level = Window::GifPauseReason::Layer,
- .mode = ChatHelpers::TabbedSelector::Mode::EmojiOnly,
- .features = {
- .stickersSettings = false,
- .openStickerSets = false,
- },
- }),
- });
- _emojiPanel->setDesiredHeightValues(
- 1.,
- st::emojiPanMinHeight / 2,
- st::emojiPanMinHeight);
- _emojiPanel->hide();
- _emojiPanel->selector()->setCurrentPeer(_captionToPeer);
- _emojiPanel->selector()->setAllowEmojiWithoutPremium(
- _limits & SendFilesAllow::EmojiWithoutPremium);
- _emojiPanel->selector()->emojiChosen(
- ) | rpl::start_with_next([=](ChatHelpers::EmojiChosen data) {
- Ui::InsertEmojiAtCursor(_caption->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
- && !_show->session().premium()
- && !(_captionToPeer
- ? Data::AllowEmojiWithoutPremium(
- _captionToPeer,
- data.document)
- : (_limits & SendFilesAllow::EmojiWithoutPremium))) {
- ShowPremiumPreviewBox(_show, PremiumFeature::AnimatedEmoji);
- } else {
- Data::InsertCustomEmoji(_caption.data(), 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.create(this, _st.files.emoji);
- _emojiToggle->setVisible(!_caption->isHidden());
- _emojiToggle->installEventFilter(_emojiPanel);
- _emojiToggle->addClickHandler([=] {
- _emojiPanel->toggleAnimated();
- });
- }
- void SendFilesBox::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 SendFilesBox::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);
- }
- void SendFilesBox::captionResized() {
- updateBoxSize();
- updateControlsGeometry();
- updateEmojiPanelGeometry();
- update();
- }
- bool SendFilesBox::addFiles(not_null<const QMimeData*> data) {
- const auto premium = _show->session().premium();
- auto list = [&] {
- const auto urls = Core::ReadMimeUrls(data);
- auto result = CanAddUrls(urls)
- ? Storage::PrepareMediaList(
- urls,
- st::sendMediaPreviewSize,
- premium)
- : Ui::PreparedList(
- Ui::PreparedList::Error::EmptyFile,
- QString());
- if (result.error == Ui::PreparedList::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;
- }();
- return addFiles(std::move(list));
- }
- bool SendFilesBox::addFiles(Ui::PreparedList list) {
- if (list.error != Ui::PreparedList::Error::None) {
- return false;
- }
- const auto count = int(_list.files.size());
- _list.filesToProcess.insert(
- _list.filesToProcess.end(),
- std::make_move_iterator(list.files.begin()),
- std::make_move_iterator(list.files.end()));
- _list.filesToProcess.insert(
- _list.filesToProcess.end(),
- std::make_move_iterator(list.filesToProcess.begin()),
- std::make_move_iterator(list.filesToProcess.end()));
- enqueueNextPrepare();
- if (_list.files.size() > count) {
- refreshAllAfterChanges(count);
- }
- return true;
- }
- void SendFilesBox::addPreparedAsyncFile(Ui::PreparedFile &&file) {
- Expects(file.information != nullptr);
- _preparing = false;
- const auto count = int(_list.files.size());
- addFile(std::move(file));
- enqueueNextPrepare();
- if (_list.files.size() > count) {
- refreshAllAfterChanges(count);
- }
- if (!_preparing && _whenReadySend) {
- _whenReadySend();
- }
- }
- void SendFilesBox::addFile(Ui::PreparedFile &&file) {
- // canBeSentInSlowmode checks for non empty filesToProcess.
- auto saved = base::take(_list.filesToProcess);
- _list.files.push_back(std::move(file));
- const auto lastOk = [&] {
- auto way = _sendWay.current();
- if (_limits & SendFilesAllow::OnlyOne) {
- way.setGroupFiles(true);
- if (!_list.canBeSentInSlowmode()) {
- return false;
- }
- } else if (!checkWithWay(way)) {
- return false;
- }
- _sendWay = way;
- return true;
- }();
- if (!lastOk) {
- _list.files.pop_back();
- }
- _list.filesToProcess = std::move(saved);
- }
- void SendFilesBox::refreshTitleText() {
- using Type = Ui::PreparedFile::Type;
- const auto count = int(_list.files.size());
- if (count > 1) {
- const auto imagesCount = ranges::count(
- _list.files,
- Type::Photo,
- &Ui::PreparedFile::type);
- _titleText = (imagesCount == count)
- ? tr::lng_send_images_selected(tr::now, lt_count, count)
- : tr::lng_send_files_selected(tr::now, lt_count, count);
- } else {
- const auto type = _list.files.empty()
- ? Type::None
- : _list.files.front().type;
- _titleText = (type == Type::Photo)
- ? tr::lng_send_image(tr::now)
- : (type == Type::Video)
- ? tr::lng_send_video(tr::now)
- : tr::lng_send_file(tr::now);
- }
- _titleHeight = st::boxTitleHeight;
- }
- void SendFilesBox::updateBoxSize() {
- auto footerHeight = 0;
- if (_caption && !_caption->isHidden()) {
- footerHeight += st::boxPhotoCaptionSkip + _caption->height();
- }
- const auto pairs = std::array<std::pair<RpWidget*, int>, 4>{ {
- { _groupFiles.data(), st::boxPhotoCompressedSkip },
- { _sendImagesAsPhotos.data(), st::boxPhotoCompressedSkip },
- { _wayRemember.data(), st::boxPhotoCompressedSkip },
- { _hintLabel.data(), st::editMediaLabelMargins.top() },
- } };
- for (const auto &pair : pairs) {
- const auto pointer = pair.first;
- if (pointer && !pointer->isHidden()) {
- footerHeight += pair.second + pointer->heightNoMargins();
- }
- }
- _footerHeight = footerHeight;
- }
- void SendFilesBox::keyPressEvent(QKeyEvent *e) {
- if (e->matches(QKeySequence::Open)) {
- openDialogToAddFileToAlbum();
- } else 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);
- if (!_titleText.isEmpty()) {
- Painter p(this);
- p.setFont(st::boxTitleFont);
- p.setPen(getDelegate()->style().title.textFg);
- p.drawTextLeft(
- st::boxPhotoTitlePosition.x(),
- st::boxTitlePosition.y() - st::boxTopMargin,
- width(),
- _titleText);
- }
- }
- void SendFilesBox::resizeEvent(QResizeEvent *e) {
- BoxContent::resizeEvent(e);
- updateControlsGeometry();
- }
- void SendFilesBox::updateControlsGeometry() {
- auto bottom = height();
- if (_caption && !_caption->isHidden()) {
- _caption->resize(st::sendMediaPreviewSize, _caption->height());
- _caption->moveToLeft(
- st::boxPhotoPadding.left(),
- bottom - _caption->height());
- bottom -= st::boxPhotoCaptionSkip + _caption->height();
- if (_emojiToggle) {
- _emojiToggle->moveToLeft(
- (st::boxPhotoPadding.left()
- + st::sendMediaPreviewSize
- - _emojiToggle->width()),
- _caption->y() + st::boxAttachEmojiTop);
- _emojiToggle->update();
- }
- }
- const auto pairs = std::array<std::pair<RpWidget*, int>, 4>{ {
- { _hintLabel.data(), st::editMediaLabelMargins.top() },
- { _groupFiles.data(), st::boxPhotoCompressedSkip },
- { _sendImagesAsPhotos.data(), st::boxPhotoCompressedSkip },
- { _wayRemember.data(), st::boxPhotoCompressedSkip },
- } };
- for (const auto &pair : ranges::views::reverse(pairs)) {
- const auto pointer = pair.first;
- if (pointer && !pointer->isHidden()) {
- pointer->moveToLeft(
- st::boxPhotoPadding.left(),
- bottom - pointer->heightNoMargins());
- bottom -= pair.second + pointer->heightNoMargins();
- }
- }
- _scroll->resize(width(), bottom - _titleHeight.current());
- _scroll->move(0, _titleHeight.current());
- }
- void SendFilesBox::showFinished() {
- if (const auto raw = _autocomplete.get()) {
- InvokeQueued(raw, [=] {
- raw->raise();
- });
- }
- }
- void SendFilesBox::setInnerFocus() {
- if (_caption && !_caption->isHidden()) {
- _caption->setFocusFast();
- } else {
- BoxContent::setInnerFocus();
- }
- }
- void SendFilesBox::saveSendWaySettings() {
- auto way = _sendWay.current();
- auto oldWay = Core::App().settings().sendFilesWay();
- if (_groupFiles->isHidden()) {
- way.setGroupFiles(oldWay.groupFiles());
- }
- if (_list.overrideSendImagesAsPhotos == way.sendImagesAsPhotos()
- || _sendImagesAsPhotos->isHidden()) {
- way.setSendImagesAsPhotos(oldWay.sendImagesAsPhotos());
- }
- if (way != oldWay) {
- Core::App().settings().setSendFilesWay(way);
- Core::App().saveSettingsDelayed();
- }
- }
- bool SendFilesBox::validateLength(const QString &text) const {
- const auto session = &_show->session();
- const auto limit = Data::PremiumLimits(session).captionLengthCurrent();
- const auto remove = int(text.size()) - limit;
- const auto way = _sendWay.current();
- if (remove <= 0
- || !_list.canAddCaption(
- way.groupFiles() && way.sendImagesAsPhotos(),
- way.sendImagesAsPhotos())) {
- return true;
- }
- _show->showBox(
- Box(CaptionLimitReachedBox, session, remove, &_st.premium));
- return false;
- }
- void SendFilesBox::send(
- Api::SendOptions options,
- bool ctrlShiftEnter) {
- if ((_sendType == Api::SendType::Scheduled
- || _sendType == Api::SendType::ScheduledToUser)
- && !options.scheduled) {
- auto child = _sendMenuDetails();
- child.spoiler = SendMenu::SpoilerState::None;
- child.caption = SendMenu::CaptionState::None;
- child.price = std::nullopt;
- return SendMenu::DefaultCallback(_show, sendCallback())(
- { .type = SendMenu::ActionType::Schedule },
- child);
- }
- if (_preparing) {
- _whenReadySend = [=] {
- send(options, ctrlShiftEnter);
- };
- return;
- }
- if (_wayRemember && _wayRemember->checked()) {
- saveSendWaySettings();
- }
- for (auto &item : _list.files) {
- item.spoiler = false;
- }
- applyBlockChanges();
- Storage::ApplyModifications(_list);
- _confirmed = true;
- if (_confirmedCallback) {
- auto caption = (_caption && !_caption->isHidden())
- ? _caption->getTextWithAppliedMarkdown()
- : TextWithTags();
- if (!validateLength(caption.text)) {
- return;
- }
- options.invertCaption = _invertCaption;
- options.price = hasPrice() ? _price.current() : 0;
- if (options.price > 0) {
- for (auto &file : _list.files) {
- file.spoiler = false;
- }
- }
- _confirmedCallback(
- std::move(_list),
- _sendWay.current(),
- std::move(caption),
- options,
- ctrlShiftEnter);
- }
- closeBox();
- }
- Fn<void(Api::SendOptions)> SendFilesBox::sendCallback() {
- return crl::guard(this, [=](Api::SendOptions options) {
- send(options, false);
- });
- }
- SendFilesBox::~SendFilesBox() = default;
|