Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1039)

Side by Side Diff: chrome/browser/browser_navigator.cc

Issue 4647007: Fix navigation bugs, patch the right file, nuke an old file. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Created 10 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
(Empty)
1 // Copyright (c) 2010 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.
4
5 #include "chrome/browser/browser_navigator.h"
6
7 #include "base/command_line.h"
8 #include "chrome/browser/browser.h"
9 #include "chrome/browser/browser_list.h"
10 #include "chrome/browser/browser_url_handler.h"
11 #include "chrome/browser/browser_window.h"
12 #include "chrome/browser/location_bar.h"
13 #include "chrome/browser/profile.h"
14 #include "chrome/browser/renderer_host/site_instance.h"
15 #include "chrome/browser/status_bubble.h"
16 #include "chrome/browser/tabs/tab_strip_model.h"
17 #include "chrome/browser/tab_contents/tab_contents.h"
18 #include "chrome/common/chrome_switches.h"
19
20 namespace {
21
22 // Returns the SiteInstance for |source_contents| if it represents the same
23 // website as |url|, or NULL otherwise. |source_contents| cannot be NULL.
24 SiteInstance* GetSiteInstance(TabContents* source_contents, const GURL& url) {
25 if (!source_contents)
26 return NULL;
27
28 // Don't use this logic when "--process-per-tab" is specified.
29 if (!CommandLine::ForCurrentProcess()->HasSwitch(switches::kProcessPerTab) &&
30 SiteInstance::IsSameWebSite(source_contents->profile(),
31 source_contents->GetURL(),
32 url)) {
33 return source_contents->GetSiteInstance();
34 }
35 return NULL;
36 }
37
38 // Returns true if the specified Browser can open tabs. Not all Browsers support
39 // multiple tabs, such as app frames and popups. This function returns false for
40 // those types of Browser.
41 bool WindowCanOpenTabs(Browser* browser) {
42 return browser->CanSupportWindowFeature(Browser::FEATURE_TABSTRIP) ||
43 browser->tabstrip_model()->empty();
44 }
45
46 // Finds an existing Browser compatible with |profile|, making a new one if no
47 // such Browser is located.
48 Browser* GetOrCreateBrowser(Profile* profile) {
49 Browser* browser = BrowserList::FindBrowserWithType(profile,
50 Browser::TYPE_NORMAL,
51 false);
52 return browser ? browser : Browser::Create(profile);
53 }
54
55 // Returns true if two URLs are equal ignoring their ref (hash fragment).
56 bool CompareURLsIgnoreRef(const GURL& url, const GURL& other) {
57 if (url == other)
58 return true;
59 // If neither has a ref than there is no point in stripping the refs and
60 // the URLs are different since the comparison failed in the previous if
61 // statement.
62 if (!url.has_ref() && !other.has_ref())
63 return false;
64 url_canon::Replacements<char> replacements;
65 replacements.ClearRef();
66 GURL url_no_ref = url.ReplaceComponents(replacements);
67 GURL other_no_ref = other.ReplaceComponents(replacements);
68 return url_no_ref == other_no_ref;
69 }
70
71 // Returns the index of an existing singleton tab in |params->browser| matching
72 // the URL specified in |params|.
73 int GetIndexOfSingletonTab(browser::NavigateParams* params) {
74 if (params->disposition != SINGLETON_TAB)
75 return -1;
76
77 // In case the URL was rewritten by the BrowserURLHandler we need to ensure
78 // that we do not open another URL that will get redirected to the rewritten
79 // URL.
80 GURL rewritten_url(params->url);
81 bool reverse_on_redirect = false;
82 BrowserURLHandler::RewriteURLIfNecessary(&rewritten_url,
83 params->browser->profile(),
84 &reverse_on_redirect);
85
86 for (int i = 0; i < params->browser->tab_count(); ++i) {
87 TabContents* tab = params->browser->GetTabContentsAt(i);
88 if (CompareURLsIgnoreRef(tab->GetURL(), params->url) ||
89 CompareURLsIgnoreRef(tab->GetURL(), rewritten_url)) {
90 params->target_contents = tab;
91 return i;
92 }
93 }
94 return -1;
95 }
96
97 // Returns a Browser that can host the navigation or tab addition specified in
98 // |params|. This might just return the same Browser specified in |params|, or
99 // some other if that Browser is deemed incompatible.
100 Browser* GetBrowserForDisposition(browser::NavigateParams* params) {
101 // If no source TabContents was specified, we use the selected one from the
102 // target browser. This must happen first, before GetBrowserForDisposition()
103 // has a chance to replace |params->browser| with another one.
104 if (!params->source_contents && params->browser)
105 params->source_contents = params->browser->GetSelectedTabContents();
106
107 Profile* profile =
108 params->browser ? params->browser->profile() : params->profile;
109
110 switch (params->disposition) {
111 case CURRENT_TAB:
112 if (!params->browser && profile) {
113 // We specified a profile instead of a browser; find or create one.
114 params->browser = Browser::GetOrCreateTabbedBrowser(profile);
115 }
116 return params->browser;
117 case SINGLETON_TAB:
118 case NEW_FOREGROUND_TAB:
119 case NEW_BACKGROUND_TAB:
120 // See if we can open the tab in the window this navigator is bound to.
121 if (params->browser && WindowCanOpenTabs(params->browser))
122 return params->browser;
123 // Find a compatible window and re-execute this command in it. Otherwise
124 // re-run with NEW_WINDOW.
125 if (profile)
126 return GetOrCreateBrowser(profile);
127 return NULL;
128 case NEW_POPUP: {
129 // Make a new popup window. Coerce app-style if |params->browser| or the
130 // |source| represents an app.
131 Browser::Type type = Browser::TYPE_POPUP;
132 if ((params->browser && params->browser->type() == Browser::TYPE_APP) ||
133 (params->source_contents && params->source_contents->is_app())) {
134 type = Browser::TYPE_APP_POPUP;
135 }
136 if (profile) {
137 Browser* browser = new Browser(type, profile);
138 browser->set_override_bounds(params->window_bounds);
139 browser->CreateBrowserWindow();
140 return browser;
141 }
142 return NULL;
143 }
144 case NEW_WINDOW:
145 // Make a new normal browser window.
146 if (profile) {
147 Browser* browser = new Browser(Browser::TYPE_NORMAL, profile);
148 browser->CreateBrowserWindow();
149 return browser;
150 }
151 return NULL;
152 case OFF_THE_RECORD:
153 // Make or find an incognito window.
154 if (profile)
155 return GetOrCreateBrowser(profile->GetOffTheRecordProfile());
156 return NULL;
157 // The following types all result in no navigation.
158 case SUPPRESS_OPEN:
159 case SAVE_TO_DISK:
160 case IGNORE_ACTION:
161 return NULL;
162 default:
163 NOTREACHED();
164 }
165 return NULL;
166 }
167
168 // Fix disposition and other parameter values depending on prevailing
169 // conditions.
170 void NormalizeDisposition(browser::NavigateParams* params) {
171 // Calculate the WindowOpenDisposition if necessary.
172 if (params->browser->tabstrip_model()->empty() &&
173 (params->disposition == NEW_BACKGROUND_TAB ||
174 params->disposition == CURRENT_TAB ||
175 params->disposition == SINGLETON_TAB)) {
176 params->disposition = NEW_FOREGROUND_TAB;
177 }
178 if (params->browser->profile()->IsOffTheRecord() &&
179 params->disposition == OFF_THE_RECORD) {
180 params->disposition = NEW_FOREGROUND_TAB;
181 }
182
183 // Disposition trumps add types. ADD_SELECTED is a default, so we need to
184 // remove it if disposition implies the tab is going to open in the
185 // background.
186 if (params->disposition == NEW_BACKGROUND_TAB)
187 params->tabstrip_add_types &= ~TabStripModel::ADD_SELECTED;
188 else if (params->disposition == NEW_FOREGROUND_TAB)
189 params->tabstrip_add_types |= TabStripModel::ADD_SELECTED;
190
191 // Code that wants to open a new window typically expects it to be shown
192 // automatically.
193 if (params->disposition == NEW_WINDOW || params->disposition == NEW_POPUP) {
194 params->show_window = true;
195 params->tabstrip_add_types |= TabStripModel::ADD_SELECTED;
196 }
197 }
198
199 // This class makes sure the Browser object held in |params| is made visible
200 // by the time it goes out of scope, provided |params| wants it to be shown.
201 class ScopedBrowserDisplayer {
202 public:
203 explicit ScopedBrowserDisplayer(browser::NavigateParams* params)
204 : params_(params) {
205 }
206 ~ScopedBrowserDisplayer() {
207 if (params_->show_window)
208 params_->browser->window()->Show();
209 }
210 private:
211 browser::NavigateParams* params_;
212 DISALLOW_COPY_AND_ASSIGN(ScopedBrowserDisplayer);
213 };
214
215 // This class manages the lifetime of a TabContents created by the Navigate()
216 // function. When Navigate() creates a TabContents for a URL, an instance of
217 // this class takes ownership of it via TakeOwnership() until the TabContents
218 // is added to a tab strip at which time ownership is relinquished via
219 // ReleaseOwnership(). If this object goes out of scope without being added
220 // to a tab strip, the created TabContents is deleted to avoid a leak and the
221 // params->target_contents field is set to NULL.
222 class ScopedTargetContentsOwner {
223 public:
224 explicit ScopedTargetContentsOwner(browser::NavigateParams* params)
225 : params_(params) {
226 }
227 ~ScopedTargetContentsOwner() {
228 if (target_contents_owner_.get())
229 params_->target_contents = NULL;
230 }
231
232 // Assumes ownership of |params_|' target_contents until ReleaseOwnership
233 // is called.
234 void TakeOwnership() {
235 target_contents_owner_.reset(params_->target_contents);
236 }
237
238 // Relinquishes ownership of |params_|' target_contents.
239 TabContents* ReleaseOwnership() {
240 return target_contents_owner_.release();
241 }
242
243 private:
244 browser::NavigateParams* params_;
245 scoped_ptr<TabContents> target_contents_owner_;
246 DISALLOW_COPY_AND_ASSIGN(ScopedTargetContentsOwner);
247 };
248
249 } // namespace
250
251 namespace browser {
252
253 NavigateParams::NavigateParams(
254 Browser* a_browser,
255 const GURL& a_url,
256 PageTransition::Type a_transition)
257 : url(a_url),
258 target_contents(NULL),
259 source_contents(NULL),
260 disposition(CURRENT_TAB),
261 transition(a_transition),
262 tabstrip_index(-1),
263 tabstrip_add_types(TabStripModel::ADD_SELECTED),
264 show_window(false),
265 browser(a_browser),
266 profile(NULL) {
267 }
268
269 NavigateParams::NavigateParams(Browser* a_browser,
270 TabContents* a_target_contents)
271 : target_contents(a_target_contents),
272 source_contents(NULL),
273 disposition(CURRENT_TAB),
274 transition(PageTransition::LINK),
275 tabstrip_index(-1),
276 tabstrip_add_types(TabStripModel::ADD_SELECTED),
277 show_window(false),
278 browser(a_browser),
279 profile(NULL) {
280 }
281
282 NavigateParams::~NavigateParams() {
283 }
284
285 void Navigate(NavigateParams* params) {
286 params->browser = GetBrowserForDisposition(params);
287 if (!params->browser)
288 return;
289 // Navigate() must not return early after this point.
290
291 // Make sure the Browser is shown if params call for it.
292 ScopedBrowserDisplayer displayer(params);
293
294 // Makes sure any TabContents created by this function is destroyed if
295 // not properly added to a tab strip.
296 ScopedTargetContentsOwner target_contents_owner(params);
297
298 // Some dispositions need coercion to base types.
299 NormalizeDisposition(params);
300
301 // Determine if the navigation was user initiated. If it was, we need to
302 // inform the target TabContents, and we may need to update the UI.
303 PageTransition::Type base_transition =
304 PageTransition::StripQualifier(params->transition);
305 bool user_initiated = base_transition == PageTransition::TYPED ||
306 base_transition == PageTransition::AUTO_BOOKMARK;
307
308 // If no target TabContents was specified, we need to construct one if we are
309 // supposed to target a new tab.
310 if (!params->target_contents) {
311 if (params->disposition != CURRENT_TAB) {
312 params->target_contents =
313 new TabContents(params->browser->profile(),
314 GetSiteInstance(params->source_contents, params->url),
315 MSG_ROUTING_NONE,
316 params->source_contents,
317 NULL);
318 // This function takes ownership of |params->target_contents| until it
319 // is added to a TabStripModel.
320 target_contents_owner.TakeOwnership();
321 params->target_contents->SetExtensionAppById(params->extension_app_id);
322 // TODO(sky): figure out why this is needed. Without it we seem to get
323 // failures in startup tests.
324 // By default, content believes it is not hidden. When adding contents
325 // in the background, tell it that it's hidden.
326 if ((params->tabstrip_add_types & TabStripModel::ADD_SELECTED) == 0) {
327 // TabStripModel::AddTabContents invokes HideContents if not foreground.
328 params->target_contents->WasHidden();
329 }
330 } else {
331 // ... otherwise if we're loading in the current tab, the target is the
332 // same as the source.
333 params->target_contents = params->source_contents;
334 DCHECK(params->target_contents);
335 }
336
337 if (user_initiated) {
338 RenderViewHostDelegate::BrowserIntegration* integration =
339 params->target_contents;
340 integration->OnUserGesture();
341 }
342
343 // Perform the actual navigation.
344 GURL url = params->url.is_empty() ? params->browser->GetHomePage()
345 : params->url;
346 params->target_contents->controller().LoadURL(url, params->referrer,
347 params->transition);
348 } else {
349 // |target_contents| was specified non-NULL, and so we assume it has already
350 // been navigated appropriately. We need to do nothing more other than
351 // add it to the appropriate tabstrip.
352 }
353
354 if (params->source_contents == params->target_contents) {
355 // The navigation occurred in the source tab, so update the UI.
356 params->browser->UpdateUIForNavigationInTab(params->target_contents,
357 params->transition,
358 user_initiated);
359 } else {
360 // The navigation occurred in some other tab.
361 int singleton_index = GetIndexOfSingletonTab(params);
362 if (params->disposition == SINGLETON_TAB && singleton_index >= 0) {
363 // The navigation should re-select an existing tab in the target Browser.
364 params->browser->SelectTabContentsAt(singleton_index, user_initiated);
365 } else {
366 // If some non-default value is set for the index, we should tell the
367 // TabStripModel to respect it.
368 if (params->tabstrip_index != -1)
369 params->tabstrip_add_types |= TabStripModel::ADD_FORCE_INDEX;
370
371 // The navigation should insert a new tab into the target Browser.
372 params->browser->tabstrip_model()->AddTabContents(
373 params->target_contents,
374 params->tabstrip_index,
375 params->transition,
376 params->tabstrip_add_types);
377 // Now that the |params->target_contents| is safely owned by the target
378 // Browser's TabStripModel, we can release ownership.
379 target_contents_owner.ReleaseOwnership();
380 }
381 }
382 }
383
384 } // namespace browser
OLDNEW
« no previous file with comments | « no previous file | chrome/browser/ui/browser_navigator.cc » ('j') | chrome/browser/ui/browser_navigator.cc » ('J')

Powered by Google App Engine
This is Rietveld 408576698