Updating trunk VERSION from 2139.0 to 2140.0
[chromium-blink-merge.git] / chrome / browser / supervised_user / supervised_user_navigation_observer.cc
blob1e186b673b1772e5fea986433c5b5fbfe863fa58
1 // Copyright 2014 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 "chrome/browser/supervised_user/supervised_user_navigation_observer.h"
7 #include "base/bind.h"
8 #include "base/callback.h"
9 #include "base/metrics/histogram.h"
10 #include "base/strings/string_util.h"
11 #include "chrome/browser/history/history_service.h"
12 #include "chrome/browser/history/history_service_factory.h"
13 #include "chrome/browser/history/history_types.h"
14 #include "chrome/browser/infobars/infobar_service.h"
15 #include "chrome/browser/profiles/profile.h"
16 #include "chrome/browser/supervised_user/supervised_user_interstitial.h"
17 #include "chrome/browser/supervised_user/supervised_user_resource_throttle.h"
18 #include "chrome/browser/supervised_user/supervised_user_service.h"
19 #include "chrome/browser/supervised_user/supervised_user_service_factory.h"
20 #include "chrome/browser/supervised_user/supervised_user_url_filter.h"
21 #include "chrome/browser/tab_contents/tab_util.h"
22 #include "chrome/grit/generated_resources.h"
23 #include "components/infobars/core/confirm_infobar_delegate.h"
24 #include "components/infobars/core/infobar.h"
25 #include "content/public/browser/browser_thread.h"
26 #include "content/public/browser/navigation_entry.h"
27 #include "content/public/browser/render_frame_host.h"
28 #include "content/public/browser/render_process_host.h"
29 #include "content/public/browser/user_metrics.h"
30 #include "ui/base/l10n/l10n_util.h"
32 #if !defined(OS_ANDROID)
33 #include "chrome/browser/ui/browser.h"
34 #include "chrome/browser/ui/browser_commands.h"
35 #include "chrome/browser/ui/browser_finder.h"
36 #include "chrome/browser/ui/browser_list.h"
37 #include "chrome/browser/ui/host_desktop.h"
38 #include "chrome/browser/ui/tabs/tab_strip_model.h"
39 #endif
41 using base::Time;
42 using content::NavigationEntry;
44 namespace {
47 // Helpers --------------------------------------------------------------------
49 #if !defined(OS_ANDROID)
50 // TODO(bauerb): Get rid of the platform-specific #ifdef here.
51 // http://crbug.com/313377
52 void GoBackToSafety(content::WebContents* web_contents) {
53 // For now, just go back one page (the user didn't retreat from that page,
54 // so it should be okay).
55 content::NavigationController* controller =
56 &web_contents->GetController();
57 if (controller->CanGoBack()) {
58 controller->GoBack();
59 return;
62 // If we can't go back (because we opened a new tab), try to close the tab.
63 // If this is the last tab on this desktop, open a new window.
64 chrome::HostDesktopType host_desktop_type =
65 chrome::GetHostDesktopTypeForNativeView(web_contents->GetNativeView());
66 const BrowserList* browser_list = BrowserList::GetInstance(host_desktop_type);
67 if (browser_list->size() == 1) {
68 Browser* browser = browser_list->get(0);
69 DCHECK(browser == chrome::FindBrowserWithWebContents(web_contents));
70 if (browser->tab_strip_model()->count() == 1)
71 chrome::NewEmptyWindow(browser->profile(), browser->host_desktop_type());
74 web_contents->GetDelegate()->CloseContents(web_contents);
76 #endif
78 // SupervisedUserWarningInfoBarDelegate ---------------------------------------
80 class SupervisedUserWarningInfoBarDelegate : public ConfirmInfoBarDelegate {
81 public:
82 // Creates a supervised user warning infobar and delegate and adds the infobar
83 // to |infobar_service|. Returns the infobar if it was successfully added.
84 static infobars::InfoBar* Create(InfoBarService* infobar_service);
86 private:
87 SupervisedUserWarningInfoBarDelegate();
88 virtual ~SupervisedUserWarningInfoBarDelegate();
90 // ConfirmInfoBarDelegate:
91 virtual bool ShouldExpire(const NavigationDetails& details) const OVERRIDE;
92 virtual void InfoBarDismissed() OVERRIDE;
93 virtual base::string16 GetMessageText() const OVERRIDE;
94 virtual int GetButtons() const OVERRIDE;
95 virtual base::string16 GetButtonLabel(InfoBarButton button) const OVERRIDE;
96 virtual bool Accept() OVERRIDE;
98 DISALLOW_COPY_AND_ASSIGN(SupervisedUserWarningInfoBarDelegate);
101 // static
102 infobars::InfoBar* SupervisedUserWarningInfoBarDelegate::Create(
103 InfoBarService* infobar_service) {
104 return infobar_service->AddInfoBar(ConfirmInfoBarDelegate::CreateInfoBar(
105 scoped_ptr<ConfirmInfoBarDelegate>(
106 new SupervisedUserWarningInfoBarDelegate())));
109 SupervisedUserWarningInfoBarDelegate::SupervisedUserWarningInfoBarDelegate()
110 : ConfirmInfoBarDelegate() {
113 SupervisedUserWarningInfoBarDelegate::~SupervisedUserWarningInfoBarDelegate() {
116 bool SupervisedUserWarningInfoBarDelegate::ShouldExpire(
117 const NavigationDetails& details) const {
118 // SupervisedUserNavigationObserver removes us below.
119 return false;
122 void SupervisedUserWarningInfoBarDelegate::InfoBarDismissed() {
123 content::WebContents* web_contents =
124 InfoBarService::WebContentsFromInfoBar(infobar());
125 SupervisedUserNavigationObserver::FromWebContents(
126 web_contents)->WarnInfoBarDismissed();
129 base::string16 SupervisedUserWarningInfoBarDelegate::GetMessageText() const {
130 return l10n_util::GetStringUTF16(IDS_SUPERVISED_USER_WARN_INFOBAR_MESSAGE);
133 int SupervisedUserWarningInfoBarDelegate::GetButtons() const {
134 return BUTTON_OK;
137 base::string16 SupervisedUserWarningInfoBarDelegate::GetButtonLabel(
138 InfoBarButton button) const {
139 DCHECK_EQ(BUTTON_OK, button);
140 return l10n_util::GetStringUTF16(IDS_SUPERVISED_USER_WARN_INFOBAR_GO_BACK);
143 bool SupervisedUserWarningInfoBarDelegate::Accept() {
144 #if defined(OS_ANDROID)
145 // TODO(bauerb): Get rid of the platform-specific #ifdef here.
146 // http://crbug.com/313377
147 NOTIMPLEMENTED();
148 #else
149 GoBackToSafety(InfoBarService::WebContentsFromInfoBar(infobar()));
150 #endif
152 return false;
156 } // namespace
158 // SupervisedUserNavigationObserver -------------------------------------------
160 DEFINE_WEB_CONTENTS_USER_DATA_KEY(SupervisedUserNavigationObserver);
162 SupervisedUserNavigationObserver::~SupervisedUserNavigationObserver() {
165 SupervisedUserNavigationObserver::SupervisedUserNavigationObserver(
166 content::WebContents* web_contents)
167 : WebContentsObserver(web_contents),
168 warn_infobar_(NULL) {
169 Profile* profile =
170 Profile::FromBrowserContext(web_contents->GetBrowserContext());
171 supervised_user_service_ =
172 SupervisedUserServiceFactory::GetForProfile(profile);
173 url_filter_ = supervised_user_service_->GetURLFilterForUIThread();
176 void SupervisedUserNavigationObserver::WarnInfoBarDismissed() {
177 DCHECK(warn_infobar_);
178 warn_infobar_ = NULL;
181 void SupervisedUserNavigationObserver::ProvisionalChangeToMainFrameUrl(
182 const GURL& url,
183 content::RenderFrameHost* render_frame_host) {
184 SupervisedUserURLFilter::FilteringBehavior behavior =
185 url_filter_->GetFilteringBehaviorForURL(url);
187 if (behavior == SupervisedUserURLFilter::WARN || !warn_infobar_)
188 return;
190 // If we shouldn't have a warn infobar remove it here.
191 InfoBarService::FromWebContents(web_contents())->RemoveInfoBar(warn_infobar_);
192 warn_infobar_ = NULL;
195 void SupervisedUserNavigationObserver::DidCommitProvisionalLoadForFrame(
196 content::RenderFrameHost* render_frame_host,
197 const GURL& url,
198 content::PageTransition transition_type) {
199 if (render_frame_host->GetParent())
200 return;
202 DVLOG(1) << "DidCommitProvisionalLoadForFrame " << url.spec();
203 SupervisedUserURLFilter::FilteringBehavior behavior =
204 url_filter_->GetFilteringBehaviorForURL(url);
206 if (behavior == SupervisedUserURLFilter::WARN && !warn_infobar_) {
207 warn_infobar_ = SupervisedUserWarningInfoBarDelegate::Create(
208 InfoBarService::FromWebContents(web_contents()));
212 // static
213 void SupervisedUserNavigationObserver::OnRequestBlocked(
214 int render_process_host_id,
215 int render_view_id,
216 const GURL& url,
217 const base::Callback<void(bool)>& callback) {
218 content::WebContents* web_contents =
219 tab_util::GetWebContentsByID(render_process_host_id, render_view_id);
220 if (!web_contents) {
221 content::BrowserThread::PostTask(
222 content::BrowserThread::IO, FROM_HERE, base::Bind(callback, false));
223 return;
226 SupervisedUserNavigationObserver* navigation_observer =
227 SupervisedUserNavigationObserver::FromWebContents(web_contents);
228 if (navigation_observer)
229 navigation_observer->OnRequestBlockedInternal(url);
231 // Show the interstitial.
232 SupervisedUserInterstitial::Show(web_contents, url, callback);
235 void SupervisedUserNavigationObserver::OnRequestBlockedInternal(
236 const GURL& url) {
237 Time timestamp = Time::Now(); // TODO(bauerb): Use SaneTime when available.
238 // Create a history entry for the attempt and mark it as such.
239 history::HistoryAddPageArgs add_page_args(
240 url, timestamp, web_contents(), 0,
241 url, history::RedirectList(),
242 content::PAGE_TRANSITION_BLOCKED, history::SOURCE_BROWSED,
243 false);
245 // Add the entry to the history database.
246 Profile* profile =
247 Profile::FromBrowserContext(web_contents()->GetBrowserContext());
248 HistoryService* history_service =
249 HistoryServiceFactory::GetForProfile(profile, Profile::IMPLICIT_ACCESS);
251 // |history_service| is null if saving history is disabled.
252 if (history_service)
253 history_service->AddPage(add_page_args);
255 scoped_ptr<NavigationEntry> entry(NavigationEntry::Create());
256 entry->SetVirtualURL(url);
257 entry->SetTimestamp(timestamp);
258 blocked_navigations_.push_back(entry.release());
259 SupervisedUserService* supervised_user_service =
260 SupervisedUserServiceFactory::GetForProfile(profile);
261 supervised_user_service->DidBlockNavigation(web_contents());