| // Copyright 2014 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. |
| |
| #include "android_webview/native/permission/permission_request_handler.h" |
| |
| #include "android_webview/native/permission/aw_permission_request.h" |
| #include "android_webview/native/permission/aw_permission_request_delegate.h" |
| #include "android_webview/native/permission/permission_request_handler_client.h" |
| #include "base/android/scoped_java_ref.h" |
| #include "base/bind.h" |
| #include "content/public/browser/navigation_details.h" |
| #include "content/public/browser/navigation_entry.h" |
| #include "content/public/browser/web_contents.h" |
| |
| using base::android::ScopedJavaLocalRef; |
| |
| namespace android_webview { |
| |
| namespace { |
| |
| int GetActiveEntryID(content::WebContents* web_contents) { |
| if (!web_contents) return 0; |
| |
| content::NavigationEntry* active_entry = |
| web_contents->GetController().GetActiveEntry(); |
| return active_entry ? active_entry->GetUniqueID() : 0; |
| } |
| |
| } // namespace |
| |
| PermissionRequestHandler::PermissionRequestHandler( |
| PermissionRequestHandlerClient* client, content::WebContents* web_contents) |
| : content::WebContentsObserver(web_contents), |
| client_(client), |
| contents_unique_id_(GetActiveEntryID(web_contents)) { |
| } |
| |
| PermissionRequestHandler::~PermissionRequestHandler() { |
| CancelAllRequests(); |
| } |
| |
| void PermissionRequestHandler::SendRequest( |
| scoped_ptr<AwPermissionRequestDelegate> request) { |
| if (Preauthorized(request->GetOrigin(), request->GetResources())) { |
| request->NotifyRequestResult(true); |
| return; |
| } |
| |
| base::WeakPtr<AwPermissionRequest> weak_request; |
| base::android::ScopedJavaLocalRef<jobject> java_peer = |
| AwPermissionRequest::Create(request.Pass(), &weak_request); |
| requests_.push_back(weak_request); |
| client_->OnPermissionRequest(java_peer, weak_request.get()); |
| PruneRequests(); |
| } |
| |
| void PermissionRequestHandler::CancelRequest(const GURL& origin, |
| int64 resources) { |
| // The request list might have multiple requests with same origin and |
| // resources. |
| RequestIterator i = FindRequest(origin, resources); |
| while (i != requests_.end()) { |
| CancelRequestInternal(i); |
| requests_.erase(i); |
| i = FindRequest(origin, resources); |
| } |
| } |
| |
| void PermissionRequestHandler::PreauthorizePermission(const GURL& origin, |
| int64 resources) { |
| if (!resources) |
| return; |
| |
| std::string key = origin.GetOrigin().spec(); |
| if (key.empty()) { |
| LOG(ERROR) << "The origin of preauthorization is empty, ignore it."; |
| return; |
| } |
| |
| preauthorized_permission_[key] |= resources; |
| } |
| |
| void PermissionRequestHandler::NavigationEntryCommitted( |
| const content::LoadCommittedDetails& details) { |
| const ui::PageTransition transition = details.entry->GetTransitionType(); |
| if (details.is_navigation_to_different_page() || |
| ui::PageTransitionStripQualifier(transition) == |
| ui::PAGE_TRANSITION_RELOAD || |
| contents_unique_id_ != details.entry->GetUniqueID()) { |
| CancelAllRequests(); |
| contents_unique_id_ = details.entry->GetUniqueID(); |
| } |
| } |
| |
| PermissionRequestHandler::RequestIterator |
| PermissionRequestHandler::FindRequest(const GURL& origin, |
| int64 resources) { |
| RequestIterator i; |
| for (i = requests_.begin(); i != requests_.end(); ++i) { |
| if (i->get() && i->get()->GetOrigin() == origin && |
| i->get()->GetResources() == resources) { |
| break; |
| } |
| } |
| return i; |
| } |
| |
| void PermissionRequestHandler::CancelRequestInternal(RequestIterator i) { |
| AwPermissionRequest* request = i->get(); |
| if (request) { |
| client_->OnPermissionRequestCanceled(request); |
| request->CancelAndDelete(); |
| } |
| } |
| |
| void PermissionRequestHandler::CancelAllRequests() { |
| for (RequestIterator i = requests_.begin(); i != requests_.end(); ++i) |
| CancelRequestInternal(i); |
| } |
| |
| void PermissionRequestHandler::PruneRequests() { |
| for (RequestIterator i = requests_.begin(); i != requests_.end();) { |
| if (!i->get()) |
| i = requests_.erase(i); |
| else |
| ++i; |
| } |
| } |
| |
| bool PermissionRequestHandler::Preauthorized(const GURL& origin, |
| int64 resources) { |
| std::map<std::string, int64>::iterator i = |
| preauthorized_permission_.find(origin.GetOrigin().spec()); |
| |
| return i != preauthorized_permission_.end() && |
| (resources & i->second) == resources; |
| } |
| |
| } // namespace android_webivew |