Separate form controller from view controller.

This commit is contained in:
John Preston 2018-03-29 23:49:31 +04:00
parent b0a9d26a94
commit a2dabfde56
19 changed files with 668 additions and 302 deletions

View File

@ -94,7 +94,12 @@ public:
setFocus(); setFocus();
} }
base::Observable<void> boxClosing; rpl::producer<> boxClosing() const {
return _boxClosingStream.events();
}
void notifyBoxClosing() {
_boxClosingStream.fire({});
}
void setDelegate(BoxContentDelegate *newDelegate) { void setDelegate(BoxContentDelegate *newDelegate) {
_delegate = newDelegate; _delegate = newDelegate;
@ -201,6 +206,8 @@ private:
object_ptr<QTimer> _draggingScrollTimer = { nullptr }; object_ptr<QTimer> _draggingScrollTimer = { nullptr };
int _draggingScrollDelta = 0; int _draggingScrollDelta = 0;
rpl::event_stream<> _boxClosingStream;
}; };
class AbstractBox class AbstractBox
@ -249,7 +256,7 @@ protected:
_content->setInnerFocus(); _content->setInnerFocus();
} }
void closeHook() override { void closeHook() override {
_content->boxClosing.notify(true); _content->notifyBoxClosing();
} }
private: private:

View File

@ -617,11 +617,13 @@ void SetupChannelBox::prepare() {
rtlupdate(_invitationLink); rtlupdate(_invitationLink);
} }
})); }));
subscribe(boxClosing, [this] {
boxClosing() | rpl::start_with_next([=] {
if (!_existing) { if (!_existing) {
AddParticipantsBoxController::Start(_channel); AddParticipantsBoxController::Start(_channel);
} }
}); }, lifetime());
updateMaxHeight(); updateMaxHeight();
} }

View File

@ -111,11 +111,13 @@ void ConfirmBox::prepare() {
if (!_informative) { if (!_informative) {
addButton([this] { return _cancelText; }, [this] { _cancelled = true; closeBox(); }); addButton([this] { return _cancelText; }, [this] { _cancelled = true; closeBox(); });
} }
subscribe(boxClosing, [this] {
boxClosing() | rpl::start_with_next([=] {
if (!_confirmed && (!_strictCancel || _cancelled) && _cancelledCallback) { if (!_confirmed && (!_strictCancel || _cancelled) && _cancelledCallback) {
_cancelledCallback(); _cancelledCallback();
} }
}); }, lifetime());
textUpdated(); textUpdated();
} }

View File

@ -659,11 +659,13 @@ void EditColorBox::prepare() {
subscribe(_picker->changed(), [=] { updateFromControls(); }); subscribe(_picker->changed(), [=] { updateFromControls(); });
subscribe(_hueSlider->changed(), [=] { updateFromControls(); }); subscribe(_hueSlider->changed(), [=] { updateFromControls(); });
subscribe(_opacitySlider->changed(), [=] { updateFromControls(); }); subscribe(_opacitySlider->changed(), [=] { updateFromControls(); });
subscribe(boxClosing, [=] {
boxClosing() | rpl::start_with_next([=] {
if (_cancelCallback) { if (_cancelCallback) {
_cancelCallback(); _cancelCallback();
} }
}); }, lifetime());
updateFromControls(); updateFromControls();
} }

View File

