/* 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/peers/edit_contact_box.h" #include "boxes/generic_box.h" #include "data/data_user.h" #include "data/data_session.h" #include "ui/wrap/vertical_layout.h" #include "ui/widgets/labels.h" #include "ui/widgets/input_fields.h" #include "ui/text/text_utilities.h" #include "info/profile/info_profile_cover.h" #include "lang/lang_keys.h" #include "window/window_controller.h" #include "ui/toast/toast.h" #include "auth_session.h" #include "apiwrap.h" #include "styles/style_boxes.h" #include "styles/style_info.h" namespace { QString UserPhone(not_null user) { const auto phone = user->phone(); return phone.isEmpty() ? user->owner().findContactPhone(user->bareId()) : phone; } class Builder { public: Builder( not_null box, not_null window, not_null user); void build(); private: void setupContent(); void setupCover(); void setupNameFields(); void setupWarning(); void initNameFields( not_null first, not_null last, bool inverted); not_null _box; not_null _window; not_null _user; QString _phone; Fn _focus; Fn _save; }; Builder::Builder( not_null box, not_null window, not_null user) : _box(box) , _window(window) , _user(user) , _phone(UserPhone(user)) { } void Builder::build() { const auto box = _box; setupContent(); box->setTitle(langFactory(_user->isContact() ? lng_edit_contact_title : lng_enter_contact_data)); box->addButton(langFactory(lng_box_done), _save); box->addButton(langFactory(lng_cancel), [=] { box->closeBox(); }); box->setFocusCallback(_focus); } void Builder::setupContent() { setupCover(); setupNameFields(); setupWarning(); } void Builder::setupCover() { _box->addRow( object_ptr( _box, _user, _window->sessionController(), (_phone.isEmpty() ? Lang::Viewer(lng_contact_mobile_hidden) : rpl::single(App::formatPhone(_phone)))), style::margins())->setAttribute(Qt::WA_TransparentForMouseEvents); } void Builder::setupNameFields() { const auto inverted = langFirstNameGoesSecond(); const auto first = _box->addRow( object_ptr( _box, st::defaultInputField, langFactory(lng_signup_firstname), _user->firstName), st::addContactFieldMargin); auto preparedLast = object_ptr( _box, st::defaultInputField, langFactory(lng_signup_lastname), _user->lastName); const auto last = inverted ? _box->insertRow( _box->rowsCount() - 1, std::move(preparedLast), st::addContactFieldMargin) : _box->addRow(std::move(preparedLast), st::addContactFieldMargin); initNameFields(first, last, inverted); } void Builder::initNameFields( not_null first, not_null last, bool inverted) { const auto box = _box; const auto phone = _phone; const auto user = _user; const auto getValue = [](not_null field) { return TextUtilities::SingleLine(field->getLastText()).trimmed(); }; if (inverted) { box->setTabOrder(last, first); } const auto focus = [=] { const auto firstValue = getValue(first); const auto lastValue = getValue(last); const auto empty = firstValue.isEmpty() && lastValue.isEmpty(); const auto focusFirst = (inverted != empty); (focusFirst ? first : last)->setFocusFast(); }; const auto save = [=] { const auto firstValue = getValue(first); const auto lastValue = getValue(last); const auto empty = firstValue.isEmpty() && lastValue.isEmpty(); if (empty) { focus(); (inverted ? last : first)->showError(); return; } const auto wasContact = user->isContact(); const auto weak = make_weak(box); user->session().api().request(MTPcontacts_AddContact( MTP_flags(0), user->inputUser, MTP_string(firstValue), MTP_string(lastValue), MTP_string(phone) )).done([=](const MTPUpdates &result) { user->setName( firstValue, lastValue, user->nameOrPhone, user->username); user->session().api().applyUpdates(result); if (const auto settings = user->settings()) { using Flag = MTPDpeerSettings::Flag; const auto flags = Flag::f_add_contact | Flag::f_block_contact | Flag::f_report_spam; user->setSettings(*settings & ~flags); } if (weak) { weak->closeBox(); } if (!wasContact) { Ui::Toast::Show( lng_new_contact_add_done(lt_user, firstValue)); } }).fail([=](const RPCError &error) { }).send(); }; const auto submit = [=] { const auto firstValue = first->getLastText().trimmed(); const auto lastValue = last->getLastText().trimmed(); const auto empty = firstValue.isEmpty() && lastValue.isEmpty(); if (inverted ? last->hasFocus() : empty) { first->setFocus(); } else if (inverted ? empty : first->hasFocus()) { last->setFocus(); } else { _save(); } }; QObject::connect(first, &Ui::InputField::submitted, [=] { submit(); }); QObject::connect(last, &Ui::InputField::submitted, [=] { submit(); }); _focus = focus; _save = save; } void Builder::setupWarning() { if (_user->isContact()) { return; } const auto name = _user->shortName(); const auto nameWithEntities = TextWithEntities{ name }; const auto text = _phone.isEmpty() ? lng_contact_phone_after__rich( lt_user, nameWithEntities, lt_visible, Ui::Text::Bold(lang(lng_contact_phone_visible)), lt_name, nameWithEntities) : lng_contact_phone_show__rich( lt_button, Ui::Text::Bold(lang(lng_box_done).toUpper()), lt_user, TextWithEntities{ name }); _box->addRow( object_ptr( _box, rpl::single(text), st::changePhoneLabel), st::addContactWarningMargin); } } // namespace void EditContactBox( not_null box, not_null window, not_null user) { Builder(box, window, user).build(); }