tdesktop/Telegram/SourceFiles/editor/photo_editor_layer_widget.cpp

191 lines
4.7 KiB
C++

/*
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<Ui::RpWidget*> parent,
not_null<Window::SessionController*> controller,
not_null<Ui::PreparedFile*> file,
int previewWidth,
Fn<void()> &&doneCallback) {
if (file->type != Ui::PreparedFile::Type::Photo) {
return;
}
using ImageInfo = Ui::PreparedFileInformation::Image;
const auto image = std::get_if<ImageInfo>(&file->information->media);
if (!image) {
return;
}
auto callback = [=, done = std::move(doneCallback)](
const PhotoModifications &mods) {
image->modifications = mods;
Storage::UpdateImageDetails(*file, previewWidth);
done();
};
auto copy = image->data;
const auto fileImage = std::make_shared<Image>(std::move(copy));
controller->showLayer(
std::make_unique<LayerWidget>(
parent,
&controller->window(),
fileImage,
image->modifications,
std::move(callback)),
Ui::LayerOption::KeepOther);
}
void PrepareProfilePhoto(
not_null<Ui::RpWidget*> parent,
not_null<Window::Controller*> controller,
Fn<void(QImage &&image)> &&doneCallback) {
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);
};
const auto callback = [=, done = std::move(doneCallback)](
const FileDialog::OpenResult &result) {
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;
if (image.isNull()
|| (image.width() > (10 * image.height()))
|| (image.height() > (10 * image.width()))) {
controller->show(Box<Ui::InformBox>(tr::lng_bad_photo(tr::now)));
return;
}
image = resizeToMinSize(
std::move(image),
Qt::KeepAspectRatioByExpanding);
const auto fileImage = std::make_shared<Image>(std::move(image));
auto applyModifications = [=, done = std::move(done)](
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<LayerWidget>(
parent,
controller,
fileImage,
PhotoModifications{ .crop = std::move(crop) },
std::move(applyModifications),
EditorData{
.cropType = EditorData::CropType::Ellipse,
.keepAspectRatio = true, }),
Ui::LayerOption::KeepOther);
};
FileDialog::GetOpenPath(
parent.get(),
tr::lng_choose_image(tr::now),
FileDialog::ImagesOrAllFilter(),
crl::guard(parent, callback));
}
LayerWidget::LayerWidget(
not_null<Ui::RpWidget*> parent,
not_null<Window::Controller*> window,
std::shared_ptr<Image> photo,
PhotoModifications modifications,
Fn<void(PhotoModifications)> &&doneCallback,
EditorData data)
: Ui::LayerWidget(parent)
, _content(base::make_unique_q<PhotoEditor>(
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