mirror of
https://github.com/AyuGram/AyuGramDesktop.git
synced 2025-04-26 19:14:02 +02:00
199 lines
5.1 KiB
C++
199 lines
5.1 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) {
|
|
using ImageInfo = Ui::PreparedFileInformation::Image;
|
|
const auto image = std::get_if<ImageInfo>(&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<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(Ui::MakeInformBox(tr::lng_bad_photo()));
|
|
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
|