Move passcode management from MainWindow.

Check for auto lock in AuthSession. Don't autolock while video plays.
Closes #3219
This commit is contained in:
John Preston 2017-04-15 21:48:54 +03:00
parent e3aacc8072
commit de7c886008
14 changed files with 88 additions and 52 deletions

View File

@ -25,6 +25,13 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "storage/file_download.h" #include "storage/file_download.h"
#include "storage/localstorage.h" #include "storage/localstorage.h"
#include "window/notifications_manager.h" #include "window/notifications_manager.h"
#include "platform/platform_specific.h"
namespace {
constexpr auto kAutoLockTimeoutLateMs = TimeMs(3000);
} // namespace
QByteArray AuthSessionData::serialize() const { QByteArray AuthSessionData::serialize() const {
auto size = sizeof(qint32) * 2; auto size = sizeof(qint32) * 2;
@ -85,11 +92,16 @@ AuthSession::AuthSession(UserId userId)
: _userId(userId) : _userId(userId)
, _api(std::make_unique<ApiWrap>()) , _api(std::make_unique<ApiWrap>())
, _downloader(std::make_unique<Storage::Downloader>()) , _downloader(std::make_unique<Storage::Downloader>())
, _notifications(std::make_unique<Window::Notifications::System>(this)) { , _notifications(std::make_unique<Window::Notifications::System>(this))
, _autoLockTimer([this] { checkAutoLock(); }) {
Expects(_userId != 0); Expects(_userId != 0);
_saveDataTimer.setCallback([this] { _saveDataTimer.setCallback([this] {
Local::writeUserSettings(); Local::writeUserSettings();
}); });
subscribe(Messenger::Instance().passcodedChanged(), [this] {
_shouldLockAt = 0;
notifications().updateAll();
});
} }
bool AuthSession::Exists() { bool AuthSession::Exists() {
@ -127,4 +139,29 @@ void AuthSession::saveDataDelayed(TimeMs delay) {
_saveDataTimer.callOnce(delay); _saveDataTimer.callOnce(delay);
} }
void AuthSession::checkAutoLock() {
if (!Global::LocalPasscode() || App::passcoded()) return;
Messenger::Instance().checkLocalTime();
auto now = getms(true);
auto shouldLockInMs = Global::AutoLock() * 1000LL;
auto idleForMs = psIdleTime();
auto notPlayingVideoForMs = now - data().lastTimeVideoPlayedAt();
auto checkTimeMs = qMin(idleForMs, notPlayingVideoForMs);
if (checkTimeMs >= shouldLockInMs || (_shouldLockAt > 0 && now > _shouldLockAt + kAutoLockTimeoutLateMs)) {
Messenger::Instance().setupPasscode();
} else {
_shouldLockAt = now + (shouldLockInMs - checkTimeMs);
_autoLockTimer.callOnce(shouldLockInMs - checkTimeMs);
}
}
void AuthSession::checkAutoLockIn(TimeMs time) {
if (_autoLockTimer.isActive()) {
auto remain = _autoLockTimer.remainingTime();
if (remain > 0 && remain <= time) return;
}
_autoLockTimer.callOnce(time);
}
AuthSession::~AuthSession() = default; AuthSession::~AuthSession() = default;

View File

@ -79,6 +79,12 @@ public:
void setTabbedSelectorSectionEnabled(bool enabled) { void setTabbedSelectorSectionEnabled(bool enabled) {
_variables.tabbedSelectorSectionEnabled = enabled; _variables.tabbedSelectorSectionEnabled = enabled;
} }
void setLastTimeVideoPlayedAt(TimeMs time) {
_lastTimeVideoPlayedAt = time;
}
TimeMs lastTimeVideoPlayedAt() const {
return _lastTimeVideoPlayedAt;
}
private: private:
struct Variables { struct Variables {
@ -92,10 +98,11 @@ private:
base::Observable<void> _moreChatsLoaded; base::Observable<void> _moreChatsLoaded;
base::Observable<void> _savedGifsUpdated; base::Observable<void> _savedGifsUpdated;
Variables _variables; Variables _variables;
TimeMs _lastTimeVideoPlayedAt = 0;
}; };
class AuthSession final { class AuthSession final : private base::Subscriber {
public: public:
AuthSession(UserId userId); AuthSession(UserId userId);
@ -137,6 +144,9 @@ public:
return *_api; return *_api;
} }
void checkAutoLock();
void checkAutoLockIn(TimeMs time);
~AuthSession(); ~AuthSession();
private: private:
@ -144,6 +154,9 @@ private:
AuthSessionData _data; AuthSessionData _data;
base::Timer _saveDataTimer; base::Timer _saveDataTimer;
TimeMs _shouldLockAt = 0;
base::Timer _autoLockTimer;
const std::unique_ptr<ApiWrap> _api; const std::unique_ptr<ApiWrap> _api;
const std::unique_ptr<Storage::Downloader> _downloader; const std::unique_ptr<Storage::Downloader> _downloader;
const std::unique_ptr<Window::Notifications::System> _notifications; const std::unique_ptr<Window::Notifications::System> _notifications;

View File

@ -52,6 +52,6 @@ void AutoLockBox::durationChanged(int seconds) {
Local::writeUserSettings(); Local::writeUserSettings();
Global::RefLocalPasscodeChanged().notify(); Global::RefLocalPasscodeChanged().notify();
App::wnd()->checkAutoLock(); AuthSession::Current().checkAutoLock();
closeBox(); closeBox();
} }

View File

@ -345,7 +345,7 @@ void PasscodeBox::onSave(bool force) {
} else { } else {
cSetPasscodeBadTries(0); cSetPasscodeBadTries(0);
Local::setPasscode(pwd.toUtf8()); Local::setPasscode(pwd.toUtf8());
App::wnd()->checkAutoLock(); AuthSession::Current().checkAutoLock();
closeBox(); closeBox();
} }
} }

