1 // Copyright 2013 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 #include "ios/web/navigation/navigation_manager_impl.h"
7 #include "base/logging.h"
8 #import "ios/web/navigation/crw_session_controller+private_constructors.h"
9 #import "ios/web/navigation/crw_session_controller.h"
10 #import "ios/web/navigation/crw_session_entry.h"
11 #include "ios/web/navigation/navigation_item_impl.h"
12 #include "ios/web/navigation/navigation_manager_delegate.h"
13 #import "ios/web/navigation/navigation_manager_facade_delegate.h"
14 #include "ios/web/public/load_committed_details.h"
15 #include "ios/web/public/navigation_item.h"
16 #include "ios/web/public/web_state/web_state.h"
17 #include "ui/base/page_transition_types.h"
21 // Checks whether or not two URL are an in-page navigation (differing only
23 bool AreURLsInPageNavigation(const GURL& existing_url, const GURL& new_url) {
24 if (existing_url == new_url || !new_url.has_ref())
27 url::Replacements<char> replacements;
28 replacements.ClearRef();
29 return existing_url.ReplaceComponents(replacements) ==
30 new_url.ReplaceComponents(replacements);
33 } // anonymous namespace
37 NavigationManagerImpl::NavigationManagerImpl(
38 NavigationManagerDelegate* delegate,
39 BrowserState* browser_state)
40 : delegate_(delegate),
41 browser_state_(browser_state),
42 facade_delegate_(nullptr) {
43 DCHECK(browser_state_);
46 NavigationManagerImpl::~NavigationManagerImpl() {
47 // The facade layer should be deleted before this object.
48 DCHECK(!facade_delegate_);
50 [session_controller_ setNavigationManager:nullptr];
53 CRWSessionController* NavigationManagerImpl::GetSessionController() {
54 return session_controller_;
57 void NavigationManagerImpl::SetSessionController(
58 CRWSessionController* session_controller) {
59 session_controller_.reset([session_controller retain]);
60 [session_controller_ setNavigationManager:this];
63 void NavigationManagerImpl::InitializeSession(NSString* window_name,
66 int opener_navigation_index) {
67 SetSessionController([[[CRWSessionController alloc]
68 initWithWindowName:window_name
70 openedByDOM:opened_by_dom
71 openerNavigationIndex:opener_navigation_index
72 browserState:browser_state_] autorelease]);
75 void NavigationManagerImpl::ReplaceSessionHistory(
76 ScopedVector<web::NavigationItem> items,
78 SetSessionController([[[CRWSessionController alloc]
79 initWithNavigationItems:items.Pass()
80 currentIndex:current_index
81 browserState:browser_state_] autorelease]);
84 void NavigationManagerImpl::SetFacadeDelegate(
85 NavigationManagerFacadeDelegate* facade_delegate) {
86 facade_delegate_ = facade_delegate;
89 NavigationManagerFacadeDelegate* NavigationManagerImpl::GetFacadeDelegate()
91 return facade_delegate_;
95 void NavigationManagerImpl::OnNavigationItemChanged() {
97 facade_delegate_->OnNavigationItemChanged();
100 void NavigationManagerImpl::OnNavigationItemCommitted() {
101 LoadCommittedDetails details;
102 details.item = GetLastCommittedItem();
103 DCHECK(details.item);
104 details.previous_item_index = [session_controller_ previousNavigationIndex];
105 if (details.previous_item_index >= 0) {
106 DCHECK([session_controller_ previousEntry]);
107 details.previous_url =
108 [session_controller_ previousEntry].navigationItem->GetURL();
110 AreURLsInPageNavigation(details.previous_url, details.item->GetURL());
112 details.previous_url = GURL();
113 details.is_in_page = NO;
116 delegate_->OnNavigationItemCommitted(details);
118 if (facade_delegate_) {
119 facade_delegate_->OnNavigationItemCommitted(details.previous_item_index,
124 NavigationItem* NavigationManagerImpl::GetVisibleItem() const {
125 CRWSessionEntry* entry = [session_controller_ visibleEntry];
126 return [entry navigationItem];
129 NavigationItem* NavigationManagerImpl::GetPendingItem() const {
130 return [[session_controller_ pendingEntry] navigationItem];
133 NavigationItem* NavigationManagerImpl::GetTransientItem() const {
134 return [[session_controller_ transientEntry] navigationItem];
137 NavigationItem* NavigationManagerImpl::GetLastCommittedItem() const {
138 CRWSessionEntry* entry = [session_controller_ lastCommittedEntry];
139 return [entry navigationItem];
142 NavigationItem* NavigationManagerImpl::GetItemAtIndex(size_t index) const {
143 NSArray* entries = [session_controller_ entries];
144 return index < entries.count ? [entries[index] navigationItem] : nullptr;
147 int NavigationManagerImpl::GetCurrentEntryIndex() const {
148 return [session_controller_ currentNavigationIndex];
151 int NavigationManagerImpl::GetLastCommittedEntryIndex() const {
152 if (![[session_controller_ entries] count])
154 return [session_controller_ currentNavigationIndex];
157 int NavigationManagerImpl::GetEntryCount() const {
158 return [[session_controller_ entries] count];
161 bool NavigationManagerImpl::RemoveEntryAtIndex(int index) {
162 if (index == GetLastCommittedEntryIndex() ||
163 index == GetPendingEntryIndex())
166 NSUInteger idx = static_cast<NSUInteger>(index);
167 NSArray* entries = [session_controller_ entries];
168 if (idx >= entries.count)
171 [session_controller_ removeEntryAtIndex:index];
175 void NavigationManagerImpl::DiscardNonCommittedEntries() {
176 [session_controller_ discardNonCommittedEntries];
179 NavigationItem* NavigationManagerImpl::GetLastUserItem() const {
180 CRWSessionEntry* entry = [session_controller_ lastUserEntry];
181 return [entry navigationItem];
184 NavigationItem* NavigationManagerImpl::GetPreviousItem() const {
185 CRWSessionEntry* entry = [session_controller_ previousEntry];
186 return [entry navigationItem];
189 int NavigationManagerImpl::GetPendingEntryIndex() const {
190 if ([session_controller_ hasPendingEntry])
191 return GetCurrentEntryIndex();
195 void NavigationManagerImpl::LoadURL(const GURL& url,
196 const web::Referrer& referrer,
197 ui::PageTransition type) {
198 WebState::OpenURLParams params(url, referrer, CURRENT_TAB, type, NO);
199 delegate_->GetWebState()->OpenURL(params);
202 bool NavigationManagerImpl::CanGoBack() const {
203 return [session_controller_ canGoBack];
206 bool NavigationManagerImpl::CanGoForward() const {
207 return [session_controller_ canGoForward];
210 void NavigationManagerImpl::GoBack() {
212 [session_controller_ goBack];
213 // Signal the delegate to load the old page.
214 delegate_->NavigateToPendingEntry();
218 void NavigationManagerImpl::GoForward() {
219 if (CanGoForward()) {
220 [session_controller_ goForward];
221 // Signal the delegate to load the new page.
222 delegate_->NavigateToPendingEntry();
226 std::vector<NavigationItem*> NavigationManagerImpl::GetItems() {
227 std::vector<NavigationItem*> items;
229 items.resize([session_controller_ entries].count);
230 for (CRWSessionEntry* entry in [session_controller_ entries]) {
231 items[i++] = entry.navigationItem;
236 BrowserState* NavigationManagerImpl::GetBrowserState() const {
237 return browser_state_;
240 WebState* NavigationManagerImpl::GetWebState() const {
241 return delegate_->GetWebState();
244 void NavigationManagerImpl::CopyState(
245 NavigationManagerImpl* navigation_manager) {
246 SetSessionController(
247 [[navigation_manager->GetSessionController() copy] autorelease]);