mirror of https://github.com/procxx/kepka.git
Move passcode management from MainWindow.
Check for auto lock in AuthSession. Don't autolock while video plays. Closes #3219
This commit is contained in:
parent
e3aacc8072
commit
de7c886008
|
@ -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;
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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();
|
||||||
}
|
}
|
||||||
|
|
|
@ -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();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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(); });
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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() {
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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();
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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: {
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue