2019-03-22 18:11:51 -04:00
|
|
|
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
|
2015-02-13 23:17:08 +00:00
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#include "libcef/browser/browser_context.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
|
|
|
|
#include <map>
|
|
|
|
#include <utility>
|
|
|
|
|
2015-02-13 23:17:08 +00:00
|
|
|
#include "libcef/browser/content_browser_client.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
#include "libcef/browser/download_manager_delegate.h"
|
2015-07-16 17:40:01 -04:00
|
|
|
#include "libcef/browser/extensions/extension_system.h"
|
2020-03-19 11:34:15 -04:00
|
|
|
#include "libcef/browser/media_router/media_router_manager.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
#include "libcef/browser/prefs/browser_prefs.h"
|
|
|
|
#include "libcef/browser/request_context_impl.h"
|
|
|
|
#include "libcef/browser/ssl_host_state_delegate.h"
|
2015-10-16 20:44:00 -04:00
|
|
|
#include "libcef/browser/thread_util.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
#include "libcef/common/cef_switches.h"
|
2015-07-16 17:40:01 -04:00
|
|
|
#include "libcef/common/extensions/extensions_util.h"
|
2015-02-13 23:17:08 +00:00
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
#include "base/command_line.h"
|
|
|
|
#include "base/files/file_util.h"
|
|
|
|
#include "base/lazy_instance.h"
|
2015-02-13 23:17:08 +00:00
|
|
|
#include "base/logging.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
#include "base/strings/string_util.h"
|
|
|
|
#include "chrome/browser/font_family_cache.h"
|
2017-12-07 16:44:24 -05:00
|
|
|
#include "chrome/browser/plugins/chrome_plugin_service_filter.h"
|
2019-11-12 11:11:44 -05:00
|
|
|
#include "chrome/browser/profiles/profile_key.h"
|
2015-10-16 20:44:00 -04:00
|
|
|
#include "chrome/browser/ui/zoom/chrome_zoom_level_prefs.h"
|
2019-07-17 14:47:27 -04:00
|
|
|
#include "chrome/common/pref_names.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
#include "components/content_settings/core/browser/host_content_settings_map.h"
|
|
|
|
#include "components/guest_view/browser/guest_view_manager.h"
|
2015-02-13 23:17:08 +00:00
|
|
|
#include "components/keyed_service/content/browser_context_dependency_manager.h"
|
2019-06-05 16:15:45 +02:00
|
|
|
#include "components/keyed_service/core/simple_dependency_manager.h"
|
|
|
|
#include "components/keyed_service/core/simple_key_map.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
#include "components/prefs/pref_service.h"
|
|
|
|
#include "components/proxy_config/pref_proxy_config_tracker_impl.h"
|
2015-02-13 23:17:08 +00:00
|
|
|
#include "components/user_prefs/user_prefs.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
#include "components/visitedlink/browser/visitedlink_event_listener.h"
|
2020-02-10 12:10:17 -05:00
|
|
|
#include "components/visitedlink/browser/visitedlink_writer.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
#include "components/zoom/zoom_event_manager.h"
|
|
|
|
#include "content/public/browser/browser_task_traits.h"
|
2015-02-13 23:17:08 +00:00
|
|
|
#include "content/public/browser/browser_thread.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
#include "content/public/browser/download_manager.h"
|
2015-10-16 20:44:00 -04:00
|
|
|
#include "content/public/browser/storage_partition.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
#include "extensions/browser/extension_protocols.h"
|
2017-08-03 18:55:19 -04:00
|
|
|
#include "extensions/browser/process_manager.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
#include "extensions/common/constants.h"
|
|
|
|
#include "net/proxy_resolution/proxy_config_service.h"
|
2019-11-12 11:11:44 -05:00
|
|
|
#include "services/network/public/mojom/cors_origin_pattern.mojom.h"
|
2019-03-22 18:11:51 -04:00
|
|
|
|
|
|
|
using content::BrowserThread;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
// Manages the global list of Impl instances.
|
|
|
|
class ImplManager {
|
|
|
|
public:
|
|
|
|
typedef std::vector<CefBrowserContext*> Vector;
|
|
|
|
|
|
|
|
ImplManager() {}
|
|
|
|
~ImplManager() {
|
|
|
|
DCHECK(all_.empty());
|
|
|
|
DCHECK(map_.empty());
|
|
|
|
}
|
|
|
|
|
|
|
|
void AddImpl(CefBrowserContext* impl) {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
DCHECK(!IsValidImpl(impl));
|
|
|
|
all_.push_back(impl);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RemoveImpl(CefBrowserContext* impl, const base::FilePath& path) {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
|
|
|
|
Vector::iterator it = GetImplPos(impl);
|
|
|
|
DCHECK(it != all_.end());
|
|
|
|
all_.erase(it);
|
|
|
|
|
|
|
|
if (!path.empty()) {
|
|
|
|
PathMap::iterator it = map_.find(path);
|
|
|
|
DCHECK(it != map_.end());
|
|
|
|
if (it != map_.end())
|
|
|
|
map_.erase(it);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsValidImpl(const CefBrowserContext* impl) {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
return GetImplPos(impl) != all_.end();
|
|
|
|
}
|
|
|
|
|
2019-10-01 13:55:16 +00:00
|
|
|
CefBrowserContext* GetImplForIDs(int render_process_id,
|
|
|
|
int render_frame_id,
|
|
|
|
int frame_tree_node_id,
|
|
|
|
bool require_frame_match) {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
for (const auto& context : all_) {
|
|
|
|
if (context->IsAssociatedContext(render_process_id, render_frame_id,
|
|
|
|
frame_tree_node_id,
|
|
|
|
require_frame_match)) {
|
|
|
|
return context;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
CefBrowserContext* GetImplForContext(const content::BrowserContext* context) {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
if (!context)
|
2019-10-01 13:55:16 +00:00
|
|
|
return nullptr;
|
2019-03-22 18:11:51 -04:00
|
|
|
|
|
|
|
Vector::iterator it = all_.begin();
|
|
|
|
for (; it != all_.end(); ++it) {
|
|
|
|
if (*it == context)
|
|
|
|
return *it;
|
|
|
|
}
|
2019-10-01 13:55:16 +00:00
|
|
|
return nullptr;
|
2019-03-22 18:11:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void SetImplPath(CefBrowserContext* impl, const base::FilePath& path) {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
DCHECK(!path.empty());
|
|
|
|
DCHECK(IsValidImpl(impl));
|
2019-10-01 13:55:16 +00:00
|
|
|
DCHECK(GetImplForPath(path) == nullptr);
|
2019-03-22 18:11:51 -04:00
|
|
|
map_.insert(std::make_pair(path, impl));
|
|
|
|
}
|
|
|
|
|
|
|
|
CefBrowserContext* GetImplForPath(const base::FilePath& path) {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
DCHECK(!path.empty());
|
|
|
|
PathMap::const_iterator it = map_.find(path);
|
|
|
|
if (it != map_.end())
|
|
|
|
return it->second;
|
2019-10-01 13:55:16 +00:00
|
|
|
return nullptr;
|
2019-03-22 18:11:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
const Vector GetAllImpl() const { return all_; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
Vector::iterator GetImplPos(const CefBrowserContext* impl) {
|
|
|
|
Vector::iterator it = all_.begin();
|
|
|
|
for (; it != all_.end(); ++it) {
|
|
|
|
if (*it == impl)
|
|
|
|
return it;
|
|
|
|
}
|
|
|
|
return all_.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef std::map<base::FilePath, CefBrowserContext*> PathMap;
|
|
|
|
PathMap map_;
|
|
|
|
|
|
|
|
Vector all_;
|
|
|
|
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(ImplManager);
|
|
|
|
};
|
|
|
|
|
|
|
|
#if DCHECK_IS_ON()
|
|
|
|
// Because of DCHECK()s in the object destructor.
|
|
|
|
base::LazyInstance<ImplManager>::DestructorAtExit g_manager =
|
|
|
|
LAZY_INSTANCE_INITIALIZER;
|
|
|
|
#else
|
|
|
|
base::LazyInstance<ImplManager>::Leaky g_manager = LAZY_INSTANCE_INITIALIZER;
|
|
|
|
#endif
|
|
|
|
|
2020-03-19 11:34:15 -04:00
|
|
|
CefBrowserContext* GetSelf(base::WeakPtr<CefBrowserContext> self) {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
return self.get();
|
|
|
|
}
|
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
// Creates and manages VisitedLinkEventListener objects for each
|
2020-02-10 12:10:17 -05:00
|
|
|
// CefBrowserContext sharing the same VisitedLinkWriter.
|
|
|
|
class CefVisitedLinkListener : public visitedlink::VisitedLinkWriter::Listener {
|
2019-03-22 18:11:51 -04:00
|
|
|
public:
|
|
|
|
CefVisitedLinkListener() { DCHECK(listener_map_.empty()); }
|
|
|
|
|
|
|
|
void CreateListenerForContext(const CefBrowserContext* context) {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
auto listener = std::make_unique<visitedlink::VisitedLinkEventListener>(
|
|
|
|
const_cast<CefBrowserContext*>(context));
|
|
|
|
listener_map_.insert(std::make_pair(context, std::move(listener)));
|
|
|
|
}
|
|
|
|
|
|
|
|
void RemoveListenerForContext(const CefBrowserContext* context) {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
ListenerMap::iterator it = listener_map_.find(context);
|
|
|
|
DCHECK(it != listener_map_.end());
|
|
|
|
listener_map_.erase(it);
|
|
|
|
}
|
|
|
|
|
2020-02-10 12:10:17 -05:00
|
|
|
// visitedlink::VisitedLinkWriter::Listener methods.
|
2019-03-22 18:11:51 -04:00
|
|
|
|
|
|
|
void NewTable(base::ReadOnlySharedMemoryRegion* table_region) override {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
ListenerMap::iterator it = listener_map_.begin();
|
|
|
|
for (; it != listener_map_.end(); ++it)
|
|
|
|
it->second->NewTable(table_region);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Add(visitedlink::VisitedLinkCommon::Fingerprint fingerprint) override {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
ListenerMap::iterator it = listener_map_.begin();
|
|
|
|
for (; it != listener_map_.end(); ++it)
|
|
|
|
it->second->Add(fingerprint);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Reset(bool invalidate_hashes) override {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
ListenerMap::iterator it = listener_map_.begin();
|
|
|
|
for (; it != listener_map_.end(); ++it)
|
|
|
|
it->second->Reset(invalidate_hashes);
|
|
|
|
}
|
2015-02-13 23:17:08 +00:00
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
private:
|
|
|
|
// Map of CefBrowserContext to the associated VisitedLinkEventListener.
|
|
|
|
typedef std::map<const CefBrowserContext*,
|
|
|
|
std::unique_ptr<visitedlink::VisitedLinkEventListener>>
|
|
|
|
ListenerMap;
|
|
|
|
ListenerMap listener_map_;
|
|
|
|
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(CefVisitedLinkListener);
|
|
|
|
};
|
|
|
|
|
|
|
|
CefBrowserContext::CefBrowserContext(const CefRequestContextSettings& settings)
|
2020-03-19 11:34:15 -04:00
|
|
|
: settings_(settings), weak_ptr_factory_(this) {
|
2019-03-22 18:11:51 -04:00
|
|
|
g_manager.Get().AddImpl(this);
|
2020-03-19 11:34:15 -04:00
|
|
|
getter_ = base::BindRepeating(GetSelf, weak_ptr_factory_.GetWeakPtr());
|
2019-03-22 18:11:51 -04:00
|
|
|
}
|
2015-02-13 23:17:08 +00:00
|
|
|
|
|
|
|
CefBrowserContext::~CefBrowserContext() {
|
2019-03-22 18:11:51 -04:00
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
|
|
|
|
// No CefRequestContext should be referencing this object any longer.
|
|
|
|
DCHECK(request_context_set_.empty());
|
|
|
|
|
|
|
|
// Unregister the context first to avoid re-entrancy during shutdown.
|
|
|
|
g_manager.Get().RemoveImpl(this, cache_path_);
|
|
|
|
|
2020-03-19 11:34:15 -04:00
|
|
|
// Destroy objects that may hold references to the MediaRouter.
|
|
|
|
media_router_manager_.reset();
|
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
// Send notifications to clean up objects associated with this Profile.
|
|
|
|
MaybeSendDestroyedNotification();
|
|
|
|
|
2019-10-01 13:55:16 +00:00
|
|
|
ChromePluginServiceFilter::GetInstance()->UnregisterProfile(this);
|
2019-03-22 18:11:51 -04:00
|
|
|
|
|
|
|
// Remove any BrowserContextKeyedServiceFactory associations. This must be
|
|
|
|
// called before the ProxyService owned by CefBrowserContext is destroyed.
|
2019-06-05 16:15:45 +02:00
|
|
|
// The SimpleDependencyManager should always be passed after the
|
|
|
|
// BrowserContextDependencyManager. This is because the KeyedService instances
|
|
|
|
// in the BrowserContextDependencyManager's dependency graph can depend on the
|
|
|
|
// ones in the SimpleDependencyManager's graph.
|
|
|
|
DependencyManager::PerformInterlockedTwoPhaseShutdown(
|
|
|
|
BrowserContextDependencyManager::GetInstance(), this,
|
|
|
|
SimpleDependencyManager::GetInstance(), key_.get());
|
|
|
|
|
|
|
|
key_.reset();
|
|
|
|
SimpleKeyMap::GetInstance()->Dissociate(this);
|
2019-03-22 18:11:51 -04:00
|
|
|
|
|
|
|
// Shuts down the storage partitions associated with this browser context.
|
|
|
|
// This must be called before the browser context is actually destroyed
|
|
|
|
// and before a clean-up task for its corresponding IO thread residents
|
|
|
|
// (e.g. ResourceContext) is posted, so that the classes that hung on
|
|
|
|
// StoragePartition can have time to do necessary cleanups on IO thread.
|
|
|
|
ShutdownStoragePartitions();
|
|
|
|
|
|
|
|
if (resource_context_.get()) {
|
|
|
|
// Destruction of the ResourceContext will trigger destruction of all
|
2019-07-29 17:27:12 -04:00
|
|
|
// associated network requests.
|
2019-03-22 18:11:51 -04:00
|
|
|
content::BrowserThread::DeleteSoon(content::BrowserThread::IO, FROM_HERE,
|
|
|
|
resource_context_.release());
|
|
|
|
}
|
|
|
|
|
|
|
|
visitedlink_listener_->RemoveListenerForContext(this);
|
|
|
|
|
|
|
|
// The FontFamilyCache references the ProxyService so delete it before the
|
|
|
|
// ProxyService is deleted.
|
2019-10-01 13:55:16 +00:00
|
|
|
SetUserData(&kFontFamilyCacheKey, nullptr);
|
2019-03-22 18:11:51 -04:00
|
|
|
|
|
|
|
pref_proxy_config_tracker_->DetachFromPrefService();
|
|
|
|
|
|
|
|
if (host_content_settings_map_)
|
|
|
|
host_content_settings_map_->ShutdownOnUIThread();
|
|
|
|
|
|
|
|
// Delete the download manager delegate here because otherwise we'll crash
|
|
|
|
// when it's accessed from the content::BrowserContext destructor.
|
|
|
|
if (download_manager_delegate_)
|
2019-10-01 13:55:16 +00:00
|
|
|
download_manager_delegate_.reset(nullptr);
|
2015-02-13 23:17:08 +00:00
|
|
|
}
|
|
|
|
|
2015-07-16 17:40:01 -04:00
|
|
|
void CefBrowserContext::Initialize() {
|
2019-03-22 18:11:51 -04:00
|
|
|
cache_path_ = base::FilePath(CefString(&settings_.cache_path));
|
|
|
|
|
|
|
|
if (!cache_path_.empty())
|
|
|
|
g_manager.Get().SetImplPath(this, cache_path_);
|
|
|
|
|
2019-07-25 15:27:15 -04:00
|
|
|
if (!!settings_.persist_session_cookies) {
|
|
|
|
set_should_persist_session_cookies(true);
|
|
|
|
}
|
|
|
|
|
2020-04-23 17:45:04 -04:00
|
|
|
key_ = std::make_unique<ProfileKey>(cache_path_);
|
2019-06-05 16:15:45 +02:00
|
|
|
SimpleKeyMap::GetInstance()->Associate(this, key_.get());
|
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
// Initialize the PrefService object.
|
|
|
|
pref_service_ = browser_prefs::CreatePrefService(
|
|
|
|
this, cache_path_, !!settings_.persist_user_preferences);
|
|
|
|
|
2019-03-23 19:40:32 -04:00
|
|
|
resource_context_.reset(new CefResourceContext(IsOffTheRecord()));
|
2017-04-20 15:28:17 -04:00
|
|
|
|
|
|
|
// This must be called before creating any services to avoid hitting
|
|
|
|
// DependencyManager::AssertContextWasntDestroyed when creating/destroying
|
|
|
|
// multiple browser contexts (due to pointer address reuse).
|
|
|
|
BrowserContextDependencyManager::GetInstance()->CreateBrowserContextServices(
|
|
|
|
this);
|
|
|
|
|
2015-07-16 17:40:01 -04:00
|
|
|
const bool extensions_enabled = extensions::ExtensionsEnabled();
|
|
|
|
if (extensions_enabled) {
|
|
|
|
// Create the custom ExtensionSystem first because other KeyedServices
|
|
|
|
// depend on it.
|
|
|
|
extension_system_ = static_cast<extensions::CefExtensionSystem*>(
|
|
|
|
extensions::ExtensionSystem::Get(this));
|
2019-03-22 18:11:51 -04:00
|
|
|
extension_system_->InitForRegularProfile(true);
|
2017-08-03 18:55:19 -04:00
|
|
|
|
|
|
|
// Make sure the ProcessManager is created so that it receives extension
|
|
|
|
// load notifications. This is necessary for the proper initialization of
|
|
|
|
// background/event pages.
|
|
|
|
extensions::ProcessManager::Get(this);
|
2015-07-16 17:40:01 -04:00
|
|
|
}
|
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
// Initialize visited links management.
|
|
|
|
base::FilePath visited_link_path;
|
|
|
|
if (!cache_path_.empty())
|
|
|
|
visited_link_path = cache_path_.Append(FILE_PATH_LITERAL("Visited Links"));
|
|
|
|
visitedlink_listener_ = new CefVisitedLinkListener;
|
2020-02-10 12:10:17 -05:00
|
|
|
visitedlink_master_.reset(new visitedlink::VisitedLinkWriter(
|
2019-03-22 18:11:51 -04:00
|
|
|
visitedlink_listener_, this, !visited_link_path.empty(), false,
|
|
|
|
visited_link_path, 0));
|
|
|
|
visitedlink_listener_->CreateListenerForContext(this);
|
|
|
|
visitedlink_master_->Init();
|
|
|
|
|
|
|
|
// Initialize proxy configuration tracker.
|
|
|
|
pref_proxy_config_tracker_.reset(new PrefProxyConfigTrackerImpl(
|
2019-10-01 13:55:16 +00:00
|
|
|
GetPrefs(), base::CreateSingleThreadTaskRunner({BrowserThread::IO})));
|
2019-03-22 18:11:51 -04:00
|
|
|
|
2015-07-16 17:40:01 -04:00
|
|
|
// Spell checking support and possibly other subsystems retrieve the
|
|
|
|
// PrefService associated with a BrowserContext via UserPrefs::Get().
|
2015-09-09 16:05:39 +02:00
|
|
|
PrefService* pref_service = GetPrefs();
|
2015-07-16 17:40:01 -04:00
|
|
|
DCHECK(pref_service);
|
|
|
|
user_prefs::UserPrefs::Set(this, pref_service);
|
2019-06-05 16:15:45 +02:00
|
|
|
key_->SetPrefs(pref_service);
|
2015-07-16 17:40:01 -04:00
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
if (extensions_enabled)
|
2015-07-16 17:40:01 -04:00
|
|
|
extension_system_->Init();
|
2017-12-07 16:44:24 -05:00
|
|
|
|
2019-10-01 13:55:16 +00:00
|
|
|
ChromePluginServiceFilter::GetInstance()->RegisterProfile(this);
|
2019-03-22 18:11:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void CefBrowserContext::AddCefRequestContext(CefRequestContextImpl* context) {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
request_context_set_.insert(context);
|
2015-07-16 17:40:01 -04:00
|
|
|
}
|
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
void CefBrowserContext::RemoveCefRequestContext(
|
|
|
|
CefRequestContextImpl* context) {
|
2015-10-16 20:44:00 -04:00
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
if (extensions::ExtensionsEnabled()) {
|
|
|
|
extension_system()->OnRequestContextDeleted(context);
|
|
|
|
}
|
2016-08-04 15:37:53 +03:00
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
request_context_set_.erase(context);
|
2017-12-07 16:44:24 -05:00
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
// Delete ourselves when the reference count reaches zero.
|
|
|
|
if (request_context_set_.empty())
|
|
|
|
delete this;
|
|
|
|
}
|
2016-08-31 14:25:56 +03:00
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
// static
|
|
|
|
CefBrowserContext* CefBrowserContext::GetForCachePath(
|
|
|
|
const base::FilePath& cache_path) {
|
|
|
|
return g_manager.Get().GetImplForPath(cache_path);
|
|
|
|
}
|
|
|
|
|
2019-10-01 13:55:16 +00:00
|
|
|
// static
|
|
|
|
CefBrowserContext* CefBrowserContext::GetForIDs(int render_process_id,
|
|
|
|
int render_frame_id,
|
|
|
|
int frame_tree_node_id,
|
|
|
|
bool require_frame_match) {
|
|
|
|
return g_manager.Get().GetImplForIDs(render_process_id, render_frame_id,
|
|
|
|
frame_tree_node_id, require_frame_match);
|
|
|
|
}
|
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
// static
|
|
|
|
CefBrowserContext* CefBrowserContext::GetForContext(
|
|
|
|
content::BrowserContext* context) {
|
|
|
|
return g_manager.Get().GetImplForContext(context);
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
std::vector<CefBrowserContext*> CefBrowserContext::GetAll() {
|
|
|
|
return g_manager.Get().GetAllImpl();
|
2015-10-16 20:44:00 -04:00
|
|
|
}
|
|
|
|
|
2015-02-13 23:17:08 +00:00
|
|
|
content::ResourceContext* CefBrowserContext::GetResourceContext() {
|
|
|
|
return resource_context_.get();
|
|
|
|
}
|
2015-10-16 20:44:00 -04:00
|
|
|
|
2019-02-01 16:42:40 +00:00
|
|
|
content::ClientHintsControllerDelegate*
|
|
|
|
CefBrowserContext::GetClientHintsControllerDelegate() {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2019-02-01 14:40:18 -05:00
|
|
|
void CefBrowserContext::SetCorsOriginAccessListForOrigin(
|
|
|
|
const url::Origin& source_origin,
|
|
|
|
std::vector<network::mojom::CorsOriginPatternPtr> allow_patterns,
|
|
|
|
std::vector<network::mojom::CorsOriginPatternPtr> block_patterns,
|
|
|
|
base::OnceClosure closure) {
|
|
|
|
// This method is called for Extension support.
|
|
|
|
base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, std::move(closure));
|
|
|
|
}
|
|
|
|
|
2015-10-16 20:44:00 -04:00
|
|
|
ChromeZoomLevelPrefs* CefBrowserContext::GetZoomLevelPrefs() {
|
|
|
|
return static_cast<ChromeZoomLevelPrefs*>(
|
2019-10-01 13:55:16 +00:00
|
|
|
GetStoragePartition(this, nullptr)->GetZoomLevelDelegate());
|
2015-10-16 20:44:00 -04:00
|
|
|
}
|
2016-10-17 14:14:44 -04:00
|
|
|
|
2018-07-23 18:32:02 -04:00
|
|
|
scoped_refptr<network::SharedURLLoaderFactory>
|
|
|
|
CefBrowserContext::GetURLLoaderFactory() {
|
|
|
|
return GetDefaultStoragePartition(this)
|
|
|
|
->GetURLLoaderFactoryForBrowserProcess();
|
|
|
|
}
|
|
|
|
|
2019-09-04 15:13:32 +00:00
|
|
|
base::FilePath CefBrowserContext::GetPath() {
|
|
|
|
return cache_path_;
|
|
|
|
}
|
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
base::FilePath CefBrowserContext::GetPath() const {
|
|
|
|
return cache_path_;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<content::ZoomLevelDelegate>
|
|
|
|
CefBrowserContext::CreateZoomLevelDelegate(
|
|
|
|
const base::FilePath& partition_path) {
|
|
|
|
if (cache_path_.empty())
|
|
|
|
return std::unique_ptr<content::ZoomLevelDelegate>();
|
|
|
|
|
|
|
|
return base::WrapUnique(new ChromeZoomLevelPrefs(
|
|
|
|
GetPrefs(), cache_path_, partition_path,
|
|
|
|
zoom::ZoomEventManager::GetForBrowserContext(this)->GetWeakPtr()));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CefBrowserContext::IsOffTheRecord() const {
|
|
|
|
// CEF contexts are never flagged as off-the-record. It causes problems
|
|
|
|
// for the extension system.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
content::DownloadManagerDelegate*
|
|
|
|
CefBrowserContext::GetDownloadManagerDelegate() {
|
|
|
|
if (!download_manager_delegate_) {
|
|
|
|
content::DownloadManager* manager =
|
|
|
|
BrowserContext::GetDownloadManager(this);
|
|
|
|
download_manager_delegate_.reset(new CefDownloadManagerDelegate(manager));
|
|
|
|
}
|
|
|
|
return download_manager_delegate_.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
content::BrowserPluginGuestManager* CefBrowserContext::GetGuestManager() {
|
|
|
|
DCHECK(extensions::ExtensionsEnabled());
|
|
|
|
return guest_view::GuestViewManager::FromBrowserContext(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
storage::SpecialStoragePolicy* CefBrowserContext::GetSpecialStoragePolicy() {
|
2019-10-01 13:55:16 +00:00
|
|
|
return nullptr;
|
2019-03-22 18:11:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
content::PushMessagingService* CefBrowserContext::GetPushMessagingService() {
|
2019-10-01 13:55:16 +00:00
|
|
|
return nullptr;
|
2019-03-22 18:11:51 -04:00
|
|
|
}
|
|
|
|
|
2019-11-12 11:11:44 -05:00
|
|
|
content::StorageNotificationService*
|
|
|
|
CefBrowserContext::GetStorageNotificationService() {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
content::SSLHostStateDelegate* CefBrowserContext::GetSSLHostStateDelegate() {
|
|
|
|
if (!ssl_host_state_delegate_.get())
|
|
|
|
ssl_host_state_delegate_.reset(new CefSSLHostStateDelegate());
|
|
|
|
return ssl_host_state_delegate_.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
content::PermissionControllerDelegate*
|
|
|
|
CefBrowserContext::GetPermissionControllerDelegate() {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
content::BackgroundFetchDelegate*
|
|
|
|
CefBrowserContext::GetBackgroundFetchDelegate() {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
content::BackgroundSyncController*
|
|
|
|
CefBrowserContext::GetBackgroundSyncController() {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
content::BrowsingDataRemoverDelegate*
|
|
|
|
CefBrowserContext::GetBrowsingDataRemoverDelegate() {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
PrefService* CefBrowserContext::GetPrefs() {
|
|
|
|
return pref_service_.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
const PrefService* CefBrowserContext::GetPrefs() const {
|
|
|
|
return pref_service_.get();
|
|
|
|
}
|
|
|
|
|
2019-06-05 16:15:45 +02:00
|
|
|
ProfileKey* CefBrowserContext::GetProfileKey() const {
|
|
|
|
DCHECK(key_);
|
|
|
|
return key_.get();
|
2019-03-22 18:11:51 -04:00
|
|
|
}
|
|
|
|
|
2019-07-16 13:59:21 -04:00
|
|
|
policy::SchemaRegistryService*
|
|
|
|
CefBrowserContext::GetPolicySchemaRegistryService() {
|
|
|
|
NOTREACHED();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
policy::UserCloudPolicyManager* CefBrowserContext::GetUserCloudPolicyManager() {
|
|
|
|
NOTREACHED();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
policy::ProfilePolicyConnector* CefBrowserContext::GetProfilePolicyConnector() {
|
|
|
|
NOTREACHED();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const policy::ProfilePolicyConnector*
|
|
|
|
CefBrowserContext::GetProfilePolicyConnector() const {
|
|
|
|
NOTREACHED();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2019-03-22 18:11:51 -04:00
|
|
|
const CefRequestContextSettings& CefBrowserContext::GetSettings() const {
|
|
|
|
return settings_;
|
|
|
|
}
|
|
|
|
|
|
|
|
HostContentSettingsMap* CefBrowserContext::GetHostContentSettingsMap() {
|
|
|
|
DCHECK_CURRENTLY_ON(BrowserThread::UI);
|
|
|
|
if (!host_content_settings_map_.get()) {
|
|
|
|
// The |is_incognito_profile| and |is_guest_profile| arguments are
|
|
|
|
// intentionally set to false as they otherwise limit the types of values
|
|
|
|
// that can be stored in the settings map (for example, default values set
|
|
|
|
// via DefaultProvider::SetWebsiteSetting).
|
|
|
|
host_content_settings_map_ =
|
2020-06-09 13:48:00 -04:00
|
|
|
new HostContentSettingsMap(GetPrefs(), false, false, false, false);
|
2019-03-22 18:11:51 -04:00
|
|
|
|
|
|
|
// Change the default plugin policy.
|
|
|
|
const base::CommandLine* command_line =
|
|
|
|
base::CommandLine::ForCurrentProcess();
|
|
|
|
const std::string& plugin_policy_str =
|
|
|
|
command_line->GetSwitchValueASCII(switches::kPluginPolicy);
|
|
|
|
if (!plugin_policy_str.empty()) {
|
|
|
|
ContentSetting plugin_policy = CONTENT_SETTING_ALLOW;
|
|
|
|
if (base::LowerCaseEqualsASCII(plugin_policy_str,
|
|
|
|
switches::kPluginPolicy_Detect)) {
|
|
|
|
plugin_policy = CONTENT_SETTING_DETECT_IMPORTANT_CONTENT;
|
|
|
|
} else if (base::LowerCaseEqualsASCII(plugin_policy_str,
|
|
|
|
switches::kPluginPolicy_Block)) {
|
|
|
|
plugin_policy = CONTENT_SETTING_BLOCK;
|
|
|
|
}
|
|
|
|
host_content_settings_map_->SetDefaultContentSetting(
|
2020-02-10 12:10:17 -05:00
|
|
|
ContentSettingsType::PLUGINS, plugin_policy);
|
2019-03-22 18:11:51 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return host_content_settings_map_.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CefBrowserContext::AddVisitedURLs(const std::vector<GURL>& urls) {
|
|
|
|
visitedlink_master_->AddURLs(urls);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CefBrowserContext::RebuildTable(
|
|
|
|
const scoped_refptr<URLEnumerator>& enumerator) {
|
|
|
|
// Called when visited links will not or cannot be loaded from disk.
|
|
|
|
enumerator->OnComplete(true);
|
|
|
|
}
|
|
|
|
|
2019-03-23 19:40:32 -04:00
|
|
|
void CefBrowserContext::OnRenderFrameCreated(
|
|
|
|
CefRequestContextImpl* request_context,
|
|
|
|
int render_process_id,
|
|
|
|
int render_frame_id,
|
2019-04-24 02:50:25 +00:00
|
|
|
int frame_tree_node_id,
|
2019-03-23 19:40:32 -04:00
|
|
|
bool is_main_frame,
|
|
|
|
bool is_guest_view) {
|
|
|
|
CEF_REQUIRE_UIT();
|
2019-04-24 02:50:25 +00:00
|
|
|
DCHECK_GE(render_process_id, 0);
|
|
|
|
DCHECK_GE(render_frame_id, 0);
|
|
|
|
DCHECK_GE(frame_tree_node_id, 0);
|
|
|
|
|
2019-10-01 13:55:16 +00:00
|
|
|
render_id_set_.insert(std::make_pair(render_process_id, render_frame_id));
|
|
|
|
node_id_set_.insert(frame_tree_node_id);
|
|
|
|
|
2019-03-23 19:40:32 -04:00
|
|
|
CefRefPtr<CefRequestContextHandler> handler = request_context->GetHandler();
|
2019-10-01 13:55:16 +00:00
|
|
|
if (handler) {
|
|
|
|
handler_map_.AddHandler(render_process_id, render_frame_id,
|
|
|
|
frame_tree_node_id, handler);
|
|
|
|
|
|
|
|
if (resource_context_) {
|
|
|
|
// Using base::Unretained() is safe because both this callback and
|
|
|
|
// possible deletion of |resource_context_| will execute on the IO thread,
|
|
|
|
// and this callback will be executed first.
|
|
|
|
CEF_POST_TASK(CEF_IOT,
|
|
|
|
base::Bind(&CefResourceContext::AddHandler,
|
|
|
|
base::Unretained(resource_context_.get()),
|
|
|
|
render_process_id, render_frame_id,
|
|
|
|
frame_tree_node_id, handler));
|
|
|
|
}
|
2019-03-23 19:40:32 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CefBrowserContext::OnRenderFrameDeleted(
|
|
|
|
CefRequestContextImpl* request_context,
|
|
|
|
int render_process_id,
|
|
|
|
int render_frame_id,
|
2019-04-24 02:50:25 +00:00
|
|
|
int frame_tree_node_id,
|
2019-03-23 19:40:32 -04:00
|
|
|
bool is_main_frame,
|
|
|
|
bool is_guest_view) {
|
2017-02-14 17:27:19 -05:00
|
|
|
CEF_REQUIRE_UIT();
|
2019-04-24 02:50:25 +00:00
|
|
|
DCHECK_GE(render_process_id, 0);
|
|
|
|
DCHECK_GE(render_frame_id, 0);
|
|
|
|
DCHECK_GE(frame_tree_node_id, 0);
|
|
|
|
|
2019-10-01 13:55:16 +00:00
|
|
|
auto it1 =
|
|
|
|
render_id_set_.find(std::make_pair(render_process_id, render_frame_id));
|
|
|
|
if (it1 != render_id_set_.end())
|
|
|
|
render_id_set_.erase(it1);
|
|
|
|
|
|
|
|
auto it2 = node_id_set_.find(frame_tree_node_id);
|
|
|
|
if (it2 != node_id_set_.end())
|
|
|
|
node_id_set_.erase(it2);
|
|
|
|
|
2019-03-23 19:40:32 -04:00
|
|
|
CefRefPtr<CefRequestContextHandler> handler = request_context->GetHandler();
|
2019-10-01 13:55:16 +00:00
|
|
|
if (handler) {
|
|
|
|
handler_map_.RemoveHandler(render_process_id, render_frame_id,
|
|
|
|
frame_tree_node_id);
|
|
|
|
|
|
|
|
if (resource_context_) {
|
|
|
|
// Using base::Unretained() is safe because both this callback and
|
|
|
|
// possible deletion of |resource_context_| will execute on the IO thread,
|
|
|
|
// and this callback will be executed first.
|
|
|
|
CEF_POST_TASK(
|
|
|
|
CEF_IOT,
|
|
|
|
base::Bind(&CefResourceContext::RemoveHandler,
|
|
|
|
base::Unretained(resource_context_.get()),
|
|
|
|
render_process_id, render_frame_id, frame_tree_node_id));
|
|
|
|
}
|
2019-03-23 19:40:32 -04:00
|
|
|
}
|
|
|
|
|
2019-10-01 13:55:16 +00:00
|
|
|
if (is_main_frame) {
|
|
|
|
ClearPluginLoadDecision(render_process_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CefRefPtr<CefRequestContextHandler> CefBrowserContext::GetHandler(
|
|
|
|
int render_process_id,
|
|
|
|
int render_frame_id,
|
|
|
|
int frame_tree_node_id,
|
|
|
|
bool require_frame_match) const {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
return handler_map_.GetHandler(render_process_id, render_frame_id,
|
|
|
|
frame_tree_node_id, require_frame_match);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CefBrowserContext::IsAssociatedContext(int render_process_id,
|
|
|
|
int render_frame_id,
|
|
|
|
int frame_tree_node_id,
|
|
|
|
bool require_frame_match) const {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
|
|
|
|
if (render_process_id >= 0 && render_frame_id >= 0) {
|
|
|
|
const auto it1 =
|
|
|
|
render_id_set_.find(std::make_pair(render_process_id, render_frame_id));
|
|
|
|
if (it1 != render_id_set_.end())
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (frame_tree_node_id >= 0) {
|
|
|
|
const auto it2 = node_id_set_.find(frame_tree_node_id);
|
|
|
|
if (it2 != node_id_set_.end())
|
|
|
|
return true;
|
2016-10-17 14:14:44 -04:00
|
|
|
}
|
2019-10-01 13:55:16 +00:00
|
|
|
|
|
|
|
if (render_process_id >= 0 && !require_frame_match) {
|
|
|
|
// Choose an arbitrary handler for the same process.
|
|
|
|
for (const auto& render_ids : render_id_set_) {
|
|
|
|
if (render_ids.first == render_process_id)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2016-10-17 14:14:44 -04:00
|
|
|
}
|
|
|
|
|
2019-10-01 13:55:16 +00:00
|
|
|
void CefBrowserContext::AddPluginLoadDecision(
|
|
|
|
int render_process_id,
|
|
|
|
const base::FilePath& plugin_path,
|
|
|
|
bool is_main_frame,
|
|
|
|
const url::Origin& main_frame_origin,
|
|
|
|
chrome::mojom::PluginStatus status) {
|
2017-02-14 17:27:19 -05:00
|
|
|
CEF_REQUIRE_UIT();
|
2019-10-01 13:55:16 +00:00
|
|
|
DCHECK_GE(render_process_id, 0);
|
|
|
|
DCHECK(!plugin_path.empty());
|
|
|
|
|
|
|
|
plugin_load_decision_map_.insert(std::make_pair(
|
|
|
|
std::make_pair(std::make_pair(render_process_id, plugin_path),
|
|
|
|
std::make_pair(is_main_frame, main_frame_origin)),
|
|
|
|
status));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CefBrowserContext::HasPluginLoadDecision(
|
|
|
|
int render_process_id,
|
|
|
|
const base::FilePath& plugin_path,
|
|
|
|
bool is_main_frame,
|
|
|
|
const url::Origin& main_frame_origin,
|
|
|
|
chrome::mojom::PluginStatus* status) const {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
DCHECK_GE(render_process_id, 0);
|
|
|
|
DCHECK(!plugin_path.empty());
|
|
|
|
|
|
|
|
PluginLoadDecisionMap::const_iterator it = plugin_load_decision_map_.find(
|
|
|
|
std::make_pair(std::make_pair(render_process_id, plugin_path),
|
|
|
|
std::make_pair(is_main_frame, main_frame_origin)));
|
|
|
|
if (it == plugin_load_decision_map_.end())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
*status = it->second;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CefBrowserContext::ClearPluginLoadDecision(int render_process_id) {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
|
|
|
|
if (render_process_id == -1) {
|
|
|
|
plugin_load_decision_map_.clear();
|
|
|
|
} else {
|
|
|
|
PluginLoadDecisionMap::iterator it = plugin_load_decision_map_.begin();
|
|
|
|
while (it != plugin_load_decision_map_.end()) {
|
|
|
|
if (it->first.first.first == render_process_id)
|
|
|
|
it = plugin_load_decision_map_.erase(it);
|
|
|
|
else
|
|
|
|
++it;
|
|
|
|
}
|
2017-02-14 17:27:19 -05:00
|
|
|
}
|
2016-10-17 14:14:44 -04:00
|
|
|
}
|
2019-04-24 02:50:25 +00:00
|
|
|
|
|
|
|
void CefBrowserContext::RegisterSchemeHandlerFactory(
|
|
|
|
const std::string& scheme_name,
|
|
|
|
const std::string& domain_name,
|
|
|
|
CefRefPtr<CefSchemeHandlerFactory> factory) {
|
|
|
|
if (resource_context_) {
|
|
|
|
// Using base::Unretained() is safe because both this callback and possible
|
|
|
|
// deletion of |resource_context_| will execute on the IO thread, and this
|
|
|
|
// callback will be executed first.
|
|
|
|
CEF_POST_TASK(CEF_IOT,
|
|
|
|
base::Bind(&CefResourceContext::RegisterSchemeHandlerFactory,
|
|
|
|
base::Unretained(resource_context_.get()),
|
|
|
|
scheme_name, domain_name, factory));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CefBrowserContext::ClearSchemeHandlerFactories() {
|
|
|
|
if (resource_context_) {
|
|
|
|
// Using base::Unretained() is safe because both this callback and possible
|
|
|
|
// deletion of |resource_context_| will execute on the IO thread, and this
|
|
|
|
// callback will be executed first.
|
|
|
|
CEF_POST_TASK(CEF_IOT,
|
|
|
|
base::Bind(&CefResourceContext::ClearSchemeHandlerFactories,
|
|
|
|
base::Unretained(resource_context_.get())));
|
|
|
|
}
|
2019-05-18 20:41:14 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
network::mojom::NetworkContext* CefBrowserContext::GetNetworkContext() {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
return GetDefaultStoragePartition(this)->GetNetworkContext();
|
|
|
|
}
|
2019-07-17 14:47:27 -04:00
|
|
|
|
|
|
|
DownloadPrefs* CefBrowserContext::GetDownloadPrefs() {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
if (!download_prefs_) {
|
|
|
|
download_prefs_.reset(new DownloadPrefs(this));
|
|
|
|
}
|
|
|
|
return download_prefs_.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CefBrowserContext::IsPrintPreviewSupported() const {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
if (!extensions::PrintPreviewEnabled())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return !GetPrefs()->GetBoolean(prefs::kPrintPreviewDisabled);
|
|
|
|
}
|
2020-03-19 11:34:15 -04:00
|
|
|
|
|
|
|
CefMediaRouterManager* CefBrowserContext::GetMediaRouterManager() {
|
|
|
|
CEF_REQUIRE_UIT();
|
|
|
|
if (!media_router_manager_) {
|
|
|
|
media_router_manager_.reset(new CefMediaRouterManager(this));
|
|
|
|
}
|
|
|
|
return media_router_manager_.get();
|
|
|
|
}
|