332 lines
		
	
	
	
		
			8.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			332 lines
		
	
	
	
		
			8.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 "boxes/username_box.h"
 | |
| 
 | |
| #include "boxes/peers/edit_peer_usernames_list.h"
 | |
| #include "base/timer.h"
 | |
| #include "boxes/peers/edit_peer_common.h"
 | |
| #include "data/data_session.h"
 | |
| #include "data/data_user.h"
 | |
| #include "lang/lang_keys.h"
 | |
| #include "main/main_session.h"
 | |
| #include "mtproto/sender.h"
 | |
| #include "settings/settings_common.h"
 | |
| #include "ui/layers/generic_box.h"
 | |
| #include "ui/painter.h"
 | |
| #include "ui/text/text_utilities.h"
 | |
| #include "ui/toast/toast.h"
 | |
| #include "ui/widgets/buttons.h"
 | |
| #include "ui/widgets/fields/special_fields.h"
 | |
| #include "ui/widgets/labels.h"
 | |
| #include "styles/style_layers.h"
 | |
| #include "styles/style_boxes.h"
 | |
| #include "styles/style_settings.h"
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| class UsernameEditor final : public Ui::RpWidget {
 | |
| public:
 | |
| 	UsernameEditor(not_null<Ui::RpWidget*>, not_null<Main::Session*> session);
 | |
| 
 | |
| 	void setInnerFocus();
 | |
| 	rpl::producer<> submitted() const;
 | |
| 	rpl::producer<> save();
 | |
| 
 | |
| protected:
 | |
| 	void paintEvent(QPaintEvent *e) override;
 | |
| 	void resizeEvent(QResizeEvent *e) override;
 | |
| 
 | |
| private:
 | |
| 	void updateFail(const QString &error);
 | |
| 	void checkFail(const QString &error);
 | |
| 
 | |
| 
 | |
| 	void check();
 | |
| 	void changed();
 | |
| 
 | |
| 	QString getName() const;
 | |
| 
 | |
| 	const not_null<Main::Session*> _session;
 | |
| 	const style::font &_font;
 | |
| 	const style::margins &_padding;
 | |
| 	MTP::Sender _api;
 | |
| 
 | |
| 	object_ptr<Ui::UsernameInput> _username;
 | |
| 
 | |
| 	mtpRequestId _saveRequestId = 0;
 | |
| 	mtpRequestId _checkRequestId = 0;
 | |
| 	QString _sentUsername, _checkUsername, _errorText, _goodText;
 | |
| 
 | |
| 	base::Timer _checkTimer;
 | |
| 
 | |
| 	rpl::event_stream<> _saved;
 | |
| 
 | |
| };
 | |
| 
 | |
| UsernameEditor::UsernameEditor(
 | |
| 	not_null<Ui::RpWidget*>,
 | |
| 	not_null<Main::Session*> session)
 | |
| : _session(session)
 | |
| , _font(st::normalFont)
 | |
| , _padding(st::usernamePadding)
 | |
| , _api(&_session->mtp())
 | |
| , _username(
 | |
| 	this,
 | |
| 	st::defaultInputField,
 | |
| 	rpl::single(qsl("@username")),
 | |
| 	session->user()->editableUsername(),
 | |
| 	QString())
 | |
| , _checkTimer([=] { check(); }) {
 | |
| 	_goodText = _session->user()->editableUsername().isEmpty()
 | |
| 		? QString()
 | |
| 		: tr::lng_username_available(tr::now);
 | |
| 
 | |
| 	connect(_username, &Ui::MaskedInputField::changed, [=] { changed(); });
 | |
| 
 | |
| 	resize(
 | |
| 		width(),
 | |
| 		(_padding.top()
 | |
| 			+ _username->height()
 | |
| 			+ st::usernameSkip));
 | |
| }
 | |
| 
 | |
| rpl::producer<> UsernameEditor::submitted() const {
 | |
| 	return [=](auto consumer) {
 | |
| 		auto lifetime = rpl::lifetime();
 | |
| 		QObject::connect(
 | |
| 			_username,
 | |
| 			&Ui::MaskedInputField::submitted,
 | |
| 			[=] { consumer.put_next({}); });
 | |
| 		return lifetime;
 | |
| 	};
 | |
| }
 | |
| 
 | |
| void UsernameEditor::setInnerFocus() {
 | |
| 	_username->setFocusFast();
 | |
| }
 | |
| 
 | |
