1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 // A helper class that stays in sync with a preference (bool, int, real,
6 // string or filepath). For example:
10 // MyClass(PrefService* prefs) {
11 // my_string_.Init(prefs::kHomePage, prefs);
14 // StringPrefMember my_string_;
17 // my_string_ should stay in sync with the prefs::kHomePage pref and will
18 // update if either the pref changes or if my_string_.SetValue is called.
20 // An optional observer can be passed into the Init method which can be used to
21 // notify MyClass of changes. Note that if you use SetValue(), the observer
22 // will not be notified.
24 #ifndef BASE_PREFS_PREF_MEMBER_H_
25 #define BASE_PREFS_PREF_MEMBER_H_
30 #include "base/basictypes.h"
31 #include "base/bind.h"
32 #include "base/callback_forward.h"
33 #include "base/files/file_path.h"
34 #include "base/logging.h"
35 #include "base/memory/ref_counted.h"
36 #include "base/prefs/base_prefs_export.h"
37 #include "base/prefs/pref_observer.h"
38 #include "base/single_thread_task_runner.h"
39 #include "base/values.h"
45 class BASE_PREFS_EXPORT PrefMemberBase
: public PrefObserver
{
47 // Type of callback you can register if you need to know the name of
48 // the pref that is changing.
49 typedef base::Callback
<void(const std::string
&)> NamedChangeCallback
;
51 PrefService
* prefs() { return prefs_
; }
52 const PrefService
* prefs() const { return prefs_
; }
55 class BASE_PREFS_EXPORT Internal
56 : public base::RefCountedThreadSafe
<Internal
> {
60 // Update the value, either by calling |UpdateValueInternal| directly
61 // or by dispatching to the right thread.
62 // Takes ownership of |value|.
63 void UpdateValue(base::Value
* value
,
65 bool is_user_modifiable
,
66 const base::Closure
& callback
) const;
69 const scoped_refptr
<base::SingleThreadTaskRunner
>& task_runner
);
71 // See PrefMember<> for description.
72 bool IsManaged() const {
76 bool IsUserModifiable() const {
77 return is_user_modifiable_
;
81 friend class base::RefCountedThreadSafe
<Internal
>;
84 void CheckOnCorrectThread() const {
85 DCHECK(IsOnCorrectThread());
89 // This method actually updates the value. It should only be called from
90 // the thread the PrefMember is on.
91 virtual bool UpdateValueInternal(const base::Value
& value
) const = 0;
93 bool IsOnCorrectThread() const;
95 scoped_refptr
<base::SingleThreadTaskRunner
> thread_loop_
;
96 mutable bool is_managed_
;
97 mutable bool is_user_modifiable_
;
99 DISALLOW_COPY_AND_ASSIGN(Internal
);
103 virtual ~PrefMemberBase();
105 // See PrefMember<> for description.
106 void Init(const char* pref_name
, PrefService
* prefs
,
107 const NamedChangeCallback
& observer
);
108 void Init(const char* pref_name
, PrefService
* prefs
);
110 virtual void CreateInternal() const = 0;
112 // See PrefMember<> for description.
116 const scoped_refptr
<base::SingleThreadTaskRunner
>& task_runner
);
119 void OnPreferenceChanged(PrefService
* service
,
120 const std::string
& pref_name
) override
;
122 void VerifyValuePrefName() const {
123 DCHECK(!pref_name_
.empty());
126 // This method is used to do the actual sync with the preference.
127 // Note: it is logically const, because it doesn't modify the state
128 // seen by the outside world. It is just doing a lazy load behind the scenes.
129 void UpdateValueFromPref(const base::Closure
& callback
) const;
131 // Verifies the preference name, and lazily loads the preference value if
132 // it hasn't been loaded yet.
133 void VerifyPref() const;
135 const std::string
& pref_name() const { return pref_name_
; }
137 virtual Internal
* internal() const = 0;
139 // Used to allow registering plain base::Closure callbacks.
140 static void InvokeUnnamedCallback(const base::Closure
& callback
,
141 const std::string
& pref_name
);
144 // Ordered the members to compact the class instance.
145 std::string pref_name_
;
146 NamedChangeCallback observer_
;
153 // This function implements StringListPrefMember::UpdateValue().
154 // It is exposed here for testing purposes.
155 bool BASE_PREFS_EXPORT
PrefMemberVectorStringUpdate(
156 const base::Value
& value
,
157 std::vector
<std::string
>* string_vector
);
159 } // namespace subtle
161 template <typename ValueType
>
162 class PrefMember
: public subtle::PrefMemberBase
{
164 // Defer initialization to an Init method so it's easy to make this class be
165 // a member variable.
167 virtual ~PrefMember() {}
169 // Do the actual initialization of the class. Use the two-parameter
170 // version if you don't want any notifications of changes. This
171 // method should only be called on the UI thread.
172 void Init(const char* pref_name
, PrefService
* prefs
,
173 const NamedChangeCallback
& observer
) {
174 subtle::PrefMemberBase::Init(pref_name
, prefs
, observer
);
176 void Init(const char* pref_name
, PrefService
* prefs
,
177 const base::Closure
& observer
) {
178 subtle::PrefMemberBase::Init(
180 base::Bind(&PrefMemberBase::InvokeUnnamedCallback
, observer
));
182 void Init(const char* pref_name
, PrefService
* prefs
) {
183 subtle::PrefMemberBase::Init(pref_name
, prefs
);
186 // Unsubscribes the PrefMember from the PrefService. After calling this
187 // function, the PrefMember may not be used any more on the UI thread.
188 // Assuming |MoveToThread| was previously called, |GetValue|, |IsManaged|,
189 // and |IsUserModifiable| can still be called from the other thread but
190 // the results will no longer update from the PrefService.
191 // This method should only be called on the UI thread.
193 subtle::PrefMemberBase::Destroy();
196 // Moves the PrefMember to another thread, allowing read accesses from there.
197 // Changes from the PrefService will be propagated asynchronously
199 // This method should only be used from the thread the PrefMember is currently
200 // on, which is the UI thread by default.
202 const scoped_refptr
<base::SingleThreadTaskRunner
>& task_runner
) {
203 subtle::PrefMemberBase::MoveToThread(task_runner
);
206 // Check whether the pref is managed, i.e. controlled externally through
207 // enterprise configuration management (e.g. windows group policy). Returns
208 // false for unknown prefs.
209 // This method should only be used from the thread the PrefMember is currently
210 // on, which is the UI thread unless changed by |MoveToThread|.
211 bool IsManaged() const {
213 return internal_
->IsManaged();
216 // Checks whether the pref can be modified by the user. This returns false
217 // when the pref is managed by a policy or an extension, and when a command
218 // line flag overrides the pref.
219 // This method should only be used from the thread the PrefMember is currently
220 // on, which is the UI thread unless changed by |MoveToThread|.
221 bool IsUserModifiable() const {
223 return internal_
->IsUserModifiable();
226 // Retrieve the value of the member variable.
227 // This method should only be used from the thread the PrefMember is currently
228 // on, which is the UI thread unless changed by |MoveToThread|.
229 ValueType
GetValue() const {
231 return internal_
->value();
234 // Provided as a convenience.
235 ValueType
operator*() const {
239 // Set the value of the member variable.
240 // This method should only be called on the UI thread.
241 void SetValue(const ValueType
& value
) {
242 VerifyValuePrefName();
243 setting_value_
= true;
245 setting_value_
= false;
248 // Returns the pref name.
249 const std::string
& GetPrefName() const {
254 class Internal
: public subtle::PrefMemberBase::Internal
{
256 Internal() : value_(ValueType()) {}
259 CheckOnCorrectThread();
264 virtual ~Internal() {}
266 virtual BASE_PREFS_EXPORT
bool UpdateValueInternal(
267 const base::Value
& value
) const override
;
269 // We cache the value of the pref so we don't have to keep walking the pref
271 mutable ValueType value_
;
273 DISALLOW_COPY_AND_ASSIGN(Internal
);
276 virtual Internal
* internal() const override
{ return internal_
.get(); }
277 virtual void CreateInternal() const override
{ internal_
= new Internal(); }
279 // This method is used to do the actual sync with pref of the specified type.
280 void BASE_PREFS_EXPORT
UpdatePref(const ValueType
& value
);
282 mutable scoped_refptr
<Internal
> internal_
;
284 DISALLOW_COPY_AND_ASSIGN(PrefMember
);
287 // Declaration of template specialization need to be repeated here
288 // specifically for each specialization (rather than just once above)
289 // or at least one of our compilers won't be happy in all cases.
290 // Specifically, it was failing on ChromeOS with a complaint about
291 // PrefMember<FilePath>::UpdateValueInternal not being defined when
292 // built in a chroot with the following parameters:
294 // FEATURES="noclean nostrip" USE="-chrome_debug -chrome_remoting
295 // -chrome_internal -chrome_pdf component_build"
296 // ~/trunk/goma/goma-wrapper cros_chrome_make --board=${BOARD}
297 // --install --runhooks
300 BASE_PREFS_EXPORT
void PrefMember
<bool>::UpdatePref(const bool& value
);
303 BASE_PREFS_EXPORT
bool PrefMember
<bool>::Internal::UpdateValueInternal(
304 const base::Value
& value
) const;
307 BASE_PREFS_EXPORT
void PrefMember
<int>::UpdatePref(const int& value
);
310 BASE_PREFS_EXPORT
bool PrefMember
<int>::Internal::UpdateValueInternal(
311 const base::Value
& value
) const;
314 BASE_PREFS_EXPORT
void PrefMember
<double>::UpdatePref(const double& value
);
317 BASE_PREFS_EXPORT
bool PrefMember
<double>::Internal::UpdateValueInternal(
318 const base::Value
& value
) const;
321 BASE_PREFS_EXPORT
void PrefMember
<std::string
>::UpdatePref(
322 const std::string
& value
);
325 BASE_PREFS_EXPORT
bool PrefMember
<std::string
>::Internal::UpdateValueInternal(
326 const base::Value
& value
) const;
329 BASE_PREFS_EXPORT
void PrefMember
<base::FilePath
>::UpdatePref(
330 const base::FilePath
& value
);
333 BASE_PREFS_EXPORT
bool
334 PrefMember
<base::FilePath
>::Internal::UpdateValueInternal(
335 const base::Value
& value
) const;
338 BASE_PREFS_EXPORT
void PrefMember
<std::vector
<std::string
> >::UpdatePref(
339 const std::vector
<std::string
>& value
);
342 BASE_PREFS_EXPORT
bool
343 PrefMember
<std::vector
<std::string
> >::Internal::UpdateValueInternal(
344 const base::Value
& value
) const;
346 typedef PrefMember
<bool> BooleanPrefMember
;
347 typedef PrefMember
<int> IntegerPrefMember
;
348 typedef PrefMember
<double> DoublePrefMember
;
349 typedef PrefMember
<std::string
> StringPrefMember
;
350 typedef PrefMember
<base::FilePath
> FilePathPrefMember
;
351 // This preference member is expensive for large string arrays.
352 typedef PrefMember
<std::vector
<std::string
> > StringListPrefMember
;
354 #endif // BASE_PREFS_PREF_MEMBER_H_