242 lines
		
	
	
	
		
			6.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			242 lines
		
	
	
	
		
			6.3 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.h"
 | 
						|
 | 
						|
#include "core/application.h"
 | 
						|
#include "core/core_settings.h"
 | 
						|
#include "editor/color_picker.h"
 | 
						|
#include "editor/controllers/controllers.h"
 | 
						|
#include "editor/photo_editor_content.h"
 | 
						|
#include "editor/photo_editor_controls.h"
 | 
						|
#include "window/window_controller.h"
 | 
						|
#include "window/window_session_controller.h"
 | 
						|
#include "ui/layers/layer_widget.h"
 | 
						|
#include "styles/style_editor.h"
 | 
						|
 | 
						|
namespace Editor {
 | 
						|
namespace {
 | 
						|
 | 
						|
constexpr auto kPrecision = 100000;
 | 
						|
 | 
						|
[[nodiscard]] QByteArray Serialize(const Brush &brush) {
 | 
						|
	auto result = QByteArray();
 | 
						|
	auto stream = QDataStream(&result, QIODevice::WriteOnly);
 | 
						|
	stream.setVersion(QDataStream::Qt_5_3);
 | 
						|
	stream << qint32(brush.sizeRatio * kPrecision) << brush.color;
 | 
						|
	stream.device()->close();
 | 
						|
 | 
						|
	return result;
 | 
						|
}
 | 
						|
 | 
						|
[[nodiscard]] Brush Deserialize(const QByteArray &data) {
 | 
						|
	auto stream = QDataStream(data);
 | 
						|
	auto result = Brush();
 | 
						|
	auto size = qint32(0);
 | 
						|
	stream >> size >> result.color;
 | 
						|
	result.sizeRatio = size / float(kPrecision);
 | 
						|
	return (stream.status() != QDataStream::Ok)
 | 
						|
		? Brush()
 | 
						|
		: result;
 | 
						|
}
 | 
						|
 | 
						|
} // namespace
 | 
						|
 | 
						|
PhotoEditor::PhotoEditor(
 | 
						|
	not_null<QWidget*> parent,
 | 
						|
	not_null<Window::Controller*> controller,
 | 
						|
	std::shared_ptr<Image> photo,
 | 
						|
	PhotoModifications modifications,
 | 
						|
	EditorData data)
 | 
						|
: PhotoEditor(
 | 
						|
	parent,
 | 
						|
	controller->uiShow(),
 | 
						|
	(controller->sessionController()
 | 
						|
		? controller->sessionController()->uiShow()
 | 
						|
		: nullptr),
 | 
						|
	std::move(photo),
 | 
						|
	std::move(modifications),
 | 
						|
	std::move(data)) {
 | 
						|
}
 | 
						|
 | 
						|
PhotoEditor::PhotoEditor(
 | 
						|
	not_null<QWidget*> parent,
 | 
						|
	std::shared_ptr<Ui::Show> show,
 | 
						|
	std::shared_ptr<ChatHelpers::Show> sessionShow,
 | 
						|
	std::shared_ptr<Image> photo,
 | 
						|
	PhotoModifications modifications,
 | 
						|
	EditorData data)
 | 
						|
: RpWidget(parent)
 | 
						|
, _modifications(std::move(modifications))
 | 
						|
, _controllers(std::make_shared<Controllers>(
 | 
						|
	sessionShow
 | 
						|
		? std::make_unique<StickersPanelController>(
 | 
						|
			this,
 | 
						|
			std::move(sessionShow))
 | 
						|
		: nullptr,
 | 
						|
	std::make_unique<UndoController>(),
 | 
						|
	std::move(show)))
 | 
						|
, _content(base::make_unique_q<PhotoEditorContent>(
 | 
						|
	this,
 | 
						|
	photo,
 | 
						|
	_modifications,
 | 
						|
	_controllers,
 | 
						|
	data))
 | 
						|
, _controls(base::make_unique_q<PhotoEditorControls>(
 | 
						|
	this,
 | 
						|
	_controllers,
 | 
						|
	_modifications,
 | 
						|
	data))
 | 
						|
, _colorPicker(std::make_unique<ColorPicker>(
 | 
						|
	this,
 | 
						|
	Deserialize(Core::App().settings().photoEditorBrush()))) {
 | 
						|
 | 
						|
	sizeValue(
 | 
						|
	) | rpl::start_with_next([=](const QSize &size) {
 | 
						|
		if (size.isEmpty()) {
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		_content->setGeometry(rect() - st::photoEditorContentMargins);
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	_content->innerRect(
 | 
						|
	) | rpl::start_with_next([=](QRect inner) {
 | 
						|
		if (inner.isEmpty()) {
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		const auto innerTop = _content->y() + inner.top();
 | 
						|
		const auto skip = st::photoEditorCropPointSize;
 | 
						|
		const auto controlsRect = rect()
 | 
						|
			- style::margins(0, innerTop + inner.height() + skip, 0, 0);
 | 
						|
		_controls->setGeometry(controlsRect);
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	_controls->colorLinePositionValue(
 | 
						|
	) | rpl::start_with_next([=](const QPoint &p) {
 | 
						|
		_colorPicker->moveLine(p);
 | 
						|
	}, _controls->lifetime());
 | 
						|
 | 
						|
	_controls->colorLineShownValue(
 | 
						|
	) | rpl::start_with_next([=](bool shown) {
 | 
						|
		_colorPicker->setVisible(shown);
 | 
						|
	}, _controls->lifetime());
 | 
						|
 | 
						|
	_mode.value(
 | 
						|
	) | rpl::start_with_next([=](const PhotoEditorMode &mode) {
 | 
						|
		_content->applyMode(mode);
 | 
						|
		_controls->applyMode(mode);
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	_controls->rotateRequests(
 | 
						|
	) | rpl::start_with_next([=](int angle) {
 | 
						|
		_modifications.angle += 90;
 | 
						|
		if (_modifications.angle >= 360) {
 | 
						|
			_modifications.angle -= 360;
 | 
						|
		}
 | 
						|
		_content->applyModifications(_modifications);
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	_controls->flipRequests(
 | 
						|
	) | rpl::start_with_next([=] {
 | 
						|
		_modifications.flipped = !_modifications.flipped;
 | 
						|
		_content->applyModifications(_modifications);
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	_controls->paintModeRequests(
 | 
						|
	) | rpl::start_with_next([=] {
 | 
						|
		_mode = PhotoEditorMode{
 | 
						|
			.mode = PhotoEditorMode::Mode::Paint,
 | 
						|
			.action = PhotoEditorMode::Action::None,
 | 
						|
		};
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	_controls->doneRequests(
 | 
						|
	) | rpl::start_with_next([=] {
 | 
						|
		const auto mode = _mode.current().mode;
 | 
						|
		if (mode == PhotoEditorMode::Mode::Paint) {
 | 
						|
			_mode = PhotoEditorMode{
 | 
						|
				.mode = PhotoEditorMode::Mode::Transform,
 | 
						|
				.action = PhotoEditorMode::Action::Save,
 | 
						|
			};
 | 
						|
		} else if (mode == PhotoEditorMode::Mode::Transform) {
 | 
						|
			_mode = PhotoEditorMode{
 | 
						|
				.mode = PhotoEditorMode::Mode::Out,
 | 
						|
				.action = PhotoEditorMode::Action::Save,
 | 
						|
			};
 | 
						|
			save();
 | 
						|
		}
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	_controls->cancelRequests(
 | 
						|
	) | rpl::start_with_next([=] {
 | 
						|
		const auto mode = _mode.current().mode;
 | 
						|
		if (mode == PhotoEditorMode::Mode::Paint) {
 | 
						|
			_mode = PhotoEditorMode{
 | 
						|
				.mode = PhotoEditorMode::Mode::Transform,
 | 
						|
				.action = PhotoEditorMode::Action::Discard,
 | 
						|
			};
 | 
						|
		} else if (mode == PhotoEditorMode::Mode::Transform) {
 | 
						|
			_mode = PhotoEditorMode{
 | 
						|
				.mode = PhotoEditorMode::Mode::Out,
 | 
						|
				.action = PhotoEditorMode::Action::Discard,
 | 
						|
			};
 | 
						|
			_cancel.fire({});
 | 
						|
		}
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	_colorPicker->saveBrushRequests(
 | 
						|
	) | rpl::start_with_next([=](const Brush &brush) {
 | 
						|
		_content->applyBrush(brush);
 | 
						|
 | 
						|
		const auto serialized = Serialize(brush);
 | 
						|
		if (Core::App().settings().photoEditorBrush() != serialized) {
 | 
						|
			Core::App().settings().setPhotoEditorBrush(serialized);
 | 
						|
			Core::App().saveSettingsDelayed();
 | 
						|
		}
 | 
						|
	}, lifetime());
 | 
						|
}
 | 
						|
 | 
						|
void PhotoEditor::keyPressEvent(QKeyEvent *e) {
 | 
						|
	if (!_colorPicker->preventHandleKeyPress()) {
 | 
						|
		_content->handleKeyPress(e) || _controls->handleKeyPress(e);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void PhotoEditor::save() {
 | 
						|
	_content->save(_modifications);
 | 
						|
	_done.fire_copy(_modifications);
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<PhotoModifications> PhotoEditor::doneRequests() const {
 | 
						|
	return _done.events();
 | 
						|
}
 | 
						|
 | 
						|
rpl::producer<> PhotoEditor::cancelRequests() const {
 | 
						|
	return _cancel.events();
 | 
						|
}
 | 
						|
 | 
						|
void InitEditorLayer(
 | 
						|
		not_null<Ui::LayerWidget*> layer,
 | 
						|
		not_null<PhotoEditor*> editor,
 | 
						|
		Fn<void(PhotoModifications)> doneCallback) {
 | 
						|
	editor->cancelRequests(
 | 
						|
	) | rpl::start_with_next([=] {
 | 
						|
		layer->closeLayer();
 | 
						|
	}, editor->lifetime());
 | 
						|
 | 
						|
	const auto weak = Ui::MakeWeak(layer.get());
 | 
						|
	editor->doneRequests(
 | 
						|
	) | rpl::start_with_next([=, done = std::move(doneCallback)](
 | 
						|
			const PhotoModifications &mods) {
 | 
						|
		done(mods);
 | 
						|
		if (const auto strong = weak.data()) {
 | 
						|
			strong->closeLayer();
 | 
						|
		}
 | 
						|
	}, editor->lifetime());
 | 
						|
}
 | 
						|
 | 
						|
} // namespace Editor
 |