View File

@ -45,6 +45,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "autoupdater.h" #include "autoupdater.h"
#include "observer_peer.h" #include "observer_peer.h"
#include "auth_session.h" #include "auth_session.h"
#include "messenger.h"
#include "window/notifications_manager.h" #include "window/notifications_manager.h"
#include "ui/effects/widget_fade_wrap.h" #include "ui/effects/widget_fade_wrap.h"
#include "window/window_controller.h" #include "window/window_controller.h"
@ -2305,7 +2306,7 @@ DialogsWidget::DialogsWidget(QWidget *parent, gsl::not_null<Window::Controller*>
subscribe(Global::RefLocalPasscodeChanged(), [this] { updateLockUnlockVisibility(); }); subscribe(Global::RefLocalPasscodeChanged(), [this] { updateLockUnlockVisibility(); });
_lockUnlock->setClickedCallback([this] { _lockUnlock->setClickedCallback([this] {
_lockUnlock->setIconOverride(&st::dialogsUnlockIcon, &st::dialogsUnlockIconOver); _lockUnlock->setIconOverride(&st::dialogsUnlockIcon, &st::dialogsUnlockIconOver);
App::wnd()->setupPasscode(); Messenger::Instance().setupPasscode();
_lockUnlock->setIconOverride(nullptr); _lockUnlock->setIconOverride(nullptr);
}); });
_mainMenuToggle->setClickedCallback([this] { showMainMenu(); }); _mainMenuToggle->setClickedCallback([this] { showMainMenu(); });

View File

@ -3501,7 +3501,7 @@ void MainWidget::ptsApplySkippedUpdates() {
} }
void MainWidget::feedDifference(const MTPVector<MTPUser> &users, const MTPVector<MTPChat> &chats, const MTPVector<MTPMessage> &msgs, const MTPVector<MTPUpdate> &other) { void MainWidget::feedDifference(const MTPVector<MTPUser> &users, const MTPVector<MTPChat> &chats, const MTPVector<MTPMessage> &msgs, const MTPVector<MTPUpdate> &other) {
App::wnd()->checkAutoLock(); AuthSession::Current().checkAutoLock();
App::feedUsers(users); App::feedUsers(users);
App::feedChats(chats); App::feedChats(chats);
feedMessageIds(other); feedMessageIds(other);
@ -4167,7 +4167,7 @@ MainWidget::~MainWidget() {
void MainWidget::updateOnline(bool gotOtherOffline) { void MainWidget::updateOnline(bool gotOtherOffline) {
if (this != App::main()) return; if (this != App::main()) return;
App::wnd()->checkAutoLock(); AuthSession::Current().checkAutoLock();
bool isOnline = App::wnd()->isActive(); bool isOnline = App::wnd()->isActive();
int updateIn = Global::OnlineUpdatePeriod(); int updateIn = Global::OnlineUpdatePeriod();
@ -4262,7 +4262,7 @@ void MainWidget::checkIdleFinish() {
void MainWidget::updateReceived(const mtpPrime *from, const mtpPrime *end) { void MainWidget::updateReceived(const mtpPrime *from, const mtpPrime *end) {
if (end <= from) return; if (end <= from) return;
App::wnd()->checkAutoLock(); AuthSession::Current().checkAutoLock();
if (mtpTypeId(*from) == mtpc_new_session_created) { if (mtpTypeId(*from) == mtpc_new_session_created) {
try { try {

View File

@ -108,13 +108,12 @@ MainWindow::MainWindow() {
_inactiveTimer.setSingleShot(true); _inactiveTimer.setSingleShot(true);
connect(&_inactiveTimer, SIGNAL(timeout()), this, SLOT(onInactiveTimer())); connect(&_inactiveTimer, SIGNAL(timeout()), this, SLOT(onInactiveTimer()));
connect(&_autoLockTimer, SIGNAL(timeout()), this, SLOT(checkAutoLock()));
subscribe(Global::RefSelfChanged(), [this] { updateGlobalMenu(); }); subscribe(Global::RefSelfChanged(), [this] { updateGlobalMenu(); });
subscribe(Window::Theme::Background(), [this](const Window::Theme::BackgroundUpdate &data) { subscribe(Window::Theme::Background(), [this](const Window::Theme::BackgroundUpdate &data) {
themeUpdated(data); themeUpdated(data);
}); });
subscribe(Messenger::Instance().authSessionChanged(), [this] { checkAuthSession(); }); subscribe(Messenger::Instance().authSessionChanged(), [this] { checkAuthSession(); });
subscribe(Messenger::Instance().passcodedChanged(), [this] { updateGlobalMenu(); });
checkAuthSession(); checkAuthSession();
setAttribute(Qt::WA_NoSystemBackground); setAttribute(Qt::WA_NoSystemBackground);
@ -237,12 +236,8 @@ void MainWindow::clearPasscode() {
if (_intro) { if (_intro) {
_intro->showAnimated(bg, true); _intro->showAnimated(bg, true);
} else { } else {
t_assert(_main != nullptr);
_main->showAnimated(bg, true); _main->showAnimated(bg, true);
}
AuthSession::Current().notifications().updateAll();
updateGlobalMenu();
if (_main) {
_main->checkStartUrl(); _main->checkStartUrl();
} }
} }
@ -262,32 +257,6 @@ void MainWindow::setupPasscode() {
} else { } else {
setInnerFocus(); setInnerFocus();
} }
_shouldLockAt = 0;
if (AuthSession::Exists()) {
AuthSession::Current().notifications().updateAll();
}
updateGlobalMenu();
}
void MainWindow::checkAutoLockIn(int msec) {
if (_autoLockTimer.isActive()) {
int remain = _autoLockTimer.remainingTime();
if (remain > 0 && remain <= msec) return;
}
_autoLockTimer.start(msec);
}
void MainWindow::checkAutoLock() {
if (!Global::LocalPasscode() || App::passcoded()) return;
App::app()->checkLocalTime();
auto ms = getms(true), idle = psIdleTime(), should = Global::AutoLock() * 1000LL;
if (idle >= should || (_shouldLockAt > 0 && ms > _shouldLockAt + 3000LL)) {
setupPasscode();
} else {
_shouldLockAt = ms + (should - idle);
_autoLockTimer.start(should - idle);
}
} }
void MainWindow::setupIntro() { void MainWindow::setupIntro() {

View File

@ -90,7 +90,6 @@ public:
void setupPasscode(); void setupPasscode();
void clearPasscode(); void clearPasscode();
void checkAutoLockIn(int msec);
void setupIntro(); void setupIntro();
void setupMain(const MTPUser *user = nullptr); void setupMain(const MTPUser *user = nullptr);
void serviceNotification(const TextWithEntities &message, const MTPMessageMedia &media = MTP_messageMediaEmpty(), int32 date = 0, bool force = false); void serviceNotification(const TextWithEntities &message, const MTPMessageMedia &media = MTP_messageMediaEmpty(), int32 date = 0, bool force = false);
@ -164,8 +163,6 @@ protected:
void updateControlsGeometry() override; void updateControlsGeometry() override;
public slots: public slots:
void checkAutoLock();
void showSettings(); void showSettings();
void layerHidden(); void layerHidden();
void setInnerFocus(); void setInnerFocus();
@ -238,9 +235,6 @@ private:
bool _inactivePress = false; bool _inactivePress = false;
QTimer _inactiveTimer; QTimer _inactiveTimer;
SingleTimer _autoLockTimer;
TimeMs _shouldLockAt = 0;
}; };
class PreLaunchWindow : public TWidget { class PreLaunchWindow : public TWidget {

View File

@ -1600,6 +1600,8 @@ void MediaView::onVideoPlayProgress(const AudioMsgId &audioId) {
if (state.duration) { if (state.duration) {
updateVideoPlaybackState(state); updateVideoPlaybackState(state);
} }
AuthSession::Current().data().setLastTimeVideoPlayedAt(getms(true));
} }
void MediaView::updateVideoPlaybackState(const Media::Player::TrackState &state) { void MediaView::updateVideoPlaybackState(const Media::Player::TrackState &state) {

View File

@ -129,7 +129,7 @@ Messenger::Messenger() : QObject()
DEBUG_LOG(("Application Info: showing.")); DEBUG_LOG(("Application Info: showing."));
if (state == Local::ReadMapPassNeeded) { if (state == Local::ReadMapPassNeeded) {
_window->setupPasscode(); setupPasscode();
} else { } else {
if (AuthSession::Exists()) { if (AuthSession::Exists()) {
_window->setupMain(); _window->setupMain();
@ -728,6 +728,17 @@ void Messenger::checkMapVersion() {
} }
} }
void Messenger::setupPasscode() {
_window->setupPasscode();
_passcodedChanged.notify();
}
void Messenger::clearPasscode() {
cSetPasscodeBadTries(0);
_window->clearPasscode();
_passcodedChanged.notify();
}
void Messenger::prepareToDestroy() { void Messenger::prepareToDestroy() {
_window.reset(); _window.reset();

View File

@ -113,6 +113,11 @@ public:
void checkLocalTime(); void checkLocalTime();
void checkMapVersion(); void checkMapVersion();
void setupPasscode();
void clearPasscode();
base::Observable<void> &passcodedChanged() {
return _passcodedChanged;
}
void handleAppActivated(); void handleAppActivated();
void handleAppDeactivated(); void handleAppDeactivated();
@ -161,5 +166,6 @@ private:
std::unique_ptr<MTP::Instance> _mtprotoForKeysDestroy; std::unique_ptr<MTP::Instance> _mtprotoForKeysDestroy;
std::unique_ptr<AuthSession> _authSession; std::unique_ptr<AuthSession> _authSession;
base::Observable<void> _authSessionChanged; base::Observable<void> _authSessionChanged;
base::Observable<void> _passcodedChanged;
}; };

View File

@ -59,8 +59,7 @@ void PasscodeWidget::onSubmit() {
if (App::main()) { if (App::main()) {
if (Local::checkPasscode(_passcode->text().toUtf8())) { if (Local::checkPasscode(_passcode->text().toUtf8())) {
cSetPasscodeBadTries(0); Messenger::Instance().clearPasscode(); // Destroys this widget.
App::wnd()->clearPasscode(); // Destroys this widget.
return; return;
} else { } else {
cSetPasscodeBadTries(cPasscodeBadTries() + 1); cSetPasscodeBadTries(cPasscodeBadTries() + 1);

View File

@ -21,6 +21,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "platform/win/windows_event_filter.h" #include "platform/win/windows_event_filter.h"
#include "mainwindow.h" #include "mainwindow.h"
#include "auth_session.h"
namespace Platform { namespace Platform {
namespace { namespace {
@ -73,7 +74,9 @@ bool EventFilter::mainWindowEvent(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lPa
switch (msg) { switch (msg) {
case WM_TIMECHANGE: { case WM_TIMECHANGE: {
App::wnd()->checkAutoLockIn(100); if (AuthSession::Exists()) {
AuthSession::Current().checkAutoLockIn(100);
}
} return false; } return false;
case WM_WTSSESSION_CHANGE: { case WM_WTSSESSION_CHANGE: {

View File

@ -23,6 +23,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "mainwindow.h" #include "mainwindow.h"
#include "passcodewidget.h" #include "passcodewidget.h"
#include "mainwidget.h" #include "mainwidget.h"
#include "messenger.h"
#include "media/player/media_player_instance.h" #include "media/player/media_player_instance.h"
#include "platform/platform_specific.h" #include "platform/platform_specific.h"
#include "base/parse_helper.h" #include "base/parse_helper.h"
@ -37,7 +38,7 @@ bool lock_telegram() {
w->passcodeWidget()->onSubmit(); w->passcodeWidget()->onSubmit();
return true; return true;
} else if (Global::LocalPasscode()) { } else if (Global::LocalPasscode()) {
w->setupPasscode(); Messenger::Instance().setupPasscode();
return true; return true;
} }
} }