diff --git a/Telegram/Resources/langs/lang.strings b/Telegram/Resources/langs/lang.strings index ba09ba80b..fa3c4c5ae 100644 --- a/Telegram/Resources/langs/lang.strings +++ b/Telegram/Resources/langs/lang.strings @@ -1548,6 +1548,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL "lng_passport_scan_index" = "Scan {index}"; "lng_passport_upload_scans" = "Upload scans"; "lng_passport_upload_more" = "Upload additional scans"; +"lng_passport_selfie_title" = "Selfie"; +"lng_passport_selfie_name" = "Photo"; +"lng_passport_selfie_description" = "Take a picture of yourself holding hour document."; +"lng_passport_upload_selfie" = "Upload selfie"; "lng_passport_personal_details" = "Personal details"; "lng_passport_choose_image" = "Choose scan image"; "lng_passport_delete_scan_undo" = "Undo"; diff --git a/Telegram/SourceFiles/passport/passport_form_controller.cpp b/Telegram/SourceFiles/passport/passport_form_controller.cpp index 1b60493b9..2554de6fe 100644 --- a/Telegram/SourceFiles/passport/passport_form_controller.cpp +++ b/Telegram/SourceFiles/passport/passport_form_controller.cpp @@ -242,6 +242,18 @@ bool FormController::validateValueSecrets(Value &value) { return false; } } + if (value.selfie) { + auto &file = *value.selfie; + file.secret = DecryptValueSecret( + file.encryptedSecret, + _secret, + file.hash); + if (file.secret.empty()) { + LOG(("API Error: Could not decrypt selfie secret. " + "Forgetting files and data :(")); + return false; + } + } return true; } @@ -266,8 +278,53 @@ void FormController::uploadScan( nonconst, File(), nullptr); - const auto fileId = rand_value(); auto &file = nonconst->filesInEdit.back(); + encryptFile(file, std::move(content), [=](UploadScanData &&result) { + Expects(fileIndex >= 0 && fileIndex < nonconst->filesInEdit.size()); + + uploadEncryptedFile( + nonconst->filesInEdit[fileIndex], + std::move(result)); + }); +} + +void FormController::deleteScan( + not_null value, + int fileIndex) { + scanDeleteRestore(value, fileIndex, true); +} + +void FormController::restoreScan( + not_null value, + int fileIndex) { + scanDeleteRestore(value, fileIndex, false); +} + +void FormController::uploadSelfie( + not_null value, + QByteArray &&content) { + const auto nonconst = findValue(value); + nonconst->selfieInEdit = EditFile{ nonconst, File(), nullptr }; + auto &file = *nonconst->selfieInEdit; + encryptFile(file, std::move(content), [=](UploadScanData &&result) { + uploadEncryptedFile( + *nonconst->selfieInEdit, + std::move(result)); + }); +} + +void FormController::deleteSelfie(not_null value) { + selfieDeleteRestore(value, true); +} + +void FormController::restoreSelfie(not_null value) { + selfieDeleteRestore(value, false); +} + +void FormController::prepareFile( + EditFile &file, + const QByteArray &content) { + const auto fileId = rand_value(); file.fields.size = content.size(); file.fields.id = fileId; file.fields.dcId = MTP::maindc(); @@ -277,17 +334,14 @@ void FormController::uploadScan( file.fields.downloadOffset = file.fields.size; _scanUpdated.fire(&file); - - encryptScan(nonconst, fileIndex, std::move(content)); } -void FormController::encryptScan( - not_null value, - int fileIndex, - QByteArray &&content) { - Expects(fileIndex >= 0 && fileIndex < value->filesInEdit.size()); +void FormController::encryptFile( + EditFile &file, + QByteArray &&content, + base::lambda callback) { + prepareFile(file, content); - const auto &file = value->filesInEdit[fileIndex]; const auto weak = std::weak_ptr(file.guard); crl::async([ =, @@ -309,27 +363,12 @@ void FormController::encryptScan( result.md5checksum.data()); crl::on_main([=, encrypted = std::move(result)]() mutable { if (weak.lock()) { - uploadEncryptedScan( - value , - fileIndex, - std::move(encrypted)); + callback(std::move(encrypted)); } }); }); } -void FormController::deleteScan( - not_null value, - int fileIndex) { - scanDeleteRestore(value, fileIndex, true); -} - -void FormController::restoreScan( - not_null value, - int fileIndex) { - scanDeleteRestore(value, fileIndex, false); -} - void FormController::scanDeleteRestore( not_null value, int fileIndex, @@ -342,6 +381,17 @@ void FormController::scanDeleteRestore( _scanUpdated.fire(&file); } +void FormController::selfieDeleteRestore( + not_null value, + bool deleted) { + Expects(value->selfieInEdit.has_value()); + + const auto nonconst = findValue(value); + auto &file = *nonconst->selfieInEdit; + file.deleted = deleted; + _scanUpdated.fire(&file); +} + void FormController::subscribeToUploader() { if (_uploaderSubscriptions) { return; @@ -365,15 +415,11 @@ void FormController::subscribeToUploader() { }, _uploaderSubscriptions); } -void FormController::uploadEncryptedScan( - not_null value, - int fileIndex, +void FormController::uploadEncryptedFile( + EditFile &file, UploadScanData &&data) { - Expects(fileIndex >= 0 && fileIndex < value->filesInEdit.size()); - subscribeToUploader(); - auto &file = value->filesInEdit[fileIndex]; file.uploadData = std::make_unique(std::move(data)); auto prepared = std::make_shared( @@ -548,53 +594,67 @@ void FormController::startValueEdit(not_null value) { if (savingValue(nonconst)) { return; } - loadFiles(nonconst->files); + for (auto &file : nonconst->files) { + loadFile(file); + } + if (nonconst->selfie) { + loadFile(*nonconst->selfie); + } nonconst->filesInEdit = ranges::view::all( nonconst->files ) | ranges::view::transform([=](const File &file) { return EditFile(nonconst, file, nullptr); }) | ranges::to_vector; + + if (nonconst->selfie) { + nonconst->selfieInEdit = EditFile( + nonconst, + *nonconst->selfie, + nullptr); + } else { + nonconst->selfieInEdit = base::none; + } + nonconst->data.parsedInEdit = nonconst->data.parsed; } -void FormController::loadFiles(std::vector &files) { - for (auto &file : files) { - if (!file.image.isNull()) { - file.downloadOffset = file.size; - continue; - } - - const auto key = FileKey{ file.id, file.dcId }; - const auto i = _fileLoaders.find(key); - if (i == _fileLoaders.end()) { - file.downloadOffset = 0; - const auto [i, ok] = _fileLoaders.emplace( - key, - std::make_unique( - file.dcId, - file.id, - file.accessHash, - 0, - SecureFileLocation, - QString(), - file.size, - LoadToCacheAsWell, - LoadFromCloudOrLocal, - false)); - const auto loader = i->second.get(); - loader->connect(loader, &mtpFileLoader::progress, [=] { - if (loader->finished()) { - fileLoadDone(key, loader->bytes()); - } else { - fileLoadProgress(key, loader->currentOffset()); - } - }); - loader->connect(loader, &mtpFileLoader::failed, [=] { - fileLoadFail(key); - }); - loader->start(); - } +void FormController::loadFile(File &file) { + if (!file.image.isNull()) { + file.downloadOffset = file.size; + return; } + + const auto key = FileKey{ file.id, file.dcId }; + const auto i = _fileLoaders.find(key); + if (i != _fileLoaders.end()) { + return; + } + file.downloadOffset = 0; + const auto [j, ok] = _fileLoaders.emplace( + key, + std::make_unique( + file.dcId, + file.id, + file.accessHash, + 0, + SecureFileLocation, + QString(), + file.size, + LoadToCacheAsWell, + LoadFromCloudOrLocal, + false)); + const auto loader = j->second.get(); + loader->connect(loader, &mtpFileLoader::progress, [=] { + if (loader->finished()) { + fileLoadDone(key, loader->bytes()); + } else { + fileLoadProgress(key, loader->currentOffset()); + } + }); + loader->connect(loader, &mtpFileLoader::failed, [=] { + fileLoadFail(key); + }); + loader->start(); } void FormController::fileLoadDone(FileKey key, const QByteArray &bytes) { @@ -666,6 +726,7 @@ void FormController::clearValueEdit(not_null value) { return; } value->filesInEdit.clear(); + value->selfieInEdit = base::none; value->data.encryptedSecretInEdit.clear(); value->data.hashInEdit.clear(); value->data.parsedInEdit = ValueMap(); @@ -787,7 +848,6 @@ void FormController::saveEncryptedValue(not_null value) { inputFiles.push_back(inputFile(file)); } - if (value->data.secret.empty()) { value->data.secret = GenerateSecretBytes(); } @@ -890,9 +950,18 @@ void FormController::sendSaveRequest( value->saveRequestId = 0; const auto &data = result.c_secureValue(); - value->files = parseFiles( - data.vfiles.v, - base::take(value->filesInEdit)); + value->files = data.has_files() + ? parseFiles( + data.vfiles.v, + base::take(value->filesInEdit)) + : std::vector(); + auto selfiesInEdit = std::vector(); + if (auto selfie = base::take(value->selfieInEdit)) { + selfiesInEdit.push_back(std::move(*selfie)); + } + value->selfie = data.has_selfie() + ? parseFile(data.vselfie, selfiesInEdit) + : base::none; value->data.encryptedSecret = std::move( value->data.encryptedSecretInEdit); value->data.parsed = std::move(value->data.parsedInEdit); @@ -1110,32 +1179,40 @@ auto FormController::parseFiles( auto result = std::vector(); result.reserve(data.size()); - auto index = 0; for (const auto &file : data) { - switch (file.type()) { - case mtpc_secureFileEmpty: { - - } break; - case mtpc_secureFile: { - const auto &fields = file.c_secureFile(); - auto normal = File(); - normal.id = fields.vid.v; - normal.accessHash = fields.vaccess_hash.v; - normal.size = fields.vsize.v; - normal.date = fields.vdate.v; - normal.dcId = fields.vdc_id.v; - normal.hash = bytes::make_vector(fields.vfile_hash.v); - normal.encryptedSecret = bytes::make_vector(fields.vsecret.v); - fillDownloadedFile(normal, editData); - result.push_back(std::move(normal)); - } break; + if (auto normal = parseFile(file, editData)) { + result.push_back(std::move(*normal)); } - ++index; } return result; } +auto FormController::parseFile( + const MTPSecureFile &data, + const std::vector &editData) const +-> base::optional { + switch (data.type()) { + case mtpc_secureFileEmpty: + return base::none; + + case mtpc_secureFile: { + const auto &fields = data.c_secureFile(); + auto result = File(); + result.id = fields.vid.v; + result.accessHash = fields.vaccess_hash.v; + result.size = fields.vsize.v; + result.date = fields.vdate.v; + result.dcId = fields.vdc_id.v; + result.hash = bytes::make_vector(fields.vfile_hash.v); + result.encryptedSecret = bytes::make_vector(fields.vsecret.v); + fillDownloadedFile(result, editData); + return result; + } break; + } + Unexpected("Type in FormController::parseFile."); +} + void FormController::fillDownloadedFile( File &destination, const std::vector &source) const { @@ -1180,6 +1257,9 @@ auto FormController::parseValue( if (data.has_files()) { result.files = parseFiles(data.vfiles.v); } + if (data.has_selfie()) { + result.selfie = parseFile(data.vselfie); + } if (data.has_plain_data()) { switch (data.vplain_data.type()) { case mtpc_securePlainPhone: { @@ -1197,35 +1277,53 @@ auto FormController::parseValue( } auto FormController::findEditFile(const FullMsgId &fullId) -> EditFile* { + const auto found = [&](const EditFile &file) { + return (file.uploadData && file.uploadData->fullId == fullId); + }; for (auto &[type, value] : _form.values) { for (auto &file : value.filesInEdit) { - if (file.uploadData && file.uploadData->fullId == fullId) { + if (found(file)) { return &file; } } + if (value.selfieInEdit && found(*value.selfieInEdit)) { + return &*value.selfieInEdit; + } } return nullptr; } auto FormController::findEditFile(const FileKey &key) -> EditFile* { + const auto found = [&](const EditFile &file) { + return (file.fields.dcId == key.dcId && file.fields.id == key.id); + }; for (auto &[type, value] : _form.values) { for (auto &file : value.filesInEdit) { - if (file.fields.dcId == key.dcId && file.fields.id == key.id) { + if (found(file)) { return &file; } } + if (value.selfieInEdit && found(*value.selfieInEdit)) { + return &*value.selfieInEdit; + } } return nullptr; } auto FormController::findFile(const FileKey &key) -> std::pair { + const auto found = [&](const File &file) { + return (file.dcId == key.dcId) && (file.id == key.id); + }; for (auto &[type, value] : _form.values) { for (auto &file : value.files) { - if (file.dcId == key.dcId && file.id == key.id) { + if (found(file)) { return { &value, &file }; } } + if (value.selfie && found(*value.selfie)) { + return { &value, &*value.selfie }; + } } return { nullptr, nullptr }; } diff --git a/Telegram/SourceFiles/passport/passport_form_controller.h b/Telegram/SourceFiles/passport/passport_form_controller.h index cca4fa2ec..1c32e23f4 100644 --- a/Telegram/SourceFiles/passport/passport_form_controller.h +++ b/Telegram/SourceFiles/passport/passport_form_controller.h @@ -212,6 +212,9 @@ public: void uploadScan(not_null value, QByteArray &&content); void deleteScan(not_null value, int fileIndex); void restoreScan(not_null value, int fileIndex); + void uploadSelfie(not_null value, QByteArray &&content); + void deleteSelfie(not_null value); + void restoreSelfie(not_null value); rpl::producer<> secretReadyEvents() const; @@ -257,6 +260,9 @@ private: std::vector parseFiles( const QVector &data, const std::vector &editData = {}) const; + base::optional parseFile( + const MTPSecureFile &data, + const std::vector &editData = {}) const; void fillDownloadedFile( File &destination, const std::vector &source) const; @@ -275,25 +281,33 @@ private: bool validateValueSecrets(Value &value); void resetValue(Value &value); - void loadFiles(std::vector &files); + void loadFile(File &file); void fileLoadDone(FileKey key, const QByteArray &bytes); void fileLoadProgress(FileKey key, int offset); void fileLoadFail(FileKey key); void generateSecret(bytes::const_span password); void subscribeToUploader(); - void encryptScan( - not_null value, - int fileIndex, - QByteArray &&content); - void uploadEncryptedScan( - not_null value, - int fileIndex, + void encryptFile( + EditFile &file, + QByteArray &&content, + base::lambda callback); + void prepareFile( + EditFile &file, + const QByteArray &content); + void uploadEncryptedFile( + EditFile &file, UploadScanData &&data); void scanUploadDone(const Storage::UploadSecureDone &data); void scanUploadProgress(const Storage::UploadSecureProgress &data); void scanUploadFail(const FullMsgId &fullId); - void scanDeleteRestore(not_null value, int fileIndex, bool deleted); + void scanDeleteRestore( + not_null value, + int fileIndex, + bool deleted); + void selfieDeleteRestore( + not_null value, + bool deleted); QString getPhoneFromValue(not_null value) const; QString getEmailFromValue(not_null value) const; diff --git a/Telegram/SourceFiles/passport/passport_panel_controller.cpp b/Telegram/SourceFiles/passport/passport_panel_controller.cpp index b1b61f7a6..7ecaf5b9f 100644 --- a/Telegram/SourceFiles/passport/passport_panel_controller.cpp +++ b/Telegram/SourceFiles/passport/passport_panel_controller.cpp @@ -356,7 +356,8 @@ QString PanelController::defaultPhoneNumber() const { void PanelController::uploadScan(QByteArray &&content) { Expects(_editScope != nullptr); - Expects(_editScopeFilesIndex >= 0); + Expects(_editScopeFilesIndex >= 0 + && _editScopeFilesIndex < _editScope->files.size()); _form->uploadScan( _editScope->files[_editScopeFilesIndex], @@ -365,7 +366,8 @@ void PanelController::uploadScan(QByteArray &&content) { void PanelController::deleteScan(int fileIndex) { Expects(_editScope != nullptr); - Expects(_editScopeFilesIndex >= 0); + Expects(_editScopeFilesIndex >= 0 + && _editScopeFilesIndex < _editScope->files.size()); _form->deleteScan( _editScope->files[_editScopeFilesIndex], @@ -374,13 +376,45 @@ void PanelController::deleteScan(int fileIndex) { void PanelController::restoreScan(int fileIndex) { Expects(_editScope != nullptr); - Expects(_editScopeFilesIndex >= 0); + Expects(_editScopeFilesIndex >= 0 + && _editScopeFilesIndex < _editScope->files.size()); _form->restoreScan( _editScope->files[_editScopeFilesIndex], fileIndex); } +void PanelController::uploadSelfie(QByteArray &&content) { + Expects(_editScope != nullptr); + Expects(_editScopeFilesIndex >= 0 + && _editScopeFilesIndex < _editScope->files.size()); + Expects(_editScope->selfieRequired); + + _form->uploadSelfie( + _editScope->files[_editScopeFilesIndex], + std::move(content)); +} + +void PanelController::deleteSelfie() { + Expects(_editScope != nullptr); + Expects(_editScopeFilesIndex >= 0 + && _editScopeFilesIndex < _editScope->files.size()); + Expects(_editScope->selfieRequired); + + _form->deleteSelfie( + _editScope->files[_editScopeFilesIndex]); +} + +void PanelController::restoreSelfie() { + Expects(_editScope != nullptr); + Expects(_editScopeFilesIndex >= 0 + && _editScopeFilesIndex < _editScope->files.size()); + Expects(_editScope->selfieRequired); + + _form->restoreSelfie( + _editScope->files[_editScopeFilesIndex]); +} + rpl::producer PanelController::scanUpdated() const { return _form->scanUpdated( ) | rpl::filter([=](not_null file) { @@ -422,11 +456,17 @@ ScanInfo PanelController::collectScanInfo(const EditFile &file) const { return formatDownloadText(0, file.fields.size); } }(); + auto isSelfie = (_editScope != nullptr) + && (_editScopeFilesIndex >= 0) + && (file.value == _editScope->files[_editScopeFilesIndex]) + && (_editScope->files[_editScopeFilesIndex]->selfieInEdit.has_value()) + && (&file == &*_editScope->files[_editScopeFilesIndex]->selfieInEdit); return { FileKey{ file.fields.id, file.fields.dcId }, status, file.fields.image, - file.deleted }; + file.deleted, + isSelfie }; } QString PanelController::getDefaultContactValue(Scope::Type type) const { @@ -594,7 +634,10 @@ void PanelController::editScope(int index, int filesIndex) { _editScope->files[_editScopeFilesIndex]->type), _editScope->fields->data.parsedInEdit, _editScope->files[_editScopeFilesIndex]->data.parsedInEdit, - valueFiles(*_editScope->files[_editScopeFilesIndex])) + valueFiles(*_editScope->files[_editScopeFilesIndex]), + (_editScope->selfieRequired + ? valueSelfie(*_editScope->files[_editScopeFilesIndex]) + : nullptr)) : object_ptr( _panel.get(), this, @@ -735,6 +778,15 @@ std::vector PanelController::valueFiles( return result; } +std::unique_ptr PanelController::valueSelfie( + const Value &value) const { + if (value.selfieInEdit) { + return std::make_unique( + collectScanInfo(*value.selfieInEdit)); + } + return std::make_unique(); +} + void PanelController::cancelValueEdit() { if (const auto scope = base::take(_editScope)) { _form->cancelValueEdit(scope->fields); diff --git a/Telegram/SourceFiles/passport/passport_panel_controller.h b/Telegram/SourceFiles/passport/passport_panel_controller.h index 44939e7ad..b4ae7efc3 100644 --- a/Telegram/SourceFiles/passport/passport_panel_controller.h +++ b/Telegram/SourceFiles/passport/passport_panel_controller.h @@ -20,6 +20,7 @@ struct ScanInfo { QString status; QImage thumb; bool deleted = false; + bool selfie = false; }; @@ -54,6 +55,9 @@ public: void uploadScan(QByteArray &&content); void deleteScan(int fileIndex); void restoreScan(int fileIndex); + void uploadSelfie(QByteArray &&content); + void deleteSelfie(); + void restoreSelfie(); rpl::producer scanUpdated() const; QString defaultEmail() const; @@ -92,6 +96,7 @@ private: void requestScopeFilesType(int index); void cancelValueEdit(); std::vector valueFiles(const Value &value) const; + std::unique_ptr valueSelfie(const Value &value) const; void processValueSaveFinished(not_null value); void processVerificationNeeded(not_null value); diff --git a/Telegram/SourceFiles/passport/passport_panel_edit_document.cpp b/Telegram/SourceFiles/passport/passport_panel_edit_document.cpp index f17920d16..dc1d28217 100644 --- a/Telegram/SourceFiles/passport/passport_panel_edit_document.cpp +++ b/Telegram/SourceFiles/passport/passport_panel_edit_document.cpp @@ -135,7 +135,8 @@ PanelEditDocument::PanelEditDocument( Scheme scheme, const ValueMap &data, const ValueMap &scanData, - std::vector &&files) + std::vector &&files, + std::unique_ptr &&selfie) : _controller(controller) , _scheme(std::move(scheme)) , _scroll(this, st::passportPanelScroll) @@ -145,7 +146,7 @@ PanelEditDocument::PanelEditDocument( this, langFactory(lng_passport_save_value), st::passportPanelSaveValue) { - setupControls(data, &scanData, std::move(files)); + setupControls(data, &scanData, std::move(files), std::move(selfie)); } PanelEditDocument::PanelEditDocument( @@ -162,14 +163,19 @@ PanelEditDocument::PanelEditDocument( this, langFactory(lng_passport_save_value), st::passportPanelSaveValue) { - setupControls(data, nullptr, {}); + setupControls(data, nullptr, {}, nullptr); } void PanelEditDocument::setupControls( const ValueMap &data, const ValueMap *scanData, - std::vector &&files) { - const auto inner = setupContent(data, scanData, std::move(files)); + std::vector &&files, + std::unique_ptr &&selfie) { + const auto inner = setupContent( + data, + scanData, + std::move(files), + std::move(selfie)); using namespace rpl::mappers; @@ -185,7 +191,8 @@ void PanelEditDocument::setupControls( not_null PanelEditDocument::setupContent( const ValueMap &data, const ValueMap *scanData, - std::vector &&files) { + std::vector &&files, + std::unique_ptr &&selfie) { const auto inner = _scroll->setOwnedWidget( object_ptr(this)); _scroll->widthValue( @@ -199,12 +206,14 @@ not_null PanelEditDocument::setupContent( inner, _controller, _scheme.scansHeader, - std::move(files))); + std::move(files), + std::move(selfie))); + } else { + inner->add(object_ptr( + inner, + st::passportFormDividerHeight)); } - inner->add(object_ptr( - inner, - st::passportFormDividerHeight)); inner->add( object_ptr( inner, diff --git a/Telegram/SourceFiles/passport/passport_panel_edit_document.h b/Telegram/SourceFiles/passport/passport_panel_edit_document.h index 1d9b40f83..8044c53e2 100644 --- a/Telegram/SourceFiles/passport/passport_panel_edit_document.h +++ b/Telegram/SourceFiles/passport/passport_panel_edit_document.h @@ -52,7 +52,8 @@ public: Scheme scheme, const ValueMap &data, const ValueMap &scanData, - std::vector &&files); + std::vector &&files, + std::unique_ptr &&selfie); PanelEditDocument( QWidget *parent, not_null controller, @@ -70,11 +71,13 @@ private: void setupControls( const ValueMap &data, const ValueMap *scanData, - std::vector &&files); + std::vector &&files, + std::unique_ptr &&selfie); not_null setupContent( const ValueMap &data, const ValueMap *scanData, - std::vector &&files); + std::vector &&files, + std::unique_ptr &&selfie); void updateControlsGeometry(); Result collect() const; diff --git a/Telegram/SourceFiles/passport/passport_panel_edit_scans.cpp b/Telegram/SourceFiles/passport/passport_panel_edit_scans.cpp index 989cc7407..c046d77ff 100644 --- a/Telegram/SourceFiles/passport/passport_panel_edit_scans.cpp +++ b/Telegram/SourceFiles/passport/passport_panel_edit_scans.cpp @@ -8,6 +8,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL #include "passport/passport_panel_edit_scans.h" #include "passport/passport_panel_controller.h" +#include "passport/passport_panel_details_row.h" #include "info/profile/info_profile_button.h" #include "info/profile/info_profile_values.h" #include "ui/widgets/buttons.h" @@ -185,10 +186,12 @@ EditScans::EditScans( QWidget *parent, not_null controller, const QString &header, - std::vector &&files) + std::vector &&files, + std::unique_ptr &&selfie) : RpWidget(parent) , _controller(controller) , _files(std::move(files)) +, _selfie(std::move(selfie)) , _content(this) { setupContent(header); } @@ -233,6 +236,48 @@ void EditScans::setupContent(const QString &header) { _upload->addClickHandler([=] { chooseScan(); }); + + inner->add(object_ptr( + inner, + st::passportFormDividerHeight)); + + if (_selfie) { + _selfieHeader = inner->add( + object_ptr>( + inner, + object_ptr( + inner, + lang(lng_passport_selfie_title), + Ui::FlatLabel::InitType::Simple, + st::passportFormHeader), + st::passportUploadHeaderPadding)); + _selfieHeader->toggle(_selfie->key.id != 0, anim::type::instant); + _selfieWrap = inner->add(object_ptr(inner)); + if (_selfie->key.id) { + createSelfieRow(*_selfie); + } + _selfieUpload = inner->add( + object_ptr( + inner, + Lang::Viewer( + lng_passport_upload_selfie + ) | Info::Profile::ToUpperValue(), + st::passportUploadButton), + st::passportUploadButtonPadding); + _selfieUpload->addClickHandler([=] { + chooseSelfie(); + }); + + inner->add(object_ptr( + inner, + object_ptr( + _content, + lang(lng_passport_selfie_description), + Ui::FlatLabel::InitType::Simple, + st::passportFormLabel), + st::passportFormLabelPadding)); + } + _controller->scanUpdated( ) | rpl::start_with_next([=](ScanInfo &&info) { updateScan(std::move(info)); @@ -250,11 +295,32 @@ void EditScans::setupContent(const QString &header) { } void EditScans::updateScan(ScanInfo &&info) { + const auto updateRow = [&]( + not_null button, + const ScanInfo &info) { + button->setStatus(info.status); + button->setImage(info.thumb); + button->setDeleted(info.deleted); + }; + if (info.selfie) { + Assert(info.key.id != 0); + Assert(_selfie != nullptr); + if (_selfie->key.id) { + updateRow(_selfieRow->entity(), info); + } else { + createSelfieRow(info); + _selfieWrap->resizeToWidth(width()); + _selfieRow->show(anim::type::normal); + _selfieHeader->show(anim::type::normal); + } + *_selfie = std::move(info); + return; + } const auto i = ranges::find(_files, info.key, [](const ScanInfo &file) { return file.key; }); if (i != _files.end()) { - *i = info; + *i = std::move(info); const auto scan = _rows[i - _files.begin()]->entity(); scan->setStatus(i->status); scan->setImage(i->thumb); @@ -270,20 +336,33 @@ void EditScans::updateScan(ScanInfo &&info) { } } +void EditScans::createSelfieRow(const ScanInfo &info) { + _selfieRow = createScan( + _selfieWrap, + info, + lang(lng_passport_selfie_name)); + const auto row = _selfieRow->entity(); + + row->deleteClicks( + ) | rpl::start_with_next([=] { + _controller->deleteSelfie(); + }, row->lifetime()); + + row->restoreClicks( + ) | rpl::start_with_next([=] { + _controller->restoreSelfie(); + }, row->lifetime()); +} + void EditScans::pushScan(const ScanInfo &info) { const auto index = _rows.size(); - _rows.push_back(base::unique_qptr>( - _wrap->add(object_ptr>( - _wrap, - object_ptr( - _wrap, - st::passportScanRow, - lng_passport_scan_index(lt_index, QString::number(index + 1)), - info.status, - info.deleted))))); + _rows.push_back(createScan( + _wrap, + info, + lng_passport_scan_index(lt_index, QString::number(index + 1)))); _rows.back()->hide(anim::type::instant); + const auto scan = _rows.back()->entity(); - scan->setImage(info.thumb); scan->deleteClicks( ) | rpl::start_with_next([=] { @@ -296,12 +375,35 @@ void EditScans::pushScan(const ScanInfo &info) { }, scan->lifetime()); } +base::unique_qptr> EditScans::createScan( + not_null parent, + const ScanInfo &info, + const QString &name) { + auto result = base::unique_qptr>( + parent->add(object_ptr>( + parent, + object_ptr( + parent, + st::passportScanRow, + name, + info.status, + info.deleted)))); + result->entity()->setImage(info.thumb); + return result; +} + void EditScans::chooseScan() { ChooseScan(base::lambda_guarded(this, [=](QByteArray &&content) { _controller->uploadScan(std::move(content)); })); } +void EditScans::chooseSelfie() { + ChooseScan(base::lambda_guarded(this, [=](QByteArray &&content) { + _controller->uploadSelfie(std::move(content)); + })); +} + void EditScans::ChooseScan(base::lambda callback) { const auto filter = FileDialog::AllFilesFilter() + qsl(";;Image files (*") diff --git a/Telegram/SourceFiles/passport/passport_panel_edit_scans.h b/Telegram/SourceFiles/passport/passport_panel_edit_scans.h index f8771e9f9..f11161b93 100644 --- a/Telegram/SourceFiles/passport/passport_panel_edit_scans.h +++ b/Telegram/SourceFiles/passport/passport_panel_edit_scans.h @@ -36,20 +36,28 @@ public: QWidget *parent, not_null controller, const QString &header, - std::vector &&files); + std::vector &&files, + std::unique_ptr &&selfie); static void ChooseScan(base::lambda callback); private: void setupContent(const QString &header); void chooseScan(); + void chooseSelfie(); void updateScan(ScanInfo &&info); void pushScan(const ScanInfo &info); + void createSelfieRow(const ScanInfo &info); + base::unique_qptr> createScan( + not_null parent, + const ScanInfo &info, + const QString &name); rpl::producer uploadButtonText() const; not_null _controller; std::vector _files; + std::unique_ptr _selfie; object_ptr _content; QPointer> _divider; @@ -59,6 +67,11 @@ private: QPointer _upload; rpl::event_stream> _uploadTexts; + QPointer> _selfieHeader; + QPointer _selfieWrap; + base::unique_qptr> _selfieRow; + QPointer _selfieUpload; + }; } // namespace Passport