1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
ash / policy / policy_recommendation_restorer.cc [blame]
// Copyright 2018 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "ash/policy/policy_recommendation_restorer.h"
#include "ash/session/session_controller_impl.h"
#include "ash/shell.h"
#include "base/check.h"
#include "base/containers/contains.h"
#include "base/functional/bind.h"
#include "components/prefs/pref_change_registrar.h"
#include "components/prefs/pref_service.h"
#include "ui/base/user_activity/user_activity_detector.h"
namespace ash {
namespace {
// The amount of idle time after which recommended values are restored.
constexpr base::TimeDelta kRestoreDelayInMinutes = base::Minutes(1);
} // namespace
PolicyRecommendationRestorer::PolicyRecommendationRestorer() {
Shell::Get()->session_controller()->AddObserver(this);
}
PolicyRecommendationRestorer::~PolicyRecommendationRestorer() {
StopTimer();
Shell::Get()->session_controller()->RemoveObserver(this);
}
void PolicyRecommendationRestorer::ObservePref(const std::string& pref_name) {
PrefService* prefs =
Shell::Get()->session_controller()->GetSigninScreenPrefService();
DCHECK(prefs);
DCHECK(!base::Contains(pref_names_, pref_name));
if (!pref_change_registrar_) {
pref_change_registrar_ = std::make_unique<PrefChangeRegistrar>();
pref_change_registrar_->Init(prefs);
}
pref_change_registrar_->Add(
pref_name, base::BindRepeating(&PolicyRecommendationRestorer::Restore,
base::Unretained(this), true));
pref_names_.insert(pref_name);
Restore(false /* allow_delay */, pref_name);
}
void PolicyRecommendationRestorer::OnActiveUserPrefServiceChanged(
PrefService* pref_service) {
active_user_pref_connected_ = true;
StopTimer();
RestoreAll();
}
void PolicyRecommendationRestorer::OnUserActivity(const ui::Event* event) {
if (restore_timer_.IsRunning())
restore_timer_.Reset();
}
void PolicyRecommendationRestorer::DisableForTesting() {
disabled_for_testing_ = true;
}
void PolicyRecommendationRestorer::Restore(bool allow_delay,
const std::string& pref_name) {
const PrefService::Preference* pref =
pref_change_registrar_->prefs()->FindPreference(pref_name);
CHECK(pref);
if (!pref->GetRecommendedValue() || !pref->HasUserSetting())
return;
if (active_user_pref_connected_) {
allow_delay = false;
} else if (allow_delay) {
// Skip the delay if there has been no user input since |pref_name| is
// started observing recommended value.
if (ui::UserActivityDetector::Get()->last_activity_time().is_null()) {
allow_delay = false;
}
}
if (allow_delay)
StartTimer();
else if (!disabled_for_testing_)
pref_change_registrar_->prefs()->ClearPref(pref->name());
}
void PolicyRecommendationRestorer::RestoreAll() {
for (const auto& pref_name : pref_names_)
Restore(false, pref_name);
}
void PolicyRecommendationRestorer::StartTimer() {
// Listen for user activity so that the timer can be reset while the user is
// active, causing it to fire only when the user remains idle for
// |kRestoreDelayInMinutes|.
ui::UserActivityDetector* user_activity_detector =
ui::UserActivityDetector::Get();
if (!user_activity_detector->HasObserver(this)) {
user_activity_detector->AddObserver(this);
}
// There should be a separate timer for each pref. However, in the common
// case of the user changing settings, a single timer is sufficient. This is
// because a change initiated by the user implies user activity, so that even
// if there was a separate timer per pref, they would all be reset at that
// point, causing them to fire at exactly the same time. In the much rarer
// case of a recommended value changing, a single timer is a close
// approximation of the behavior that would be obtained by resetting the timer
// for the affected pref only.
restore_timer_.Start(FROM_HERE, kRestoreDelayInMinutes,
base::BindOnce(&PolicyRecommendationRestorer::RestoreAll,
base::Unretained(this)));
}
void PolicyRecommendationRestorer::StopTimer() {
restore_timer_.Stop();
ui::UserActivityDetector::Get()->RemoveObserver(this);
}
} // namespace ash