| void UsernameEditor::paintEvent(QPaintEvent *e) {
 | |
| 	Painter p(this);
 | |
| 
 | |
| 	const auto textTop = _username->y()
 | |
| 		+ _username->height()
 | |
| 		+ ((st::usernameSkip - _font->height) / 2);
 | |
| 
 | |
| 	p.setFont(_font);
 | |
| 	if (!_errorText.isEmpty()) {
 | |
| 		p.setPen(st::boxTextFgError);
 | |
| 		p.drawTextLeft(
 | |
| 			_padding.left(),
 | |
| 			textTop,
 | |
| 			width(),
 | |
| 			_errorText);
 | |
| 	} else if (!_goodText.isEmpty()) {
 | |
| 		p.setPen(st::boxTextFgGood);
 | |
| 		p.drawTextLeft(
 | |
| 			_padding.left(),
 | |
| 			textTop,
 | |
| 			width(),
 | |
| 			_goodText);
 | |
| 	} else {
 | |
| 		p.setPen(st::usernameDefaultFg);
 | |
| 		p.drawTextLeft(
 | |
| 			_padding.left(),
 | |
| 			textTop,
 | |
| 			width(),
 | |
| 			tr::lng_username_choose(tr::now));
 | |
| 	}
 | |
| 	p.setPen(st::boxTextFg);
 | |
| }
 | |
| 
 | |
| void UsernameEditor::resizeEvent(QResizeEvent *e) {
 | |
| 	_username->resize(
 | |
| 		width() - _padding.left() - _padding.right(),
 | |
| 		_username->height());
 | |
| 	_username->moveToLeft(_padding.left(), _padding.top());
 | |
| }
 | |
| 
 | |
| rpl::producer<> UsernameEditor::save() {
 | |
| 	if (_saveRequestId) {
 | |
| 		return _saved.events();
 | |
| 	}
 | |
| 
 | |
| 	_sentUsername = getName();
 | |
| 	_saveRequestId = _api.request(MTPaccount_UpdateUsername(
 | |
| 		MTP_string(_sentUsername)
 | |
| 	)).done([=](const MTPUser &result) {
 | |
| 		_saveRequestId = 0;
 | |
| 		_session->data().processUser(result);
 | |
| 		_saved.fire_done();
 | |
| 	}).fail([=](const MTP::Error &error) {
 | |
| 		_saveRequestId = 0;
 | |
| 		updateFail(error.type());
 | |
| 	}).send();
 | |
| 	return _saved.events();
 | |
| }
 | |
| 
 | |
| void UsernameEditor::check() {
 | |
| 	_api.request(base::take(_checkRequestId)).cancel();
 | |
| 
 | |
| 	const auto name = getName();
 | |
| 	if (name.size() < Ui::EditPeer::kMinUsernameLength) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_checkUsername = name;
 | |
| 	_checkRequestId = _api.request(MTPaccount_CheckUsername(
 | |
| 		MTP_string(name)
 | |
| 	)).done([=](const MTPBool &result) {
 | |
| 		_checkRequestId = 0;
 | |
| 
 | |
| 		_errorText = (mtpIsTrue(result)
 | |
| 				|| _checkUsername == _session->user()->editableUsername())
 | |
| 			? QString()
 | |
| 			: tr::lng_username_occupied(tr::now);
 | |
| 		_goodText = _errorText.isEmpty()
 | |
| 			? tr::lng_username_available(tr::now)
 | |
| 			: QString();
 | |
| 
 | |
| 		update();
 | |
| 	}).fail([=](const MTP::Error &error) {
 | |
| 		_checkRequestId = 0;
 | |
| 		checkFail(error.type());
 | |
| 	}).send();
 | |
| }
 | |
| 
 | |
