1 // Copyright 2015 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 #import "ios/web/browsing_data_partition_impl.h"
7 #include "base/logging.h"
8 #include "ios/web/public/browser_state.h"
9 #include "ios/web/public/browsing_data_partition_client.h"
10 #import "ios/web/public/crw_browsing_data_store.h"
11 #include "ios/web/public/web_thread.h"
13 // A class that observes the |mode| changes in a CRWBrowsingDataStore using KVO.
14 // Maintains a count of all the CRWBrowsingDataStores whose mode is out of sync
15 // with their corresponding ActiveStateManager's active state.
16 @interface CRWBrowsingDataStoreModeObserver : NSObject
18 // The total count of the CRWBrowsingDataStores -- that are being observed --
19 // whose mode is out of sync with their associated ActiveStateManager's active
21 @property(nonatomic, assign) NSUInteger outOfSyncStoreCount;
23 // Adds |browsingDataStore| to the list of the CRWBrowsingDataStores that are
24 // being observed for KVO changes in the |mode| value. |browserState| cannot be
25 // null and the |browserState|'s associated CRWBrowsingDataStore must be
26 // |browsingDataStore|.
27 // The |browsingDataStore|'s mode must not already be |CHANGING|.
28 - (void)startObservingBrowsingDataStore:(CRWBrowsingDataStore*)browsingDataStore
29 browserState:(web::BrowserState*)browserState;
31 // Stops observing |browsingDataStore| for its |mode| change.
32 // The |browsingDataStore|'s mode must not be |CHANGING|.
33 - (void)stopObservingBrowsingDataStore:(CRWBrowsingDataStore*)browsingDataStore;
36 @implementation CRWBrowsingDataStoreModeObserver
38 @synthesize outOfSyncStoreCount = _outOfSyncStoreCount;
40 - (void)startObservingBrowsingDataStore:(CRWBrowsingDataStore*)browsingDataStore
41 browserState:(web::BrowserState*)browserState {
42 web::BrowsingDataPartition* browsing_data_partition =
43 web::BrowserState::GetBrowsingDataPartition(browserState);
44 DCHECK(browsing_data_partition);
45 DCHECK_EQ(browsing_data_partition->GetBrowsingDataStore(), browsingDataStore);
46 DCHECK_NE(web::CHANGING, browsingDataStore.mode);
48 [browsingDataStore addObserver:self
51 context:browserState];
54 - (void)stopObservingBrowsingDataStore:
55 (CRWBrowsingDataStore*)browsingDataStore {
56 DCHECK_NE(web::CHANGING, browsingDataStore.mode);
58 [browsingDataStore removeObserver:self forKeyPath:@"mode"];
61 - (void)observeValueForKeyPath:(NSString*)keyPath
62 ofObject:(CRWBrowsingDataStore*)browsingDataStore
63 change:(NSDictionary*)change
64 context:(void*)context {
65 DCHECK([keyPath isEqual:@"mode"]);
66 DCHECK([browsingDataStore isKindOfClass:[CRWBrowsingDataStore class]]);
69 if (browsingDataStore.mode == web::CHANGING) {
70 ++self.outOfSyncStoreCount;
73 web::BrowserState* browserState = static_cast<web::BrowserState*>(context);
74 web::ActiveStateManager* activeStateManager =
75 web::BrowserState::GetActiveStateManager(browserState);
76 DCHECK(activeStateManager);
77 bool activeState = activeStateManager->IsActive();
78 // Check if the |browsingDataStore|'s associated ActiveStateManager's active
79 // state is still out of sync.
80 if (activeState && browsingDataStore.mode == web::INACTIVE) {
81 [browsingDataStore makeActiveWithCompletionHandler:nil];
82 } else if (!activeState && browsingDataStore.mode == web::ACTIVE) {
83 [browsingDataStore makeInactiveWithCompletionHandler:nil];
86 DCHECK_GE(self.outOfSyncStoreCount, 1U);
87 --self.outOfSyncStoreCount;
88 web::BrowsingDataPartitionClient* client =
89 web::GetBrowsingDataPartitionClient();
91 client->DidBecomeSynchronized();
100 // The global observer that tracks the number of CRWBrowsingDataStores whose
101 // modes are out of sync with their associated ActiveStateManager's active
103 CRWBrowsingDataStoreModeObserver* g_browsing_data_store_mode_observer = nil;
106 BrowsingDataPartitionImpl::BrowsingDataPartitionImpl(
107 BrowserState* browser_state)
108 : browser_state_(browser_state) {
109 DCHECK_CURRENTLY_ON_WEB_THREAD(WebThread::UI);
110 DCHECK(browser_state);
112 active_state_manager_ = BrowserState::GetActiveStateManager(browser_state);
113 DCHECK(active_state_manager_);
114 active_state_manager_->AddObserver(this);
117 BrowsingDataPartitionImpl::~BrowsingDataPartitionImpl() {
118 if (active_state_manager_) {
119 active_state_manager_->RemoveObserver(this);
121 DCHECK_NE(CHANGING, [browsing_data_store_ mode]);
122 [g_browsing_data_store_mode_observer
123 stopObservingBrowsingDataStore:browsing_data_store_];
127 bool BrowsingDataPartition::IsSynchronized() {
128 return [g_browsing_data_store_mode_observer outOfSyncStoreCount] == 0U;
131 CRWBrowsingDataStore* BrowsingDataPartitionImpl::GetBrowsingDataStore() {
132 DCHECK_CURRENTLY_ON_WEB_THREAD(WebThread::UI);
134 if (!browsing_data_store_) {
135 browsing_data_store_.reset(
136 [[CRWBrowsingDataStore alloc] initWithBrowserState:browser_state_]);
137 if (!g_browsing_data_store_mode_observer) {
138 g_browsing_data_store_mode_observer =
139 [[CRWBrowsingDataStoreModeObserver alloc] init];
141 [g_browsing_data_store_mode_observer
142 startObservingBrowsingDataStore:browsing_data_store_
143 browserState:browser_state_];
145 return browsing_data_store_;
148 void BrowsingDataPartitionImpl::OnActive() {
149 DCHECK_CURRENTLY_ON_WEB_THREAD(WebThread::UI);
151 [GetBrowsingDataStore() makeActiveWithCompletionHandler:nil];
154 void BrowsingDataPartitionImpl::OnInactive() {
155 DCHECK_CURRENTLY_ON_WEB_THREAD(WebThread::UI);
157 [GetBrowsingDataStore() makeInactiveWithCompletionHandler:nil];
160 void BrowsingDataPartitionImpl::WillBeDestroyed() {
161 DCHECK_CURRENTLY_ON_WEB_THREAD(WebThread::UI);
163 active_state_manager_->RemoveObserver(this);
164 active_state_manager_ = nullptr;