@ -507,7 +507,7 @@ void AddParticipantsBoxController::Start(
base::flat_set<not_null<UserData*>> &&alreadyIn, base::flat_set<not_null<UserData*>> &&alreadyIn,
bool justCreated) { bool justCreated) {
auto initBox = [channel, justCreated](not_null<PeerListBox*> box) { auto initBox = [channel, justCreated](not_null<PeerListBox*> box) {
auto subscription = std::make_shared<base::Subscription>(); auto subscription = std::make_shared<rpl::lifetime>();
box->addButton(langFactory(lng_participant_invite), [box, channel, subscription] { box->addButton(langFactory(lng_participant_invite), [box, channel, subscription] {
auto rows = box->peerListCollectSelectedRows(); auto rows = box->peerListCollectSelectedRows();
if (!rows.empty()) { if (!rows.empty()) {
@ -528,9 +528,9 @@ void AddParticipantsBoxController::Start(
}); });
box->addButton(langFactory(justCreated ? lng_create_group_skip : lng_cancel), [box] { box->closeBox(); }); box->addButton(langFactory(justCreated ? lng_create_group_skip : lng_cancel), [box] { box->closeBox(); });
if (justCreated) { if (justCreated) {
*subscription = box->boxClosing.add_subscription([channel] { box->boxClosing() | rpl::start_with_next([=] {
Ui::showPeerHistory(channel, ShowAtTheEndMsgId); Ui::showPeerHistory(channel, ShowAtTheEndMsgId);
}); }, *subscription);
} }
}; };
Ui::show(Box<PeerListBox>(std::make_unique<AddParticipantsBoxController>(channel, std::move(alreadyIn)), std::move(initBox))); Ui::show(Box<PeerListBox>(std::make_unique<AddParticipantsBoxController>(channel, std::move(alreadyIn)), std::move(initBox)));

View File

@ -1431,11 +1431,11 @@ void SendFilesBox::prepare() {
setupCaption(); setupCaption();
initSendWay(); initSendWay();
preparePreview(); preparePreview();
subscribe(boxClosing, [this] { boxClosing() | rpl::start_with_next([=] {
if (!_confirmed && _cancelledCallback) { if (!_confirmed && _cancelledCallback) {
_cancelledCallback(); _cancelledCallback();
} }
}); }, lifetime());
} }
void SendFilesBox::initSendWay() { void SendFilesBox::initSendWay() {

View File

@ -274,7 +274,9 @@ void StickersBox::prepare() {
if (_installed.widget()) { if (_installed.widget()) {
connect(_installed.widget(), SIGNAL(draggingScrollDelta(int)), this, SLOT(onDraggingScrollDelta(int))); connect(_installed.widget(), SIGNAL(draggingScrollDelta(int)), this, SLOT(onDraggingScrollDelta(int)));
if (!_megagroupSet) { if (!_megagroupSet) {
subscribe(boxClosing, [this] { saveChanges(); }); boxClosing() | rpl::start_with_next([=] {
saveChanges();
}, lifetime());
} }
} }

View File

@ -41,11 +41,13 @@ void ShowSearchFromBox(
return nullptr; return nullptr;
}; };
if (auto controller = createController()) { if (auto controller = createController()) {
auto subscription = std::make_shared<base::Subscription>(); auto subscription = std::make_shared<rpl::lifetime>();
auto box = Ui::show(Box<PeerListBox>(std::move(controller), [subscription](not_null<PeerListBox*> box) { auto box = Ui::show(Box<PeerListBox>(std::move(controller), [subscription](not_null<PeerListBox*> box) {
box->addButton(langFactory(lng_cancel), [box, subscription] { box->closeBox(); }); box->addButton(langFactory(lng_cancel), [box, subscription] { box->closeBox(); });
}), LayerOption::KeepOther); }), LayerOption::KeepOther);
*subscription = box->boxClosing.add_subscription(std::move(closedCallback)); box->boxClosing() | rpl::start_with_next(
std::move(closedCallback),
*subscription);
} }
} }

View File

@ -7,7 +7,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/ */
#include "passport/passport_edit_identity_box.h" #include "passport/passport_edit_identity_box.h"
#include "passport/passport_form_controller.h" #include "passport/passport_form_view_separate.h"
#include "ui/widgets/input_fields.h" #include "ui/widgets/input_fields.h"
#include "ui/widgets/buttons.h" #include "ui/widgets/buttons.h"
#include "ui/text_options.h" #include "ui/text_options.h"
@ -142,7 +142,7 @@ void ScanButton::paintEvent(QPaintEvent *e) {
IdentityBox::IdentityBox( IdentityBox::IdentityBox(
QWidget*, QWidget*,
not_null<FormController*> controller, not_null<ViewSeparate*> controller,
int valueIndex, int valueIndex,
const IdentityData &data, const IdentityData &data,
std::vector<ScanInfo> &&files) std::vector<ScanInfo> &&files)

View File

@ -16,7 +16,7 @@ class InputField;
namespace Passport { namespace Passport {
class FormController; class ViewSeparate;
struct ScanInfo; struct ScanInfo;
class ScanButton; class ScanButton;
@ -29,7 +29,7 @@ class IdentityBox : public BoxContent {
public: public:
IdentityBox( IdentityBox(
QWidget*, QWidget*,
not_null<FormController*> controller, not_null<ViewSeparate*> controller,
int valueIndex, int valueIndex,
const IdentityData &data, const IdentityData &data,
std::vector<ScanInfo> &&files); std::vector<ScanInfo> &&files);
@ -49,7 +49,7 @@ private:
void updateControlsPosition(); void updateControlsPosition();
void save(); void save();
not_null<FormController*> _controller; not_null<ViewSeparate*> _controller;
int _valueIndex = -1; int _valueIndex = -1;
std::vector<ScanInfo> _files; std::vector<ScanInfo> _files;

View File

@ -7,7 +7,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/ */
#include "passport/passport_form_box.h" #include "passport/passport_form_box.h"
#include "passport/passport_form_controller.h" #include "passport/passport_form_view_separate.h"
#include "passport/passport_form_row.h" #include "passport/passport_form_row.h"
#include "lang/lang_keys.h" #include "lang/lang_keys.h"
#include "ui/widgets/input_fields.h" #include "ui/widgets/input_fields.h"
@ -22,7 +22,7 @@ namespace Passport {
class FormBox::CheckWidget : public Ui::RpWidget { class FormBox::CheckWidget : public Ui::RpWidget {
public: public:
CheckWidget(QWidget *parent, not_null<FormController*> controller); CheckWidget(QWidget *parent, not_null<ViewSeparate*> controller);
void setInnerFocus(); void setInnerFocus();
void submit(); void submit();
@ -34,7 +34,7 @@ private:
void showError(const QString &error); void showError(const QString &error);
void hideError(); void hideError();
not_null<FormController*> _controller; not_null<ViewSeparate*> _controller;
object_ptr<Ui::PasswordInput> _password; object_ptr<Ui::PasswordInput> _password;
object_ptr<Ui::FlatLabel> _hint = { nullptr }; object_ptr<Ui::FlatLabel> _hint = { nullptr };
@ -46,7 +46,7 @@ private:
class FormBox::Inner : public Ui::RpWidget { class FormBox::Inner : public Ui::RpWidget {
public: public:
Inner(QWidget *parent, not_null<FormController*> controller); Inner(QWidget *parent, not_null<ViewSeparate*> controller);
void refresh(); void refresh();
@ -56,14 +56,14 @@ protected:
void paintEvent(QPaintEvent *e) override; void paintEvent(QPaintEvent *e) override;
private: private:
not_null<FormController*> _controller; not_null<ViewSeparate*> _controller;
std::vector<object_ptr<FormRow>> _rows; std::vector<object_ptr<FormRow>> _rows;
}; };
FormBox::CheckWidget::CheckWidget( FormBox::CheckWidget::CheckWidget(
QWidget *parent, QWidget *parent,
not_null<FormController*> controller) not_null<ViewSeparate*> controller)
: RpWidget(parent) : RpWidget(parent)
, _controller(controller) , _controller(controller)
, _password( , _password(
@ -157,7 +157,7 @@ int FormBox::CheckWidget::resizeGetHeight(int newWidth) {
FormBox::Inner::Inner( FormBox::Inner::Inner(
QWidget *parent, QWidget *parent,
not_null<FormController*> controller) not_null<ViewSeparate*> controller)
: RpWidget(parent) : RpWidget(parent)
, _controller(controller) { , _controller(controller) {
refresh(); refresh();
@ -198,7 +198,7 @@ void FormBox::Inner::paintEvent(QPaintEvent *e) {
} }
FormBox::FormBox(QWidget*, not_null<FormController*> controller) FormBox::FormBox(QWidget*, not_null<ViewSeparate*> controller)
: _controller(controller) { : _controller(controller) {
} }

View File

@ -11,11 +11,11 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
namespace Passport { namespace Passport {
class FormController; class ViewSeparate;
class FormBox : public BoxContent { class FormBox : public BoxContent {
public: public:
FormBox(QWidget*, not_null<FormController*> controller); FormBox(QWidget*, not_null<ViewSeparate*> controller);
protected: protected:
void prepare() override; void prepare() override;
@ -29,7 +29,7 @@ private:
void showForm(); void showForm();
void submitForm(); void submitForm();
not_null<FormController*> _controller; not_null<ViewSeparate*> _controller;
object_ptr<Inner> _innerCached = { nullptr }; object_ptr<Inner> _innerCached = { nullptr };
QPointer<CheckWidget> _passwordCheck; QPointer<CheckWidget> _passwordCheck;
QPointer<Inner> _inner; QPointer<Inner> _inner;

View File

@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "passport/passport_form_box.h" #include "passport/passport_form_box.h"
#include "passport/passport_edit_identity_box.h" #include "passport/passport_edit_identity_box.h"
#include "passport/passport_encryption.h" #include "passport/passport_encryption.h"
#include "passport/passport_form_view_separate.h"
#include "boxes/confirm_box.h" #include "boxes/confirm_box.h"
#include "lang/lang_keys.h" #include "lang/lang_keys.h"
#include "base/openssl_help.h" #include "base/openssl_help.h"
@ -42,26 +43,57 @@ FormRequest::FormRequest(
, publicKey(publicKey) { , publicKey(publicKey) {
} }
FormController::UploadScanData::~UploadScanData() { EditFile::EditFile(
if (fullId) {
Auth().uploader().cancel(fullId);
}
}
FormController::EditFile::EditFile(
const File &fields, const File &fields,
std::unique_ptr<UploadScanData> &&uploadData) std::unique_ptr<UploadScanData> &&uploadData)
: fields(std::move(fields)) : fields(std::move(fields))
, uploadData(std::move(uploadData)) { , uploadData(std::move(uploadData))
, guard(std::make_shared<bool>(true)) {
} }
FormController::Value::Value(Type type) : type(type) { UploadScanDataPointer::UploadScanDataPointer(
std::unique_ptr<UploadScanData> &&value)
: _value(std::move(value)) {
}
UploadScanDataPointer::UploadScanDataPointer(
UploadScanDataPointer &&other) = default;
UploadScanDataPointer &UploadScanDataPointer::operator=(
UploadScanDataPointer &&other) = default;
UploadScanDataPointer::~UploadScanDataPointer() {
if (const auto value = _value.get()) {
if (const auto fullId = value->fullId) {
Auth().uploader().cancel(fullId);
}
}
}
UploadScanData *UploadScanDataPointer::get() const {
return _value.get();
}
UploadScanDataPointer::operator UploadScanData*() const {
return _value.get();
}
UploadScanDataPointer::operator bool() const {
return _value.get();
}
UploadScanData *UploadScanDataPointer::operator->() const {
return _value.get();
}
Value::Value(Type type) : type(type) {
} }
FormController::FormController( FormController::FormController(
not_null<Window::Controller*> controller, not_null<Window::Controller*> controller,
const FormRequest &request) const FormRequest &request)
: _controller(controller) : _controller(controller)
, _view(std::make_unique<ViewSeparate>(this))
, _request(request) { , _request(request) {
} }
@ -70,6 +102,10 @@ void FormController::show() {
requestPassword(); requestPassword();
} }
UserData *FormController::bot() const {
return _bot;
}
bytes::vector FormController::passwordHashForAuth( bytes::vector FormController::passwordHashForAuth(
bytes::const_span password) const { bytes::const_span password) const {
return openssl::Sha256(bytes::concatenate( return openssl::Sha256(bytes::concatenate(
@ -214,7 +250,6 @@ QString FormController::passwordHint() const {
} }
void FormController::uploadScan(int valueIndex, QByteArray &&content) { void FormController::uploadScan(int valueIndex, QByteArray &&content) {
Expects(_editBox != nullptr);
Expects(valueIndex >= 0 && valueIndex < _form.rows.size()); Expects(valueIndex >= 0 && valueIndex < _form.rows.size());
auto &value = _form.rows[valueIndex]; auto &value = _form.rows[valueIndex];
@ -232,7 +267,7 @@ void FormController::uploadScan(int valueIndex, QByteArray &&content) {
file.fields.image = ReadImage(bytes::make_span(content)); file.fields.image = ReadImage(bytes::make_span(content));
file.fields.downloadOffset = file.fields.size; file.fields.downloadOffset = file.fields.size;
_scanUpdated.fire(collectScanInfo(file)); _scanUpdated.fire(&file);
encryptScan(valueIndex, fileIndex, std::move(content)); encryptScan(valueIndex, fileIndex, std::move(content));
} }
@ -241,19 +276,18 @@ void FormController::encryptScan(
int valueIndex, int valueIndex,
int fileIndex, int fileIndex,
QByteArray &&content) { QByteArray &&content) {
Expects(_editBox != nullptr);
Expects(valueIndex >= 0 && valueIndex < _form.rows.size()); Expects(valueIndex >= 0 && valueIndex < _form.rows.size());
Expects(fileIndex >= 0 Expects(fileIndex >= 0
&& fileIndex < _form.rows[valueIndex].filesInEdit.size()); && fileIndex < _form.rows[valueIndex].filesInEdit.size());
const auto &value = _form.rows[valueIndex]; const auto &value = _form.rows[valueIndex];
const auto &file = value.filesInEdit[fileIndex].fields; const auto &file = value.filesInEdit[fileIndex];
const auto weak = _editBox; const auto weak = std::weak_ptr<bool>(file.guard);
crl::async([ crl::async([
=, =,
fileId = file.id, fileId = file.fields.id,
bytes = std::move(content), bytes = std::move(content),
fileSecret = file.secret fileSecret = file.fields.secret
] { ] {
auto data = EncryptData( auto data = EncryptData(
bytes::make_span(bytes), bytes::make_span(bytes),
@ -268,7 +302,7 @@ void FormController::encryptScan(
result.bytes.size(), result.bytes.size(),
result.md5checksum.data()); result.md5checksum.data());
crl::on_main([=, encrypted = std::move(result)]() mutable { crl::on_main([=, encrypted = std::move(result)]() mutable {
if (weak) { if (weak.lock()) {
uploadEncryptedScan( uploadEncryptedScan(
valueIndex, valueIndex,
fileIndex, fileIndex,
@ -287,7 +321,7 @@ void FormController::deleteScan(
auto &file = _form.rows[valueIndex].filesInEdit[fileIndex]; auto &file = _form.rows[valueIndex].filesInEdit[fileIndex];
file.deleted = !file.deleted; file.deleted = !file.deleted;
_scanUpdated.fire(collectScanInfo(file)); _scanUpdated.fire(&file);
} }
void FormController::subscribeToUploader() { void FormController::subscribeToUploader() {
@ -317,7 +351,6 @@ void FormController::uploadEncryptedScan(
int valueIndex, int valueIndex,
int fileIndex, int fileIndex,
UploadScanData &&data) { UploadScanData &&data) {
Expects(_editBox != nullptr);
Expects(valueIndex >= 0 && valueIndex < _form.rows.size()); Expects(valueIndex >= 0 && valueIndex < _form.rows.size());
Expects(fileIndex >= 0 Expects(fileIndex >= 0
&& fileIndex < _form.rows[valueIndex].filesInEdit.size()); && fileIndex < _form.rows[valueIndex].filesInEdit.size());
@ -357,7 +390,7 @@ void FormController::scanUploadDone(const Storage::UploadSecureDone &data) {
file->fields.hash); file->fields.hash);
file->uploadData->fullId = FullMsgId(); file->uploadData->fullId = FullMsgId();
_scanUpdated.fire(collectScanInfo(*file)); _scanUpdated.fire(file);
} }
} }
@ -368,7 +401,7 @@ void FormController::scanUploadProgress(
file->uploadData->offset = data.offset; file->uploadData->offset = data.offset;
_scanUpdated.fire(collectScanInfo(*file)); _scanUpdated.fire(file);
} }
} }
@ -378,7 +411,7 @@ void FormController::scanUploadFail(const FullMsgId &fullId) {
file->uploadData->offset = -1; file->uploadData->offset = -1;
_scanUpdated.fire(collectScanInfo(*file)); _scanUpdated.fire(file);
} }
} }
@ -397,46 +430,16 @@ QString FormController::defaultPhoneNumber() const {
return QString(); return QString();
} }
rpl::producer<ScanInfo> FormController::scanUpdated() const { rpl::producer<not_null<const EditFile*>> FormController::scanUpdated() const {
return _scanUpdated.events(); return _scanUpdated.events();
} }
void FormController::fillRows( void FormController::enumerateRows(
base::lambda<void( base::lambda<void(const Value &value)> callback) {
QString title, ranges::for_each(_form.rows, callback);
QString description,
bool ready)> callback) {
for (const auto &value : _form.rows) {
switch (value.type) {
case Value::Type::Identity:
callback(
lang(lng_passport_identity_title),
lang(lng_passport_identity_description),
false);
break;
case Value::Type::Address:
callback(
lang(lng_passport_address_title),
lang(lng_passport_address_description),
false);
break;
case Value::Type::Phone:
callback(
lang(lng_passport_phone_title),
App::self()->phone(),
true);
break;
case Value::Type::Email:
callback(
lang(lng_passport_email_title),
lang(lng_passport_email_description),
false);
break;
}
}
} }
void FormController::editValue(int index) { not_null<Value*> FormController::startValueEdit(int index) {
Expects(index >= 0 && index < _form.rows.size()); Expects(index >= 0 && index < _form.rows.size());
auto &value = _form.rows[index]; auto &value = _form.rows[index];
@ -447,20 +450,7 @@ void FormController::editValue(int index) {
return EditFile(file, nullptr); return EditFile(file, nullptr);
}) | ranges::to_vector; }) | ranges::to_vector;
auto box = [&]() -> object_ptr<BoxContent> { return &value;
switch (value.type) {
case Value::Type::Identity:
return Box<IdentityBox>(
this,
index,
valueDataIdentity(value),
valueFilesIdentity(value));
}
return { nullptr };
}();
if (box) {
_editBox = Ui::show(std::move(box), LayerOption::KeepOther);
}
} }
void FormController::loadFiles(std::vector<File> &files) { void FormController::loadFiles(std::vector<File> &files) {
@ -520,7 +510,7 @@ void FormController::fileLoadDone(FileKey key, const QByteArray &bytes) {
if (const auto fileInEdit = findEditFile(key)) { if (const auto fileInEdit = findEditFile(key)) {
fileInEdit->fields.image = file->image; fileInEdit->fields.image = file->image;
fileInEdit->fields.downloadOffset = file->downloadOffset; fileInEdit->fields.downloadOffset = file->downloadOffset;
_scanUpdated.fire(collectScanInfo(*fileInEdit)); _scanUpdated.fire(fileInEdit);
} }
} }
} }
@ -530,7 +520,7 @@ void FormController::fileLoadProgress(FileKey key, int offset) {
file->downloadOffset = offset; file->downloadOffset = offset;
if (const auto fileInEdit = findEditFile(key)) { if (const auto fileInEdit = findEditFile(key)) {
fileInEdit->fields.downloadOffset = file->downloadOffset; fileInEdit->fields.downloadOffset = file->downloadOffset;
_scanUpdated.fire(collectScanInfo(*fileInEdit)); _scanUpdated.fire(fileInEdit);
} }
} }
} }
@ -540,95 +530,43 @@ void FormController::fileLoadFail(FileKey key) {
file->downloadOffset = -1; file->downloadOffset = -1;
if (const auto fileInEdit = findEditFile(key)) { if (const auto fileInEdit = findEditFile(key)) {
fileInEdit->fields.downloadOffset = file->downloadOffset; fileInEdit->fields.downloadOffset = file->downloadOffset;
_scanUpdated.fire(collectScanInfo(*fileInEdit)); _scanUpdated.fire(fileInEdit);
} }
} }
} }
ScanInfo FormController::collectScanInfo(const EditFile &file) const { void FormController::cancelValueEdit(int index) {
const auto status = [&] {
if (file.deleted) {
return QString("deleted");
} else if (file.fields.accessHash) {
if (file.fields.downloadOffset < 0) {
return QString("download failed");
} else if (file.fields.downloadOffset < file.fields.size) {
return QString("downloading %1 / %2"
).arg(file.fields.downloadOffset
).arg(file.fields.size);
} else {
return QString("uploaded ")
+ langDateTimeFull(ParseDateTime(file.fields.date));
}
} else if (file.uploadData) {
if (file.uploadData->offset < 0) {
return QString("upload failed");
} else if (file.uploadData->fullId) {
return QString("uploading %1 / %2"
).arg(file.uploadData->offset
).arg(file.uploadData->bytes.size());
} else {
return QString("upload ready");
}
} else {
return QString("preparing");
}
}();
return {
FileKey{ file.fields.id, file.fields.dcId },
status,
file.fields.image };
}
IdentityData FormController::valueDataIdentity(const Value &value) const {
const auto &map = value.data.parsed;
auto result = IdentityData();
if (const auto i = map.find(qsl("first_name")); i != map.cend()) {
result.name = i->second;
}
if (const auto i = map.find(qsl("last_name")); i != map.cend()) {
result.surname = i->second;
}
return result;
}
std::vector<ScanInfo> FormController::valueFilesIdentity(
const Value &value) const {
auto result = std::vector<ScanInfo>();
for (const auto &file : value.filesInEdit) {
result.push_back(collectScanInfo(file));
}
return result;
}
void FormController::saveValueIdentity(
int index,
const IdentityData &data) {
Expects(_editBox != nullptr);
Expects(index >= 0 && index < _form.rows.size()); Expects(index >= 0 && index < _form.rows.size());
Expects(_form.rows[index].type == Value::Type::Identity);
_form.rows[index].data.parsed[qsl("first_name")] = data.name; _form.rows[index].filesInEdit.clear();
_form.rows[index].data.parsed[qsl("last_name")] = data.surname;
saveIdentity(index);
} }
void FormController::saveIdentity(int index) { bool FormController::isEncryptedValue(Value::Type type) const {
Expects(index >= 0 && index < _form.rows.size()); return (type == Value::Type::Identity || type == Value::Type::Address);
Expects(_form.rows[index].type == Value::Type::Identity); }
void FormController::saveValueEdit(int index) {
Expects(index >= 0 && index < _form.rows.size());
if (isEncryptedValue(_form.rows[index].type)) {
saveEncryptedValue(index);
} else {
savePlainTextValue(index);
}
}
void FormController::saveEncryptedValue(int index) {
Expects(index >= 0 && index < _form.rows.size());
Expects(isEncryptedValue(_form.rows[index].type));
auto &value = _form.rows[index];
if (_secret.empty()) { if (_secret.empty()) {
_secretCallbacks.push_back([=] { _secretCallbacks.push_back([=] {
saveIdentity(index); saveEncryptedValue(index);
}); });
return; return;
} }
_editBox->closeBox();
auto &value = _form.rows[index];
auto inputFiles = QVector<MTPInputSecureFile>(); auto inputFiles = QVector<MTPInputSecureFile>();
inputFiles.reserve(value.filesInEdit.size()); inputFiles.reserve(value.filesInEdit.size());
for (const auto &file : value.filesInEdit) { for (const auto &file : value.filesInEdit) {
@ -674,15 +612,52 @@ void FormController::saveIdentity(int index) {
value.data.encryptedSecret, value.data.encryptedSecret,
bytes::concatenate(fileHashesSecrets))); bytes::concatenate(fileHashesSecrets)));
const auto wrap = [&] {
switch (value.type) {
case Value::Type::Identity: return MTP_inputSecureValueIdentity;
case Value::Type::Address: return MTP_inputSecureValueAddress;
}
Unexpected("Value type in saveEncryptedValue().");
}();
sendSaveRequest(index, wrap(
MTP_secureData(
MTP_bytes(encryptedData.bytes),
MTP_bytes(value.data.hash),
MTP_bytes(value.data.encryptedSecret)),
MTP_vector<MTPInputSecureFile>(inputFiles),
MTP_bytes(value.consistencyHash)));
}
void FormController::savePlainTextValue(int index) {
Expects(index >= 0 && index < _form.rows.size());
Expects(!isEncryptedValue(_form.rows[index].type));
auto &value = _form.rows[index];
const auto text = value.data.parsed[QString("value")];
QVector<MTPInputSecureFile>();
value.consistencyHash = openssl::Sha256(
bytes::make_span(text.toUtf8()));
const auto wrap = [&] {
switch (value.type) {
case Value::Type::Phone: return MTP_inputSecureValuePhone;
case Value::Type::Email: return MTP_inputSecureValueEmail;
}
Unexpected("Value type in savePlainTextValue().");
}();
sendSaveRequest(index, wrap(
MTP_string(text),
MTP_bytes(value.consistencyHash)));
}
void FormController::sendSaveRequest(
int index,
const MTPInputSecureValue &value) {
Expects(index >= 0 && index < _form.rows.size());
request(MTPaccount_SaveSecureValue( request(MTPaccount_SaveSecureValue(
MTP_inputSecureValueIdentity( value,
MTP_secureData( MTP_long(_secretId)
MTP_bytes(encryptedData.bytes),
MTP_bytes(value.data.hash),
MTP_bytes(value.data.encryptedSecret)),
MTP_vector<MTPInputSecureFile>(inputFiles),
MTP_bytes(value.consistencyHash)),
MTP_long(CountSecureSecretHash(_secret))
)).done([=](const MTPSecureValueSaved &result) { )).done([=](const MTPSecureValueSaved &result) {
Expects(result.type() == mtpc_secureValueSaved); Expects(result.type() == mtpc_secureValueSaved);
@ -844,21 +819,21 @@ void FormController::fillDownloadedFile(
template <typename DataType> template <typename DataType>
auto FormController::parsePlainTextValue( auto FormController::parsePlainTextValue(
Value::Type type, Value::Type type,
const QByteArray &value, const QByteArray &text,
const DataType &data) const -> Value { const DataType &data) const -> Value {
auto result = Value(type); auto result = Value(type);
const auto check = bytes::compare( const auto check = bytes::compare(
bytes::make_span(data.vhash.v), bytes::make_span(data.vhash.v),
openssl::Sha256(bytes::make_span(value))); openssl::Sha256(bytes::make_span(text)));
if (check != 0) { if (check != 0) {
LOG(("API Error: Bad hash for plain text value. " LOG(("API Error: Bad hash for plain text value. "
"Value '%1', hash '%2'" "Value '%1', hash '%2'"
).arg(QString::fromUtf8(value) ).arg(QString::fromUtf8(text)
).arg(Logs::mb(data.vhash.v.data(), data.vhash.v.size()).str() ).arg(Logs::mb(data.vhash.v.data(), data.vhash.v.size()).str()
)); ));
return result; return result;
} }
result.data.parsed[QString("value")] = QString::fromUtf8(value); result.data.parsed[QString("value")] = QString::fromUtf8(text);
if (data.has_verified()) { if (data.has_verified()) {
result.verification = parseVerified(data.vverified); result.verification = parseVerified(data.vverified);
} }
@ -942,7 +917,7 @@ auto FormController::findFile(const FileKey &key)
void FormController::formDone(const MTPaccount_AuthorizationForm &result) { void FormController::formDone(const MTPaccount_AuthorizationForm &result) {
parseForm(result); parseForm(result);
if (!_passwordRequestId) { if (!_passwordRequestId) {
showForm(); _view->showForm();
} }
} }
@ -987,14 +962,6 @@ void FormController::parseForm(const MTPaccount_AuthorizationForm &result) {
_bot = App::userLoaded(_request.botId); _bot = App::userLoaded(_request.botId);
} }
void FormController::showForm() {
if (!_bot) {
Ui::show(Box<InformBox>("Could not get authorization bot."));
return;
}
Ui::show(Box<FormBox>(this));
}
void FormController::formFail(const RPCError &error) { void FormController::formFail(const RPCError &error) {
Ui::show(Box<InformBox>(lang(lng_passport_form_error))); Ui::show(Box<InformBox>(lang(lng_passport_form_error)));
} }
@ -1020,7 +987,7 @@ void FormController::passwordDone(const MTPaccount_Password &result) {
break; break;
} }
if (!_formRequestId) { if (!_formRequestId) {
showForm(); _view->showForm();
} }
} }

View File

@ -23,6 +23,8 @@ class Controller;
namespace Passport { namespace Passport {
class ViewController;
struct FormRequest { struct FormRequest {
FormRequest( FormRequest(
UserId botId, UserId botId,
@ -37,7 +39,104 @@ struct FormRequest {
}; };
struct IdentityData; struct UploadScanData {
FullMsgId fullId;
uint64 fileId = 0;
int partsCount = 0;
QByteArray md5checksum;
bytes::vector hash;
bytes::vector bytes;
int offset = 0;
};
class UploadScanDataPointer {
public:
UploadScanDataPointer(std::unique_ptr<UploadScanData> &&value);
UploadScanDataPointer(UploadScanDataPointer &&other);
UploadScanDataPointer &operator=(UploadScanDataPointer &&other);
~UploadScanDataPointer();
UploadScanData *get() const;
operator UploadScanData*() const;
explicit operator bool() const;
UploadScanData *operator->() const;
private:
std::unique_ptr<UploadScanData> _value;
};
struct File {
uint64 id = 0;
uint64 accessHash = 0;
int32 size = 0;
int32 dcId = 0;
TimeId date = 0;
bytes::vector hash;
bytes::vector secret;
bytes::vector encryptedSecret;
int downloadOffset = 0;
QImage image;
};
struct EditFile {
EditFile(
const File &fields,
std::unique_ptr<UploadScanData> &&uploadData);
File fields;
UploadScanDataPointer uploadData;
std::shared_ptr<bool> guard;
bool deleted = false;
};
struct Verification {
TimeId date;
};
struct ValueData {
QByteArray original;
std::map<QString, QString> parsed;
bytes::vector hash;
bytes::vector secret;
bytes::vector encryptedSecret;
};
struct Value {
enum class Type {
Identity,
Address,
Phone,
Email,
};
explicit Value(Type type);
Value(Value &&other) = default;
Value &operator=(Value &&other) = default;
Type type;
ValueData data;
std::vector<File> files;
std::vector<EditFile> filesInEdit;
bytes::vector consistencyHash;
base::optional<Verification> verification;
};
struct Form {
std::vector<Value> rows;
};
struct PasswordSettings {
bytes::vector salt;
bytes::vector newSalt;
bytes::vector newSecureSalt;
QString hint;
QString unconfirmedPattern;
QString confirmedEmail;
bool hasRecovery = false;
};
struct FileKey { struct FileKey {
uint64 id = 0; uint64 id = 0;
@ -64,13 +163,6 @@ struct FileKey {
}; };
struct ScanInfo {
FileKey key;
QString status;
QImage thumb;
};
class FormController : private MTP::Sender, public base::has_weak_ptr { class FormController : private MTP::Sender, public base::has_weak_ptr {
public: public:
FormController( FormController(
@ -78,6 +170,7 @@ public:
const FormRequest &request); const FormRequest &request);
void show(); void show();
UserData *bot() const;
void submitPassword(const QString &password); void submitPassword(const QString &password);
rpl::producer<QString> passwordError() const; rpl::producer<QString> passwordError() const;
@ -90,96 +183,21 @@ public:
QString defaultEmail() const; QString defaultEmail() const;
QString defaultPhoneNumber() const; QString defaultPhoneNumber() const;
rpl::producer<ScanInfo> scanUpdated() const;
void fillRows( rpl::producer<not_null<const EditFile*>> scanUpdated() const;
base::lambda<void(
QString title,
QString description,
bool ready)> callback);
void editValue(int index);
void saveValueIdentity(int index, const IdentityData &data); void enumerateRows(base::lambda<void(const Value &value)> callback);
not_null<Value*> startValueEdit(int index);
void cancelValueEdit(int index);
void saveValueEdit(int index);
rpl::lifetime &lifetime() {
return _lifetime;
}
~FormController(); ~FormController();
private: private:
struct UploadScanData {
~UploadScanData();
FullMsgId fullId;
uint64 fileId = 0;
int partsCount = 0;
QByteArray md5checksum;
bytes::vector hash;
bytes::vector bytes;
int offset = 0;
};
struct File {
uint64 id = 0;
uint64 accessHash = 0;
int32 size = 0;
int32 dcId = 0;
TimeId date = 0;
bytes::vector hash;
bytes::vector secret;
bytes::vector encryptedSecret;
int downloadOffset = 0;
QImage image;
};
struct EditFile {
EditFile(
const File &fields,
std::unique_ptr<UploadScanData> &&uploadData);
File fields;
std::unique_ptr<UploadScanData> uploadData;
bool deleted = false;
};
struct Verification {
TimeId date;
};
struct ValueData {
QByteArray original;
std::map<QString, QString> parsed;
bytes::vector hash;
bytes::vector secret;
bytes::vector encryptedSecret;
};
struct Value {
enum class Type {
Identity,
Address,
Phone,
Email,
};
explicit Value(Type type);
Value(Value &&other) = default;
Value &operator=(Value &&other) = default;
Type type;
ValueData data;
std::vector<File> files;
std::vector<EditFile> filesInEdit;
bytes::vector consistencyHash;
base::optional<Verification> verification;
};
struct Form {
std::vector<Value> rows;
};
struct PasswordSettings {
bytes::vector salt;
bytes::vector newSalt;
bytes::vector newSecureSalt;
QString hint;
QString unconfirmedPattern;
QString confirmedEmail;
bool hasRecovery = false;
};
EditFile *findEditFile(const FullMsgId &fullId); EditFile *findEditFile(const FullMsgId &fullId);
EditFile *findEditFile(const FileKey &key); EditFile *findEditFile(const FileKey &key);
std::pair<Value*, File*> findFile(const FileKey &key); std::pair<Value*, File*> findFile(const FileKey &key);
@ -223,10 +241,6 @@ private:
bool validateValueSecrets(Value &value); bool validateValueSecrets(Value &value);
void resetValue(Value &value); void resetValue(Value &value);
IdentityData valueDataIdentity(const Value &value) const;
std::vector<ScanInfo> valueFilesIdentity(const Value &value) const;
void saveIdentity(int index);
void loadFiles(std::vector<File> &files); void loadFiles(std::vector<File> &files);
void fileLoadDone(FileKey key, const QByteArray &bytes); void fileLoadDone(FileKey key, const QByteArray &bytes);
void fileLoadProgress(FileKey key, int offset); void fileLoadProgress(FileKey key, int offset);
@ -245,9 +259,14 @@ private:
void scanUploadDone(const Storage::UploadSecureDone &data); void scanUploadDone(const Storage::UploadSecureDone &data);
void scanUploadProgress(const Storage::UploadSecureProgress &data); void scanUploadProgress(const Storage::UploadSecureProgress &data);
void scanUploadFail(const FullMsgId &fullId); void scanUploadFail(const FullMsgId &fullId);
ScanInfo collectScanInfo(const EditFile &file) const;
bool isEncryptedValue(Value::Type type) const;
void saveEncryptedValue(int index);
void savePlainTextValue(int index);
void sendSaveRequest(int index, const MTPInputSecureValue &value);
not_null<Window::Controller*> _controller; not_null<Window::Controller*> _controller;
std::unique_ptr<ViewController> _view;
FormRequest _request; FormRequest _request;
UserData *_bot = nullptr; UserData *_bot = nullptr;
@ -258,7 +277,8 @@ private:
PasswordSettings _password; PasswordSettings _password;
Form _form; Form _form;
std::map<FileKey, std::unique_ptr<mtpFileLoader>> _fileLoaders; std::map<FileKey, std::unique_ptr<mtpFileLoader>> _fileLoaders;
rpl::event_stream<ScanInfo> _scanUpdated;
rpl::event_stream<not_null<const EditFile*>> _scanUpdated;
bytes::vector _secret; bytes::vector _secret;
uint64 _secretId = 0; uint64 _secretId = 0;
@ -267,9 +287,8 @@ private:
rpl::event_stream<> _secretReady; rpl::event_stream<> _secretReady;
rpl::event_stream<QString> _passwordError; rpl::event_stream<QString> _passwordError;
QPointer<BoxContent> _editBox;
rpl::lifetime _uploaderSubscriptions; rpl::lifetime _uploaderSubscriptions;
rpl::lifetime _lifetime;
}; };

View File

@ -0,0 +1,24 @@
/*
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
*/
#pragma once
namespace Passport {
struct Value;
class ViewController {
public:
virtual void showForm() = 0;
virtual void editValue(int index) = 0;
virtual ~ViewController() {
}
};
} // namespace Passport

View File

@ -0,0 +1,248 @@
/*
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 "passport/passport_form_view_separate.h"
#include "lang/lang_keys.h"
#include "passport/passport_edit_identity_box.h"
#include "passport/passport_form_box.h"
#include "boxes/confirm_box.h"
namespace Passport {
BoxPointer::BoxPointer(QPointer<BoxContent> value)
: _value(value) {
}
BoxPointer::BoxPointer(BoxPointer &&other)
: _value(base::take(other._value)) {
}
BoxPointer &BoxPointer::operator=(BoxPointer &&other) {
std::swap(_value, other._value);
return *this;
}
BoxPointer::~BoxPointer() {
if (const auto strong = get()) {
strong->closeBox();
}
}
BoxContent *BoxPointer::get() const {
return _value.data();
}
BoxPointer::operator BoxContent*() const {
return get();
}
BoxPointer::operator bool() const {
return get();
}
BoxContent *BoxPointer::operator->() const {
return get();
}
ViewSeparate::ViewSeparate(not_null<FormController*> form)
: _form(form) {
}
not_null<UserData*> ViewSeparate::bot() const {
return _form->bot();
}
void ViewSeparate::fillRows(
base::lambda<void(
QString title,
QString description,
bool ready)> callback) {
_form->enumerateRows([&](const Value &value) {
switch (value.type) {
case Value::Type::Identity:
callback(
lang(lng_passport_identity_title),
lang(lng_passport_identity_description),
false);
break;
case Value::Type::Address:
callback(
lang(lng_passport_address_title),
lang(lng_passport_address_description),
false);
break;
case Value::Type::Phone:
callback(
lang(lng_passport_phone_title),
App::self()->phone(),
true);
break;
case Value::Type::Email:
callback(
lang(lng_passport_email_title),
lang(lng_passport_email_description),
false);
break;
}
});
}
void ViewSeparate::submitPassword(const QString &password) {
_form->submitPassword(password);
}
rpl::producer<QString> ViewSeparate::passwordError() const {
return _form->passwordError();
}
QString ViewSeparate::passwordHint() const {
return _form->passwordHint();
}
rpl::producer<> ViewSeparate::secretReadyEvents() const {
return _form->secretReadyEvents();
}
QString ViewSeparate::defaultEmail() const {
return _form->defaultEmail();
}
QString ViewSeparate::defaultPhoneNumber() const {
return _form->defaultPhoneNumber();
}
void ViewSeparate::uploadScan(int valueIndex, QByteArray &&content) {
Expects(_editBox != nullptr);
_form->uploadScan(valueIndex, std::move(content));
}
void ViewSeparate::deleteScan(int valueIndex, int fileIndex) {
Expects(_editBox != nullptr);
_form->deleteScan(valueIndex, fileIndex);
}
rpl::producer<ScanInfo> ViewSeparate::scanUpdated() const {
return _form->scanUpdated(
) | rpl::map([=](not_null<const EditFile*> file) {
return collectScanInfo(*file);
});
}
ScanInfo ViewSeparate::collectScanInfo(const EditFile &file) const {
const auto status = [&] {
if (file.deleted) {
return QString("deleted");
} else if (file.fields.accessHash) {
if (file.fields.downloadOffset < 0) {
return QString("download failed");
} else if (file.fields.downloadOffset < file.fields.size) {
return QString("downloading %1 / %2"
).arg(file.fields.downloadOffset
).arg(file.fields.size);
} else {
return QString("uploaded ")
+ langDateTimeFull(ParseDateTime(file.fields.date));
}
} else if (file.uploadData) {
if (file.uploadData->offset < 0) {
return QString("upload failed");
} else if (file.uploadData->fullId) {
return QString("uploading %1 / %2"
).arg(file.uploadData->offset
).arg(file.uploadData->bytes.size());
} else {
return QString("upload ready");
}
} else {
return QString("preparing");
}
}();
return {
FileKey{ file.fields.id, file.fields.dcId },
status,
file.fields.image };
}
void ViewSeparate::showForm() {
if (!_form->bot()) {
Ui::show(Box<InformBox>("Could not get authorization bot."));
return;
}
Ui::show(Box<FormBox>(this));
}
void ViewSeparate::editValue(int index) {
_editValue = _form->startValueEdit(index);
Assert(_editValue != nullptr);
auto box = [&]() -> object_ptr<BoxContent> {
switch (_editValue->type) {
case Value::Type::Identity:
return Box<IdentityBox>(
this,
index,
valueDataIdentity(*_editValue),
valueFiles(*_editValue));
}
return { nullptr };
}();
if (box) {
_editBox = Ui::show(std::move(box), LayerOption::KeepOther);
_editBox->boxClosing() | rpl::start_with_next([=] {
cancelValueEdit(index);
}, _form->lifetime());
} else {
cancelValueEdit(index);
}
}
IdentityData ViewSeparate::valueDataIdentity(const Value &value) const {
const auto &map = value.data.parsed;
auto result = IdentityData();
if (const auto i = map.find(qsl("first_name")); i != map.cend()) {
result.name = i->second;
}
if (const auto i = map.find(qsl("last_name")); i != map.cend()) {
result.surname = i->second;
}
return result;
}
std::vector<ScanInfo> ViewSeparate::valueFiles(const Value &value) const {
auto result = std::vector<ScanInfo>();
for (const auto &file : value.filesInEdit) {
result.push_back(collectScanInfo(file));
}
return result;
}
void ViewSeparate::cancelValueEdit(int index) {
if (base::take(_editValue)) {
_form->cancelValueEdit(index);
}
}
void ViewSeparate::saveValueIdentity(
int index,
const IdentityData &data) {
Expects(_editBox != nullptr);
Expects(_editValue != nullptr);
Expects(_editValue->type == Value::Type::Identity);
_editValue->data.parsed[qsl("first_name")] = data.name;
_editValue->data.parsed[qsl("last_name")] = data.surname;
_editValue = nullptr;
_editBox->closeBox();
_form->saveValueEdit(index);
}
} // namespace Passport

View File

@ -0,0 +1,86 @@
/*
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
*/
#pragma once
#include "passport/passport_form_view_controller.h"
#include "passport/passport_form_controller.h"
namespace Passport {
class FormController;
struct IdentityData;
struct ScanInfo {
FileKey key;
QString status;
QImage thumb;
};
class BoxPointer {
public:
BoxPointer(QPointer<BoxContent> value = nullptr);
BoxPointer(BoxPointer &&other);
BoxPointer &operator=(BoxPointer &&other);
~BoxPointer();
BoxContent *get() const;
operator BoxContent*() const;
explicit operator bool() const;
BoxContent *operator->() const;
private:
QPointer<BoxContent> _value;
};
class ViewSeparate : public ViewController {
public:
ViewSeparate(not_null<FormController*> form);
not_null<UserData*> bot() const;
void submitPassword(const QString &password);
rpl::producer<QString> passwordError() const;
QString passwordHint() const;
void uploadScan(int valueIndex, QByteArray &&content);
void deleteScan(int valueIndex, int fileIndex);
rpl::producer<ScanInfo> scanUpdated() const;
rpl::producer<> secretReadyEvents() const;
QString defaultEmail() const;
QString defaultPhoneNumber() const;
void showForm() override;
void fillRows(
base::lambda<void(
QString title,
QString description,
bool ready)> callback);
void editValue(int index) override;
void saveValueIdentity(int index, const IdentityData &data);
private:
void cancelValueEdit(int index);
IdentityData valueDataIdentity(const Value &value) const;
std::vector<ScanInfo> valueFiles(const Value &value) const;
ScanInfo collectScanInfo(const EditFile &file) const;
not_null<FormController*> _form;
Value *_editValue = nullptr;
BoxPointer _editBox;
};
} // namespace Passport

View File

@ -377,7 +377,9 @@ void CoverWidget::showSetPhotoBox(const QImage &img) {
std::move(image), std::move(image),
peer->id); peer->id);
}, box->lifetime()); }, box->lifetime());
subscribe(box->boxClosing, [this] { onPhotoUploadStatusChanged(); }); box->boxClosing() | rpl::start_with_next([=] {
onPhotoUploadStatusChanged();
}, lifetime());
} }
void CoverWidget::onPhotoUploadStatusChanged(PeerId peerId) { void CoverWidget::onPhotoUploadStatusChanged(PeerId peerId) {

View File

@ -464,6 +464,9 @@
<(src_loc)/passport/passport_form_controller.h <(src_loc)/passport/passport_form_controller.h
<(src_loc)/passport/passport_form_row.cpp <(src_loc)/passport/passport_form_row.cpp
<(src_loc)/passport/passport_form_row.h <(src_loc)/passport/passport_form_row.h
<(src_loc)/passport/passport_form_view_controller.h
<(src_loc)/passport/passport_form_view_separate.cpp
<(src_loc)/passport/passport_form_view_separate.h
<(src_loc)/platform/linux/linux_desktop_environment.cpp <(src_loc)/platform/linux/linux_desktop_environment.cpp
<(src_loc)/platform/linux/linux_desktop_environment.h <(src_loc)/platform/linux/linux_desktop_environment.h
<(src_loc)/platform/linux/linux_gdk_helper.cpp <(src_loc)/platform/linux/linux_gdk_helper.cpp