| void UsernameEditor::changed() {
 | |
| 	const auto name = getName();
 | |
| 	if (name.isEmpty()) {
 | |
| 		if (!_errorText.isEmpty() || !_goodText.isEmpty()) {
 | |
| 			_errorText = _goodText = QString();
 | |
| 			update();
 | |
| 		}
 | |
| 		_checkTimer.cancel();
 | |
| 	} else {
 | |
| 		const auto len = int(name.size());
 | |
| 		for (auto i = 0; i < len; ++i) {
 | |
| 			const auto ch = name.at(i);
 | |
| 			if ((ch < 'A' || ch > 'Z')
 | |
| 				&& (ch < 'a' || ch > 'z')
 | |
| 				&& (ch < '0' || ch > '9')
 | |
| 				&& ch != '_'
 | |
| 				&& (ch != '@' || i > 0)) {
 | |
| 				if (_errorText != tr::lng_username_bad_symbols(tr::now)) {
 | |
| 					_errorText = tr::lng_username_bad_symbols(tr::now);
 | |
| 					update();
 | |
| 				}
 | |
| 				_checkTimer.cancel();
 | |
| 				return;
 | |
| 			}
 | |
| 		}
 | |
| 		if (name.size() < Ui::EditPeer::kMinUsernameLength) {
 | |
| 			if (_errorText != tr::lng_username_too_short(tr::now)) {
 | |
| 				_errorText = tr::lng_username_too_short(tr::now);
 | |
| 				update();
 | |
| 			}
 | |
| 			_checkTimer.cancel();
 | |
| 		} else {
 | |
| 			if (!_errorText.isEmpty() || !_goodText.isEmpty()) {
 | |
| 				_errorText = _goodText = QString();
 | |
| 				update();
 | |
| 			}
 | |
| 			_checkTimer.callOnce(Ui::EditPeer::kUsernameCheckTimeout);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void UsernameEditor::updateFail(const QString &error) {
 | |
| 	const auto self = _session->user();
 | |
| 	if ((error == qstr("USERNAME_NOT_MODIFIED"))
 | |
| 		|| (_sentUsername == self->editableUsername())) {
 | |
| 		self->setName(
 | |
| 			TextUtilities::SingleLine(self->firstName),
 | |
| 			TextUtilities::SingleLine(self->lastName),
 | |
| 			TextUtilities::SingleLine(self->nameOrPhone),
 | |
| 			TextUtilities::SingleLine(_sentUsername));
 | |
| 		_saved.fire_done();
 | |
| 	} else if (error == qstr("USERNAME_INVALID")) {
 | |
| 		_username->setFocus();
 | |
| 		_username->showError();
 | |
| 		_errorText = tr::lng_username_invalid(tr::now);
 | |
| 		update();
 | |
| 	} else if ((error == qstr("USERNAME_OCCUPIED"))
 | |
| 		|| (error == qstr("USERNAMES_UNAVAILABLE"))) {
 | |
| 		_username->setFocus();
 | |
| 		_username->showError();
 | |
| 		_errorText = tr::lng_username_occupied(tr::now);
 | |
| 		update();
 | |
| 	} else {
 | |
| 		_username->setFocus();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void UsernameEditor::checkFail(const QString &error) {
 | |
| 	if (error == qstr("USERNAME_INVALID")) {
 | |
| 		_errorText = tr::lng_username_invalid(tr::now);
 | |
| 		update();
 | |
| 	} else if ((error == qstr("USERNAME_OCCUPIED"))
 | |
| 		&& (_checkUsername != _session->user()->editableUsername())) {
 | |
| 		_errorText = tr::lng_username_occupied(tr::now);
 | |
| 		update();
 | |
| 	} else {
 | |
| 		_goodText = QString();
 | |
| 		_username->setFocus();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| QString UsernameEditor::getName() const {
 | |
| 	return _username->text().replace('@', QString()).trimmed();
 | |
| }
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| void UsernamesBox(
 | |
| 		not_null<Ui::GenericBox*> box,
 | |
| 		not_null<Main::Session*> session) {
 | |
| 	box->setTitle(tr::lng_username_title());
 | |
| 
 | |
| 	const auto container = box->verticalLayout();
 | |
| 
 | |
| 	const auto editor = box->addRow(
 | |
| 		object_ptr<UsernameEditor>(box, session),
 | |
| 		{});
 | |
| 	box->setFocusCallback([=] { editor->setInnerFocus(); });
 | |
| 
 | |
| 	container->add(object_ptr<Ui::DividerLabel>(
 | |
| 		container,
 | |
| 		object_ptr<Ui::FlatLabel>(
 | |
| 			container,
 | |
| 			tr::lng_username_description(Ui::Text::RichLangValue),
 | |
| 			st::boxDividerLabel),
 | |
| 		st::settingsDividerLabelPadding));
 | |
| 
 | |
| 	const auto list = box->addRow(
 | |
| 		object_ptr<UsernamesList>(
 | |
| 			box,
 | |
| 			session->user(),
 | |
| 			std::make_shared<Ui::BoxShow>(box),
 | |
| 			[=] {
 | |
| 				box->scrollToY(0);
 | |
| 				editor->setInnerFocus();
 | |
| 			}),
 | |
| 		{});
 | |
| 
 | |
| 	const auto finish = [=] {
 | |
| 		list->save(
 | |
| 		) | rpl::start_with_done([=] {
 | |
| 			editor->save(
 | |
| 			) | rpl::start_with_done([=] {
 | |
| 				box->closeBox();
 | |
| 			}, box->lifetime());
 | |
| 		}, box->lifetime());
 | |
| 	};
 | |
| 	editor->submitted(
 | |
| 	) | rpl::start_with_next(finish, editor->lifetime());
 | |
| 
 | |
| 	box->addButton(tr::lng_settings_save(), finish);
 | |
| 	box->addButton(tr::lng_cancel(), [=] { box->closeBox(); });
 | |
| }
 | 
