/* 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 "editor/photo_editor_layer_widget.h" #include "ui/boxes/confirm_box.h" // InformBox #include "editor/photo_editor.h" #include "storage/storage_media_prepare.h" #include "ui/chat/attach/attach_prepare.h" #include "window/window_controller.h" #include "window/window_session_controller.h" namespace Editor { namespace { constexpr auto kProfilePhotoSize = 640; } // namespace void OpenWithPreparedFile( not_null parent, not_null controller, not_null file, int previewWidth, Fn &&doneCallback) { using ImageInfo = Ui::PreparedFileInformation::Image; const auto image = std::get_if(&file->information->media); if (!image) { return; } const auto photoType = (file->type == Ui::PreparedFile::Type::Photo); const auto modifiedFileType = (file->type == Ui::PreparedFile::Type::File) && !image->modifications.empty(); if (!photoType && !modifiedFileType) { return; } auto callback = [=, done = std::move(doneCallback)]( const PhotoModifications &mods) { image->modifications = mods; Storage::UpdateImageDetails(*file, previewWidth); { using namespace Ui; const auto size = file->preview.size(); file->type = ValidateThumbDimensions(size.width(), size.height()) ? PreparedFile::Type::Photo : PreparedFile::Type::File; } done(); }; auto copy = image->data; const auto fileImage = std::make_shared(std::move(copy)); controller->showLayer( std::make_unique( parent, &controller->window(), fileImage, image->modifications, std::move(callback)), Ui::LayerOption::KeepOther); } void PrepareProfilePhoto( not_null parent, not_null controller, Fn &&doneCallback, QImage &&image) { const auto resizeToMinSize = [=]( QImage &&image, Qt::AspectRatioMode mode) { const auto &minSize = kProfilePhotoSize; if ((image.width() < minSize) || (image.height() < minSize)) { return image.scaled( minSize, minSize, mode, Qt::SmoothTransformation); } return std::move(image); }; if (image.isNull() || (image.width() > (10 * image.height())) || (image.height() > (10 * image.width()))) { controller->show(Ui::MakeInformBox(tr::lng_bad_photo())); return; } image = resizeToMinSize( std::move(image), Qt::KeepAspectRatioByExpanding); const auto fileImage = std::make_shared(std::move(image)); auto applyModifications = [=, done = std::move(doneCallback)]( const PhotoModifications &mods) { done(resizeToMinSize( ImageModified(fileImage->original(), mods), Qt::KeepAspectRatio)); }; auto crop = [&] { const auto &i = fileImage; const auto minSide = std::min(i->width(), i->height()); return QRect( (i->width() - minSide) / 2, (i->height() - minSide) / 2, minSide, minSide); }(); controller->showLayer( std::make_unique( parent, controller, fileImage, PhotoModifications{ .crop = std::move(crop) }, std::move(applyModifications), EditorData{ .cropType = EditorData::CropType::Ellipse, .keepAspectRatio = true, }), Ui::LayerOption::KeepOther); } void PrepareProfilePhotoFromFile( not_null parent, not_null controller, Fn &&doneCallback) { const auto callback = [=, done = std::move(doneCallback)]( const FileDialog::OpenResult &result) mutable { if (result.paths.isEmpty() && result.remoteContent.isEmpty()) { return; } auto image = Images::Read({ .path = result.paths.isEmpty() ? QString() : result.paths.front(), .content = result.remoteContent, .forceOpaque = true, }).image; PrepareProfilePhoto( parent, controller, std::move(done), std::move(image)); }; FileDialog::GetOpenPath( parent.get(), tr::lng_choose_image(tr::now), FileDialog::ImagesOrAllFilter(), crl::guard(parent, callback)); } LayerWidget::LayerWidget( not_null parent, not_null window, std::shared_ptr photo, PhotoModifications modifications, Fn &&doneCallback, EditorData data) : Ui::LayerWidget(parent) , _content(base::make_unique_q( this, window, photo, std::move(modifications), std::move(data))) { paintRequest( ) | rpl::start_with_next([=](const QRect &clip) { Painter p(this); p.fillRect(clip, st::photoEditorBg); }, lifetime()); _content->cancelRequests( ) | rpl::start_with_next([=] { closeLayer(); }, lifetime()); _content->doneRequests( ) | rpl::start_with_next([=, done = std::move(doneCallback)]( const PhotoModifications &mods) { done(mods); closeLayer(); }, lifetime()); sizeValue( ) | rpl::start_with_next([=](const QSize &size) { _content->resize(size); }, lifetime()); } void LayerWidget::parentResized() { resizeToWidth(parentWidget()->width()); } void LayerWidget::keyPressEvent(QKeyEvent *e) { _content->handleKeyPress(e); } int LayerWidget::resizeGetHeight(int newWidth) { return parentWidget()->height(); } bool LayerWidget::closeByOutsideClick() const { return false; } } // namespace Editor