/* 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 "payments/ui/payments_field.h" #include "ui/widgets/input_fields.h" #include "ui/boxes/country_select_box.h" #include "ui/text/format_values.h" #include "ui/ui_utility.h" #include "ui/special_fields.h" #include "countries/countries_instance.h" #include "base/platform/base_platform_info.h" #include "base/event_filter.h" #include "base/qt_adapters.h" #include "styles/style_payments.h" #include namespace Payments::Ui { namespace { struct SimpleFieldState { QString value; int position = 0; }; [[nodiscard]] char FieldThousandsSeparator(const CurrencyRule &rule) { return (rule.thousands == '.' || rule.thousands == ',') ? ' ' : rule.thousands; } [[nodiscard]] QString RemoveNonNumbers(QString value) { return value.replace(QRegularExpression("[^0-9]"), QString()); } [[nodiscard]] SimpleFieldState CleanMoneyState( const CurrencyRule &rule, SimpleFieldState state) { const auto withDecimal = state.value.replace( QChar('.'), rule.decimal ).replace( QChar(','), rule.decimal ); const auto digitsLimit = 16 - rule.exponent; const auto beforePosition = state.value.mid(0, state.position); auto decimalPosition = int(withDecimal.lastIndexOf(rule.decimal)); if (decimalPosition < 0) { state = { .value = RemoveNonNumbers(state.value), .position = int(RemoveNonNumbers(beforePosition).size()), }; } else { const auto onlyNumbersBeforeDecimal = RemoveNonNumbers( state.value.mid(0, decimalPosition)); state = { .value = (onlyNumbersBeforeDecimal + QChar(rule.decimal) + RemoveNonNumbers(state.value.mid(decimalPosition + 1))), .position = int(RemoveNonNumbers(beforePosition).size() + (state.position > decimalPosition ? 1 : 0)), }; decimalPosition = onlyNumbersBeforeDecimal.size(); const auto maxLength = decimalPosition + 1 + rule.exponent; if (state.value.size() > maxLength) { state = { .value = state.value.mid(0, maxLength), .position = std::min(state.position, maxLength), }; } } if (!state.value.isEmpty() && state.value[0] == QChar(rule.decimal)) { state = { .value = QChar('0') + state.value, .position = state.position + 1, }; if (decimalPosition >= 0) { ++decimalPosition; } } auto skip = 0; while (state.value.size() > skip + 1 && state.value[skip] == QChar('0') && state.value[skip + 1] != QChar(rule.decimal)) { ++skip; } state = { .value = state.value.mid(skip), .position = std::max(state.position - skip, 0), }; if (decimalPosition >= 0) { Assert(decimalPosition >= skip); decimalPosition -= skip; if (decimalPosition > digitsLimit) { state = { .value = (state.value.mid(0, digitsLimit) + state.value.mid(decimalPosition)), .position = (state.position > digitsLimit ? std::max( state.position - (decimalPosition - digitsLimit), digitsLimit) : state.position), }; } } else if (state.value.size() > digitsLimit) { state = { .value = state.value.mid(0, digitsLimit), .position = std::min(state.position, digitsLimit), }; } return state; } [[nodiscard]] SimpleFieldState PostprocessMoneyResult( const CurrencyRule &rule, SimpleFieldState result) { const auto position = result.value.indexOf(rule.decimal); const auto from = (position >= 0) ? position : result.value.size(); for (auto insertAt = from - 3; insertAt > 0; insertAt -= 3) { result.value.insert(insertAt, QChar(FieldThousandsSeparator(rule))); if (result.position >= insertAt) { ++result.position; } } return result; } [[nodiscard]] bool IsBackspace(const FieldValidateRequest &request) { return (request.wasAnchor == request.wasPosition) && (request.wasPosition == request.nowPosition + 1) && (base::StringViewMid(request.wasValue, 0, request.wasPosition - 1) == base::StringViewMid(request.nowValue, 0, request.nowPosition)) && (base::StringViewMid(request.wasValue, request.wasPosition) == base::StringViewMid(request.nowValue, request.nowPosition)); } [[nodiscard]] bool IsDelete(const FieldValidateRequest &request) { return (request.wasAnchor == request.wasPosition) && (request.wasPosition == request.nowPosition) && (base::StringViewMid(request.wasValue, 0, request.wasPosition) == base::StringViewMid(request.nowValue, 0, request.nowPosition)) && (base::StringViewMid(request.wasValue, request.wasPosition + 1) == base::StringViewMid(request.nowValue, request.nowPosition)); } [[nodiscard]] auto MoneyValidator(const CurrencyRule &rule) { return [=](FieldValidateRequest request) { const auto realNowState = [&] { const auto backspaced = IsBackspace(request); const auto deleted = IsDelete(request); if (!backspaced && !deleted) { return CleanMoneyState(rule, { .value = request.nowValue, .position = request.nowPosition, }); } const auto realWasState = CleanMoneyState(rule, { .value = request.wasValue, .position = request.wasPosition, }); const auto changedValue = deleted ? (realWasState.value.mid(0, realWasState.position) + realWasState.value.mid(realWasState.position + 1)) : (realWasState.position > 1) ? (realWasState.value.mid(0, realWasState.position - 1) + realWasState.value.mid(realWasState.position)) : realWasState.value.mid(realWasState.position); return SimpleFieldState{ .value = changedValue, .position = (deleted ? realWasState.position : std::max(realWasState.position - 1, 0)) }; }(); const auto postprocessed = PostprocessMoneyResult( rule, realNowState); return FieldValidateResult{ .value = postprocessed.value, .position = postprocessed.position, }; }; } [[nodiscard]] QString Parse(const FieldConfig &config) { if (config.type == FieldType::Country) { return Countries::Instance().countryNameByISO2(config.value); } else if (config.type == FieldType::Money) { const auto amount = config.value.toLongLong(); if (!amount) { return QString(); } const auto rule = LookupCurrencyRule(config.currency); const auto value = std::abs(amount) / std::pow(10., rule.exponent); const auto precision = (!rule.stripDotZero || std::floor(value) != value) ? rule.exponent : 0; return FormatWithSeparators( value, precision, rule.decimal, FieldThousandsSeparator(rule)); } return config.value; } [[nodiscard]] QString Format( const FieldConfig &config, const QString &parsed, const QString &countryIso2) { if (config.type == FieldType::Country) { return countryIso2; } else if (config.type == FieldType::Money) { const auto rule = LookupCurrencyRule(config.currency); const auto real = QString(parsed).replace( QChar(rule.decimal), QChar('.') ).replace( QChar(','), QChar('.') ).replace( QRegularExpression("[^0-9\\.]"), QString() ).toDouble(); return QString::number( int64(base::SafeRound(real * std::pow(10., rule.exponent)))); } else if (config.type == FieldType::CardNumber || config.type == FieldType::CardCVC) { return QString(parsed).replace( QRegularExpression("[^0-9\\.]"), QString()); } return parsed; } [[nodiscard]] bool UseMaskedField(FieldType type) { switch (type) { case FieldType::Text: case FieldType::Email: return false; case FieldType::CardNumber: case FieldType::CardExpireDate: case FieldType::CardCVC: case FieldType::Country: case FieldType::Phone: case FieldType::Money: return true; } Unexpected("FieldType in Payments::Ui::UseMaskedField."); } [[nodiscard]] base::unique_qptr CreateWrap( QWidget *parent, FieldConfig &config) { switch (config.type) { case FieldType::Text: case FieldType::Email: return base::make_unique_q( parent, st::paymentsField, std::move(config.placeholder), Parse(config)); case FieldType::CardNumber: case FieldType::CardExpireDate: case FieldType::CardCVC: case FieldType::Country: case FieldType::Phone: case FieldType::Money: return base::make_unique_q(parent); } Unexpected("FieldType in Payments::Ui::CreateWrap."); } [[nodiscard]] InputField *LookupInputField( not_null wrap, FieldConfig &config) { return UseMaskedField(config.type) ? nullptr : static_cast(wrap.get()); } [[nodiscard]] MaskedInputField *CreateMoneyField( not_null wrap, FieldConfig &config, rpl::producer<> textPossiblyChanged) { struct State { CurrencyRule rule; style::InputField st; QString currencyText; int currencySkip = 0; FlatLabel *left = nullptr; FlatLabel *right = nullptr; }; const auto state = wrap->lifetime().make_state(State{ .rule = LookupCurrencyRule(config.currency), .st = st::paymentsMoneyField, }); const auto &rule = state->rule; state->currencySkip = rule.space ? state->st.font->spacew : 0; state->currencyText = ((!rule.left && rule.space) ? QString(QChar(' ')) : QString()) + (*rule.international ? QString(rule.international) : config.currency) + ((rule.left && rule.space) ? QString(QChar(' ')) : QString()); if (rule.left) { state->left = CreateChild( wrap.get(), state->currencyText, st::paymentsFieldAdditional); } state->right = CreateChild( wrap.get(), QString(), st::paymentsFieldAdditional); const auto leftSkip = state->left ? (state->left->naturalWidth() + state->currencySkip) : 0; const auto rightSkip = st::paymentsFieldAdditional.style.font->width( QString(QChar(rule.decimal)) + QString(QChar('0')).repeated(rule.exponent) + (rule.left ? QString() : state->currencyText)); state->st.textMargins += QMargins(leftSkip, 0, rightSkip, 0); state->st.placeholderMargins -= QMargins(leftSkip, 0, rightSkip, 0); const auto result = CreateChild( wrap.get(), state->st, std::move(config.placeholder), Parse(config)); result->setPlaceholderHidden(true); if (state->left) { state->left->move(0, state->st.textMargins.top()); } const auto updateRight = [=] { const auto text = result->getLastText(); const auto width = state->st.font->width(text); const auto &rule = state->rule; const auto symbol = QChar(rule.decimal); const auto decimal = text.indexOf(symbol); const auto zeros = (decimal >= 0) ? std::max(rule.exponent - int(text.size() - decimal - 1), 0) : rule.stripDotZero ? 0 : rule.exponent; const auto valueDecimalSeparator = (decimal >= 0 || !zeros) ? QString() : QString(symbol); const auto zeroString = QString(QChar('0')); const auto valueRightPart = (text.isEmpty() ? zeroString : QString()) + valueDecimalSeparator + zeroString.repeated(zeros); const auto right = valueRightPart + (rule.left ? QString() : state->currencyText); state->right->setText(right); state->right->setTextColorOverride(valueRightPart.isEmpty() ? std::nullopt : std::make_optional(st::windowSubTextFg->c)); state->right->move( (state->st.textMargins.left() + width + ((rule.left || !valueRightPart.isEmpty()) ? 0 : state->currencySkip)), state->st.textMargins.top()); }; std::move( textPossiblyChanged ) | rpl::start_with_next(updateRight, result->lifetime()); if (state->left) { state->left->raise(); } state->right->raise(); return result; } [[nodiscard]] MaskedInputField *LookupMaskedField( not_null wrap, FieldConfig &config, rpl::producer<> textPossiblyChanged) { if (!UseMaskedField(config.type)) { return nullptr; } switch (config.type) { case FieldType::Text: case FieldType::Email: return nullptr; case FieldType::CardNumber: case FieldType::CardExpireDate: case FieldType::CardCVC: case FieldType::Country: return CreateChild( wrap.get(), st::paymentsField, std::move(config.placeholder), Parse(config)); case FieldType::Phone: return CreateChild( wrap.get(), st::paymentsField, std::move(config.placeholder), Countries::ExtractPhoneCode(config.defaultPhone), Parse(config)); case FieldType::Money: return CreateMoneyField( wrap, config, std::move(textPossiblyChanged)); } Unexpected("FieldType in Payments::Ui::LookupMaskedField."); } } // namespace Field::Field(QWidget *parent, FieldConfig &&config) : _config(config) , _wrap(CreateWrap(parent, config)) , _input(LookupInputField(_wrap.get(), config)) , _masked(LookupMaskedField( _wrap.get(), config, _textPossiblyChanged.events_starting_with({}))) , _countryIso2(config.value) { if (_masked) { setupMaskedGeometry(); } if (_config.type == FieldType::Country) { setupCountry(); } if (const auto &validator = config.validator) { setupValidator(validator); } else if (config.type == FieldType::Money) { setupValidator(MoneyValidator(LookupCurrencyRule(config.currency))); } setupFrontBackspace(); setupSubmit(); } RpWidget *Field::widget() const { return _wrap.get(); } object_ptr Field::ownedWidget() const { return object_ptr::fromRaw(_wrap.get()); } QString Field::value() const { return Format( _config, _input ? _input->getLastText() : _masked->getLastText(), _countryIso2); } rpl::producer<> Field::frontBackspace() const { return _frontBackspace.events(); } rpl::producer<> Field::finished() const { return _finished.events(); } rpl::producer<> Field::submitted() const { return _submitted.events(); } void Field::setupMaskedGeometry() { Expects(_masked != nullptr); _wrap->resize(_masked->size()); _wrap->widthValue( ) | rpl::start_with_next([=](int width) { _masked->resize(width, _masked->height()); }, _masked->lifetime()); _masked->heightValue( ) | rpl::start_with_next([=](int height) { _wrap->resize(_wrap->width(), height); }, _masked->lifetime()); } void Field::setupCountry() { Expects(_config.type == FieldType::Country); Expects(_masked != nullptr); QObject::connect(_masked, &MaskedInputField::focused, [=] { setFocus(); const auto name = Countries::Instance().countryNameByISO2( _countryIso2); const auto country = !name.isEmpty() ? _countryIso2 : !_config.defaultCountry.isEmpty() ? _config.defaultCountry : Platform::SystemCountry(); auto box = Box( country, CountrySelectBox::Type::Countries); const auto raw = box.data(); raw->countryChosen( ) | rpl::start_with_next([=](QString iso2) { _countryIso2 = iso2; _masked->setText(Countries::Instance().countryNameByISO2(iso2)); _masked->hideError(); raw->closeBox(); if (!iso2.isEmpty()) { if (_nextField) { _nextField->activate(); } else { _submitted.fire({}); } } }, _masked->lifetime()); raw->boxClosing() | rpl::start_with_next([=] { setFocus(); }, _masked->lifetime()); _config.showBox(std::move(box)); }); } void Field::setupValidator(Fn validator) { Expects(validator != nullptr); const auto state = [=]() -> State { if (_masked) { const auto position = _masked->cursorPosition(); const auto selectionStart = _masked->selectionStart(); const auto selectionEnd = _masked->selectionEnd(); return { .value = _masked->getLastText(), .position = position, .anchor = (selectionStart == selectionEnd ? position : (selectionStart == position) ? selectionEnd : selectionStart), }; } const auto cursor = _input->textCursor(); return { .value = _input->getLastText(), .position = cursor.position(), .anchor = cursor.anchor(), }; }; const auto save = [=] { _was = state(); }; const auto setText = [=](const QString &text) { if (_masked) { _masked->setText(text); } else { _input->setText(text); } }; const auto setPosition = [=](int position) { if (_masked) { _masked->setCursorPosition(position); } else { auto cursor = _input->textCursor(); cursor.setPosition(position); _input->setTextCursor(cursor); } }; const auto validate = [=] { if (_validating) { return; } _validating = true; const auto guard = gsl::finally([&] { _validating = false; save(); _textPossiblyChanged.fire({}); }); const auto now = state(); const auto result = validator(ValidateRequest{ .wasValue = _was.value, .wasPosition = _was.position, .wasAnchor = _was.anchor, .nowValue = now.value, .nowPosition = now.position, }); _valid = result.finished || !result.invalid; const auto changed = (result.value != now.value); if (changed) { setText(result.value); } if (changed || result.position != now.position) { setPosition(result.position); } if (result.finished) { _finished.fire({}); } else if (result.invalid) { Ui::PostponeCall( _masked ? (QWidget*)_masked : _input, [=] { showErrorNoFocus(); }); } }; if (_masked) { QObject::connect(_masked, &QLineEdit::cursorPositionChanged, save); QObject::connect(_masked, &MaskedInputField::changed, validate); } else { const auto raw = _input->rawTextEdit(); QObject::connect(raw, &QTextEdit::cursorPositionChanged, save); QObject::connect(_input, &InputField::changed, validate); } } void Field::setupFrontBackspace() { const auto filter = [=](not_null e) { const auto frontBackspace = (e->type() == QEvent::KeyPress) && (static_cast(e.get())->key() == Qt::Key_Backspace) && (_masked ? (_masked->cursorPosition() == 0 && _masked->selectionLength() == 0) : (_input->textCursor().position() == 0 && _input->textCursor().anchor() == 0)); if (frontBackspace) { _frontBackspace.fire({}); } return base::EventFilterResult::Continue; }; if (_masked) { base::install_event_filter(_masked, filter); } else { base::install_event_filter(_input->rawTextEdit(), filter); } } void Field::setupSubmit() { const auto submitted = [=] { if (!_valid) { showError(); } else if (_nextField) { _nextField->activate(); } else { _submitted.fire({}); } }; if (_masked) { QObject::connect(_masked, &MaskedInputField::submitted, submitted); } else { QObject::connect(_input, &InputField::submitted, submitted); } } void Field::setNextField(not_null field) { _nextField = field; finished() | rpl::start_with_next([=] { field->setFocus(); }, _masked ? _masked->lifetime() : _input->lifetime()); } void Field::setPreviousField(not_null field) { frontBackspace( ) | rpl::start_with_next([=] { field->setFocus(); }, _masked ? _masked->lifetime() : _input->lifetime()); } void Field::activate() { if (_input) { _input->setFocus(); } else { _masked->setFocus(); } } void Field::setFocus() { if (_config.type == FieldType::Country) { _wrap->setFocus(); } else { activate(); } } void Field::setFocusFast() { if (_config.type == FieldType::Country) { setFocus(); } else if (_input) { _input->setFocusFast(); } else { _masked->setFocusFast(); } } void Field::showError() { if (_config.type == FieldType::Country) { setFocus(); _masked->showErrorNoFocus(); } else if (_input) { _input->showError(); } else { _masked->showError(); } } void Field::showErrorNoFocus() { if (_input) { _input->showErrorNoFocus(); } else { _masked->showErrorNoFocus(); } } } // namespace Payments::Ui