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
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
|
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef CHROME_BROWSER_MEDIA_GALLERIES_MEDIA_GALLERIES_PREFERENCES_H_
#define CHROME_BROWSER_MEDIA_GALLERIES_MEDIA_GALLERIES_PREFERENCES_H_
#include <map>
#include <set>
#include <string>
#include "base/basictypes.h"
#include "base/callback_forward.h"
#include "base/files/file_path.h"
#include "base/memory/weak_ptr.h"
#include "base/observer_list.h"
#include "base/strings/string16.h"
#include "base/time/time.h"
#include "chrome/browser/storage_monitor/removable_storage_observer.h"
#include "components/browser_context_keyed_service/browser_context_keyed_service.h"
class Profile;
namespace base {
class DictionaryValue;
}
namespace extensions {
class Extension;
class ExtensionPrefs;
}
namespace user_prefs {
class PrefRegistrySyncable;
}
typedef uint64 MediaGalleryPrefId;
const MediaGalleryPrefId kInvalidMediaGalleryPrefId = 0;
struct MediaGalleryPermission {
MediaGalleryPrefId pref_id;
bool has_permission;
};
struct MediaGalleryPrefInfo {
enum Type {
kAutoDetected, // Auto added to the list of galleries.
kUserAdded, // Explicitly added by the user.
kBlackListed, // Auto added but then removed by the user.
kInvalidType,
};
MediaGalleryPrefInfo();
~MediaGalleryPrefInfo();
// The absolute path of the gallery.
base::FilePath AbsolutePath() const;
// The ID that identifies this gallery in this Profile.
MediaGalleryPrefId pref_id;
// The user-visible name of this gallery.
string16 display_name;
// A string which uniquely and persistently identifies the device that the
// gallery lives on.
std::string device_id;
// The root of the gallery, relative to the root of the device.
base::FilePath path;
// The type of gallery.
Type type;
// The volume label of the volume/device on which the gallery
// resides. Empty if there is no such label or it is unknown.
string16 volume_label;
// Vendor name for the volume/device on which the gallery is located.
// Will be empty if unknown.
string16 vendor_name;
// Model name for the volume/device on which the gallery is located.
// Will be empty if unknown.
string16 model_name;
// The capacity in bytes of the volume/device on which the gallery is
// located. Will be zero if unknown.
uint64 total_size_in_bytes;
// If the gallery is on a removable device, the time that device was last
// attached. It is stored in preferences by the base::Time internal value,
// which is microseconds since the epoch.
base::Time last_attach_time;
// Set to true if the volume metadata fields (volume_label, vendor_name,
// model_name, total_size_in_bytes) were set. False if these fields were
// never written.
bool volume_metadata_valid;
// 0 if the display_name is set externally and always used for display.
// 1 if the display_name is only set externally when it is overriding
// the name constructed from volume metadata.
int prefs_version;
// Called by views to provide details for the gallery permission entries.
string16 GetGalleryDisplayName() const;
string16 GetGalleryTooltip() const;
string16 GetGalleryAdditionalDetails() const;
// Returns true if the gallery is currently a removable device gallery which
// is now attached, or a fixed storage gallery.
bool IsGalleryAvailable() const;
};
typedef std::map<MediaGalleryPrefId, MediaGalleryPrefInfo>
MediaGalleriesPrefInfoMap;
typedef std::set<MediaGalleryPrefId> MediaGalleryPrefIdSet;
// A class to manage the media gallery preferences. There is one instance per
// user profile.
class MediaGalleriesPreferences : public BrowserContextKeyedService,
public RemovableStorageObserver {
public:
class GalleryChangeObserver {
public:
// |extension_id| specifies the extension affected by this change.
// |pref_id| refers to the gallery.
virtual void OnPermissionAdded(MediaGalleriesPreferences* pref,
const std::string& extension_id,
MediaGalleryPrefId pref_id) {}
virtual void OnPermissionRemoved(MediaGalleriesPreferences* pref,
const std::string& extension_id,
MediaGalleryPrefId pref_id) {}
virtual void OnGalleryAdded(MediaGalleriesPreferences* pref,
MediaGalleryPrefId pref_id) {}
virtual void OnGalleryRemoved(MediaGalleriesPreferences* pref,
MediaGalleryPrefId pref_id) {}
virtual void OnGalleryInfoUpdated(MediaGalleriesPreferences* pref,
MediaGalleryPrefId pref_id) {}
protected:
virtual ~GalleryChangeObserver();
};
explicit MediaGalleriesPreferences(Profile* profile);
virtual ~MediaGalleriesPreferences();
// Ensures that the preferences is initialized. The provided callback, if
// non-null, will be called when initialization is complete. If initialization
// has already completed, this callback will be invoked in the calling stack.
// Before the callback is run, other calls may not return the correct results.
// Should be invoked on the UI thread; callbacks will be run on the UI thread.
// This call also ensures that the StorageMonitor is initialized.
// Note for unit tests: This requires an active FILE thread and
// EnsureMediaDirectoriesExists instance to complete reliably.
void EnsureInitialized(base::Closure callback);
// Return true if the storage monitor has already been initialized.
bool IsInitialized() const;
Profile* profile();
void AddGalleryChangeObserver(GalleryChangeObserver* observer);
void RemoveGalleryChangeObserver(GalleryChangeObserver* observer);
// RemovableStorageObserver implementation.
virtual void OnRemovableStorageAttached(const StorageInfo& info) OVERRIDE;
// Lookup a media gallery and fill in information about it and return true if
// it exists. Return false if it does not, filling in default information.
// TODO(vandebo) figure out if we want this to be async, in which case:
// void LookUpGalleryByPath(base::FilePath& path,
// callback(const MediaGalleryInfo&))
bool LookUpGalleryByPath(const base::FilePath& path,
MediaGalleryPrefInfo* gallery) const;
MediaGalleryPrefIdSet LookUpGalleriesByDeviceId(
const std::string& device_id) const;
// Returns the absolute file path of the gallery specified by the
// |gallery_id|. Returns an empty file path if the |gallery_id| is invalid.
// Set |include_unpermitted_galleries| to true to get the file path of the
// gallery to which this |extension| has no access permission.
base::FilePath LookUpGalleryPathForExtension(
MediaGalleryPrefId gallery_id,
const extensions::Extension* extension,
bool include_unpermitted_galleries);
// Teaches the registry about a new gallery.
// Returns the gallery's pref id.
MediaGalleryPrefId AddGallery(const std::string& device_id,
const base::FilePath& relative_path,
bool user_added,
const string16& volume_label,
const string16& vendor_name,
const string16& model_name,
uint64 total_size_in_bytes,
base::Time last_attach_time);
// Teach the registry about a user added registry simply from the path.
// Returns the gallery's pref id.
MediaGalleryPrefId AddGalleryByPath(const base::FilePath& path);
// Removes the gallery identified by |id| from the store.
void ForgetGalleryById(MediaGalleryPrefId id);
MediaGalleryPrefIdSet GalleriesForExtension(
const extensions::Extension& extension) const;
// Returns true if the permission changed. Returns false if there was
// no change.
bool SetGalleryPermissionForExtension(const extensions::Extension& extension,
MediaGalleryPrefId pref_id,
bool has_permission);
const MediaGalleriesPrefInfoMap& known_galleries() const;
// BrowserContextKeyedService implementation:
virtual void Shutdown() OVERRIDE;
static void RegisterProfilePrefs(user_prefs::PrefRegistrySyncable* registry);
// Returns true if the media gallery preferences system has ever been used
// for this profile. To be exact, it checks if a gallery has ever been added
// (including defaults).
static bool APIHasBeenUsed(Profile* profile);
private:
friend class MediaGalleriesPreferencesTest;
friend class MediaGalleriesPermissionsTest;
typedef std::map<std::string /*device id*/, MediaGalleryPrefIdSet>
DeviceIdPrefIdsMap;
// These must be called on the UI thread.
void OnInitializationCallbackReturned();
void FinishInitialization();
// Populates the default galleries. Call only on fresh profiles.
void AddDefaultGalleries();
// This is a hack - Some devices (iTunes, Picasa) are singletons in that only
// one instance of that type is supported at a time. As such, the device id
// should just be "itunes:" or "picasa:" but that would mean finding the
// location of the database file multiple times, which may be an async
// operation. Storing the location of the backing database in the device
// id allows that look up to be avoided. However, the cost is that if the
// database moves, the device id in preferences has to be updated. This
// method searches for a gallery of the type passed in and updates its
// device id. It returns true if the device id is up to date.
bool UpdateDeviceIDForSingletonType(const std::string& device_id);
void OnStorageMonitorInit(bool add_default_galleries);
// Handle an iPhoto, iTunes, or Picasa finder returning a device ID to us.
void OnFinderDeviceID(const std::string& device_id);
// Builds |known_galleries_| from the persistent store.
void InitFromPrefs();
MediaGalleryPrefId AddGalleryInternal(const std::string& device_id,
const string16& display_name,
const base::FilePath& relative_path,
bool user_added,
const string16& volume_label,
const string16& vendor_name,
const string16& model_name,
uint64 total_size_in_bytes,
base::Time last_attach_time,
bool volume_metadata_valid,
int prefs_version);
// Sets permission for the media galleries identified by |gallery_id| for the
// extension in the given |prefs|. Returns true only if anything changed.
bool SetGalleryPermissionInPrefs(const std::string& extension_id,
MediaGalleryPrefId gallery_id,
bool has_access);
// Removes the entry for the media galleries permissions identified by
// |gallery_id| for the extension in the given |prefs|.
// Returns true only if anything changed.
bool UnsetGalleryPermissionInPrefs(const std::string& extension_id,
MediaGalleryPrefId gallery_id);
// Return all media gallery permissions for the extension in the given
// |prefs|.
std::vector<MediaGalleryPermission> GetGalleryPermissionsFromPrefs(
const std::string& extension_id) const;
// Remove all the media gallery permissions in |prefs| for the gallery
// specified by |gallery_id|.
void RemoveGalleryPermissionsFromPrefs(MediaGalleryPrefId gallery_id);
// Get the ExtensionPrefs to use; this will be either the ExtensionPrefs
// object associated with |profile_|, or extension_prefs_for_testing_, if
// SetExtensionPrefsForTesting() has been called.
extensions::ExtensionPrefs* GetExtensionPrefs() const;
// Set the ExtensionPrefs object to be returned by GetExtensionPrefs().
void SetExtensionPrefsForTesting(extensions::ExtensionPrefs* extension_prefs);
base::WeakPtrFactory<MediaGalleriesPreferences> weak_factory_;
bool initialized_;
std::vector<base::Closure> on_initialize_callbacks_;
int pre_initialization_callbacks_waiting_;
// The profile that owns |this|.
Profile* profile_;
// The ExtensionPrefs used in a testing environment, where
// BrowserContextKeyedServices aren't used. This will be NULL unless it is
// set with SetExtensionPrefsForTesting().
extensions::ExtensionPrefs* extension_prefs_for_testing_;
// An in-memory cache of known galleries.
MediaGalleriesPrefInfoMap known_galleries_;
// A mapping from device id to the set of gallery pref ids on that device.
// All pref ids in |device_map_| are also in |known_galleries_|.
DeviceIdPrefIdsMap device_map_;
ObserverList<GalleryChangeObserver> gallery_change_observers_;
DISALLOW_COPY_AND_ASSIGN(MediaGalleriesPreferences);
};
#endif // CHROME_BROWSER_MEDIA_GALLERIES_MEDIA_GALLERIES_PREFERENCES_H_
|