// Copyright (c) 2012 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 "chrome/browser/ui/webui/inspect_ui.h" #include #include "base/bind.h" #include "base/bind_helpers.h" #include "base/json/json_writer.h" #include "base/memory/ref_counted_memory.h" #include "base/strings/string_number_conversions.h" #include "base/strings/string_util.h" #include "base/strings/stringprintf.h" #include "base/strings/utf_string_conversions.h" #include "base/values.h" #include "chrome/browser/devtools/devtools_window.h" #include "chrome/browser/devtools/port_forwarding_controller.h" #include "chrome/browser/extensions/extension_service.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/ui/tab_contents/tab_contents_iterator.h" #include "chrome/browser/ui/webui/theme_source.h" #include "chrome/common/pref_names.h" #include "chrome/common/url_constants.h" #include "content/public/browser/browser_child_process_observer.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/child_process_data.h" #include "content/public/browser/devtools_agent_host.h" #include "content/public/browser/devtools_client_host.h" #include "content/public/browser/devtools_manager.h" #include "content/public/browser/favicon_status.h" #include "content/public/browser/navigation_entry.h" #include "content/public/browser/notification_service.h" #include "content/public/browser/notification_source.h" #include "content/public/browser/notification_types.h" #include "content/public/browser/render_process_host.h" #include "content/public/browser/render_view_host.h" #include "content/public/browser/web_contents.h" #include "content/public/browser/web_ui.h" #include "content/public/browser/web_ui_data_source.h" #include "content/public/browser/web_ui_message_handler.h" #include "content/public/browser/worker_service.h" #include "content/public/browser/worker_service_observer.h" #include "content/public/common/process_type.h" #include "grit/browser_resources.h" #include "grit/generated_resources.h" #include "net/base/escape.h" #include "net/base/net_errors.h" #include "ui/base/resource/resource_bundle.h" using content::BrowserThread; using content::ChildProcessData; using content::DevToolsAgentHost; using content::DevToolsClientHost; using content::DevToolsManager; using content::RenderProcessHost; using content::RenderViewHost; using content::RenderViewHostDelegate; using content::RenderWidgetHost; using content::WebContents; using content::WebUIMessageHandler; using content::WorkerService; using content::WorkerServiceObserver; namespace { const char kAppTargetType[] = "app"; const char kExtensionTargetType[] = "extension"; const char kPageTargetType[] = "page"; const char kWorkerTargetType[] = "worker"; const char kAdbTargetType[] = "adb_page"; const char kInitUICommand[] = "init-ui"; const char kInspectCommand[] = "inspect"; const char kActivateCommand[] = "activate"; const char kTerminateCommand[] = "terminate"; const char kReloadCommand[] = "reload"; const char kOpenCommand[] = "open"; const char kPortForwardingEnabledCommand[] = "set-port-forwarding-enabled"; const char kPortForwardingConfigCommand[] = "set-port-forwarding-config"; const char kPortForwardingDefaultPort[] = "8080"; const char kPortForwardingDefaultLocation[] = "localhost:8080"; const char kTargetTypeField[] = "type"; const char kAttachedField[] = "attached"; const char kProcessIdField[] = "processId"; const char kRouteIdField[] = "routeId"; const char kUrlField[] = "url"; const char kNameField[] = "name"; const char kFaviconUrlField[] = "faviconUrl"; const char kDescription[] = "description"; const char kPidField[] = "pid"; const char kAdbSerialField[] = "adbSerial"; const char kAdbModelField[] = "adbModel"; const char kAdbBrowserProductField[] = "adbBrowserProduct"; const char kAdbBrowserPackageField[] = "adbBrowserPackage"; const char kAdbBrowserVersionField[] = "adbBrowserVersion"; const char kAdbGlobalIdField[] = "adbGlobalId"; const char kAdbBrowsersField[] = "browsers"; const char kAdbPagesField[] = "pages"; const char kAdbPortStatus[] = "adbPortStatus"; const char kAdbScreenWidthField[] = "adbScreenWidth"; const char kAdbScreenHeightField[] = "adbScreenHeight"; const char kAdbAttachedForeignField[] = "adbAttachedForeign"; const char kGuestList[] = "guests"; DictionaryValue* BuildTargetDescriptor( const std::string& target_type, bool attached, const GURL& url, const std::string& name, const GURL& favicon_url, const std::string& description, int process_id, int route_id, base::ProcessHandle handle = base::kNullProcessHandle) { DictionaryValue* target_data = new DictionaryValue(); target_data->SetString(kTargetTypeField, target_type); target_data->SetBoolean(kAttachedField, attached); target_data->SetInteger(kProcessIdField, process_id); target_data->SetInteger(kRouteIdField, route_id); target_data->SetString(kUrlField, url.spec()); target_data->SetString(kNameField, net::EscapeForHTML(name)); target_data->SetInteger(kPidField, base::GetProcId(handle)); target_data->SetString(kFaviconUrlField, favicon_url.spec()); target_data->SetString(kDescription, description); return target_data; } bool HasClientHost(RenderViewHost* rvh) { if (!DevToolsAgentHost::HasFor(rvh)) return false; scoped_refptr agent( DevToolsAgentHost::GetOrCreateFor(rvh)); return agent->IsAttached(); } bool HasClientHost(int process_id, int route_id) { if (!DevToolsAgentHost::HasForWorker(process_id, route_id)) return false; scoped_refptr agent( DevToolsAgentHost::GetForWorker(process_id, route_id)); return agent->IsAttached(); } DictionaryValue* BuildTargetDescriptor(RenderViewHost* rvh, bool is_tab) { WebContents* web_contents = WebContents::FromRenderViewHost(rvh); std::string title; std::string target_type = is_tab ? kPageTargetType : ""; GURL url; GURL favicon_url; if (web_contents) { url = web_contents->GetURL(); title = UTF16ToUTF8(web_contents->GetTitle()); content::NavigationController& controller = web_contents->GetController(); content::NavigationEntry* entry = controller.GetActiveEntry(); if (entry != NULL && entry->GetURL().is_valid()) favicon_url = entry->GetFavicon().url; Profile* profile = Profile::FromBrowserContext( web_contents->GetBrowserContext()); if (profile) { ExtensionService* extension_service = profile->GetExtensionService(); const extensions::Extension* extension = extension_service-> extensions()->GetByID(url.host()); if (extension) { if (extension->is_hosted_app() || extension->is_legacy_packaged_app() || extension->is_platform_app()) target_type = kAppTargetType; else target_type = kExtensionTargetType; title = extension->name(); } } } return BuildTargetDescriptor(target_type, HasClientHost(rvh), url, title, favicon_url, "", rvh->GetProcess()->GetID(), rvh->GetRoutingID()); } class InspectMessageHandler : public WebUIMessageHandler { public: explicit InspectMessageHandler(InspectUI* inspect_ui) : inspect_ui_(inspect_ui) {} virtual ~InspectMessageHandler() {} private: // WebUIMessageHandler implementation. virtual void RegisterMessages() OVERRIDE; void HandleInitUICommand(const ListValue* args); void HandleInspectCommand(const ListValue* args); void HandleActivateCommand(const ListValue* args); void HandleTerminateCommand(const ListValue* args); void HandleReloadCommand(const ListValue* args); void HandleOpenCommand(const ListValue* args); void HandlePortForwardingEnabledCommand(const ListValue* args); void HandlePortForwardingConfigCommand(const ListValue* args); static bool GetProcessAndRouteId(const ListValue* args, int* process_id, int* route_id); static bool GetRemotePageId(const ListValue* args, std::string* page_id); InspectUI* inspect_ui_; DISALLOW_COPY_AND_ASSIGN(InspectMessageHandler); }; void InspectMessageHandler::RegisterMessages() { web_ui()->RegisterMessageCallback(kInitUICommand, base::Bind(&InspectMessageHandler::HandleInitUICommand, base::Unretained(this))); web_ui()->RegisterMessageCallback(kInspectCommand, base::Bind(&InspectMessageHandler::HandleInspectCommand, base::Unretained(this))); web_ui()->RegisterMessageCallback(kActivateCommand, base::Bind(&InspectMessageHandler::HandleActivateCommand, base::Unretained(this))); web_ui()->RegisterMessageCallback(kTerminateCommand, base::Bind(&InspectMessageHandler::HandleTerminateCommand, base::Unretained(this))); web_ui()->RegisterMessageCallback(kPortForwardingEnabledCommand, base::Bind(&InspectMessageHandler::HandlePortForwardingEnabledCommand, base::Unretained(this))); web_ui()->RegisterMessageCallback(kPortForwardingConfigCommand, base::Bind(&InspectMessageHandler::HandlePortForwardingConfigCommand, base::Unretained(this))); web_ui()->RegisterMessageCallback(kReloadCommand, base::Bind(&InspectMessageHandler::HandleReloadCommand, base::Unretained(this))); web_ui()->RegisterMessageCallback(kOpenCommand, base::Bind(&InspectMessageHandler::HandleOpenCommand, base::Unretained(this))); } void InspectMessageHandler::HandleInitUICommand(const ListValue*) { inspect_ui_->InitUI(); } void InspectMessageHandler::HandleInspectCommand(const ListValue* args) { Profile* profile = Profile::FromWebUI(web_ui()); if (!profile) return; std::string page_id; if (GetRemotePageId(args, &page_id)) { inspect_ui_->InspectRemotePage(page_id); return; } int process_id; int route_id; if (!GetProcessAndRouteId(args, &process_id, &route_id) || process_id == 0 || route_id == 0) { return; } RenderViewHost* rvh = RenderViewHost::FromID(process_id, route_id); if (rvh) { DevToolsWindow::OpenDevToolsWindow(rvh); return; } scoped_refptr agent_host( DevToolsAgentHost::GetForWorker(process_id, route_id)); if (!agent_host.get()) return; DevToolsWindow::OpenDevToolsWindowForWorker(profile, agent_host.get()); } void InspectMessageHandler::HandleActivateCommand(const ListValue* args) { std::string page_id; if (GetRemotePageId(args, &page_id)) inspect_ui_->ActivateRemotePage(page_id); } static void TerminateWorker(int process_id, int route_id) { WorkerService::GetInstance()->TerminateWorker(process_id, route_id); } void InspectMessageHandler::HandleTerminateCommand(const ListValue* args) { std::string page_id; if (GetRemotePageId(args, &page_id)) { inspect_ui_->CloseRemotePage(page_id); return; } int process_id; int route_id; if (!GetProcessAndRouteId(args, &process_id, &route_id)) return; BrowserThread::PostTask(BrowserThread::IO, FROM_HERE, base::Bind(&TerminateWorker, process_id, route_id)); } void InspectMessageHandler::HandleReloadCommand(const ListValue* args) { std::string page_id; if (GetRemotePageId(args, &page_id)) inspect_ui_->ReloadRemotePage(page_id); } void InspectMessageHandler::HandleOpenCommand(const ListValue* args) { std::string browser_id; std::string url; if (args->GetSize() == 2 && args->GetString(0, &browser_id) && args->GetString(1, &url)) { inspect_ui_->OpenRemotePage(browser_id, url); } } bool InspectMessageHandler::GetProcessAndRouteId(const ListValue* args, int* process_id, int* route_id) { const DictionaryValue* data; if (args->GetSize() == 1 && args->GetDictionary(0, &data) && data->GetInteger(kProcessIdField, process_id) && data->GetInteger(kRouteIdField, route_id)) { return true; } return false; } void InspectMessageHandler::HandlePortForwardingEnabledCommand( const ListValue* args) { Profile* profile = Profile::FromWebUI(web_ui()); if (!profile) return; bool enabled; if (args->GetSize() == 1 && args->GetBoolean(0, &enabled)) { profile->GetPrefs()->SetBoolean( prefs::kDevToolsPortForwardingEnabled, enabled); } } void InspectMessageHandler::HandlePortForwardingConfigCommand( const ListValue* args) { Profile* profile = Profile::FromWebUI(web_ui()); if (!profile) return; const DictionaryValue* dict_src; if (args->GetSize() == 1 && args->GetDictionary(0, &dict_src)) profile->GetPrefs()->Set(prefs::kDevToolsPortForwardingConfig, *dict_src); } bool InspectMessageHandler::GetRemotePageId(const ListValue* args, std::string* page_id) { const DictionaryValue* data; if (args->GetSize() == 1 && args->GetDictionary(0, &data) && data->GetString(kAdbGlobalIdField, page_id)) { return true; } return false; } } // namespace class InspectUI::WorkerCreationDestructionListener : public WorkerServiceObserver, public content::BrowserChildProcessObserver, public base::RefCountedThreadSafe { public: WorkerCreationDestructionListener() : discovery_ui_(NULL) {} void Init(InspectUI* workers_ui) { DCHECK(workers_ui); DCHECK(!discovery_ui_); discovery_ui_ = workers_ui; BrowserChildProcessObserver::Add(this); BrowserThread::PostTask( BrowserThread::IO, FROM_HERE, base::Bind(&WorkerCreationDestructionListener::RegisterObserver, this)); } void InspectUIDestroyed() { DCHECK(discovery_ui_); discovery_ui_ = NULL; BrowserChildProcessObserver::Remove(this); BrowserThread::PostTask( BrowserThread::IO, FROM_HERE, base::Bind(&WorkerCreationDestructionListener::UnregisterObserver, this)); } void UpdateUI() { BrowserThread::PostTask( BrowserThread::IO, FROM_HERE, base::Bind(&WorkerCreationDestructionListener::CollectWorkersData, this)); } private: friend class base::RefCountedThreadSafe; virtual ~WorkerCreationDestructionListener() {} virtual void WorkerCreated( const GURL& url, const string16& name, int process_id, int route_id) OVERRIDE { CollectWorkersData(); } virtual void WorkerDestroyed(int process_id, int route_id) OVERRIDE { CollectWorkersData(); } virtual void BrowserChildProcessHostConnected( const content::ChildProcessData& data) OVERRIDE { if (data.process_type == content::PROCESS_TYPE_WORKER) UpdateUI(); } virtual void BrowserChildProcessHostDisconnected( const content::ChildProcessData& data) OVERRIDE { if (data.process_type == content::PROCESS_TYPE_WORKER) UpdateUI(); } void CollectWorkersData() { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); BrowserThread::PostTask( BrowserThread::UI, FROM_HERE, base::Bind(&WorkerCreationDestructionListener::PopulateWorkersList, this, WorkerService::GetInstance()->GetWorkers())); } void RegisterObserver() { WorkerService::GetInstance()->AddObserver(this); } void UnregisterObserver() { WorkerService::GetInstance()->RemoveObserver(this); } void PopulateWorkersList( const std::vector& worker_info) { DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); if (!discovery_ui_) return; ListValue target_list; for (size_t i = 0; i < worker_info.size(); ++i) { if (!worker_info[i].handle) continue; // Process is still being created. target_list.Append(BuildTargetDescriptor( kWorkerTargetType, HasClientHost(worker_info[i].process_id, worker_info[i].route_id), worker_info[i].url, UTF16ToUTF8(worker_info[i].name), GURL(), "", worker_info[i].process_id, worker_info[i].route_id, worker_info[i].handle)); } discovery_ui_->web_ui()->CallJavascriptFunction( "populateWorkersList", target_list); } InspectUI* discovery_ui_; }; InspectUI::InspectUI(content::WebUI* web_ui) : WebUIController(web_ui) { web_ui->AddMessageHandler(new InspectMessageHandler(this)); Profile* profile = Profile::FromWebUI(web_ui); content::WebUIDataSource::Add(profile, CreateInspectUIHTMLSource()); // Set up the chrome://theme/ source. ThemeSource* theme = new ThemeSource(profile); content::URLDataSource::Add(profile, theme); } InspectUI::~InspectUI() { StopListeningNotifications(); } void InspectUI::InitUI() { SetPortForwardingDefaults(); StartListeningNotifications(); PopulateLists(); UpdatePortForwardingEnabled(); UpdatePortForwardingConfig(); observer_->UpdateUI(); } void InspectUI::InspectRemotePage(const std::string& id) { RemotePages::iterator it = remote_pages_.find(id); if (it != remote_pages_.end()) { Profile* profile = Profile::FromWebUI(web_ui()); it->second->Inspect(profile); } } void InspectUI::ActivateRemotePage(const std::string& id) { RemotePages::iterator it = remote_pages_.find(id); if (it != remote_pages_.end()) it->second->Activate(); } void InspectUI::ReloadRemotePage(const std::string& id) { RemotePages::iterator it = remote_pages_.find(id); if (it != remote_pages_.end()) it->second->Reload(); } void InspectUI::CloseRemotePage(const std::string& id) { RemotePages::iterator it = remote_pages_.find(id); if (it != remote_pages_.end()) it->second->Close(); } void InspectUI::OpenRemotePage(const std::string& browser_id, const std::string& url) { GURL gurl(url); if (!gurl.is_valid()) { gurl = GURL("http://" + url); if (!gurl.is_valid()) return; } RemoteBrowsers::iterator it = remote_browsers_.find(browser_id); if (it != remote_browsers_.end()) it->second->Open(gurl.spec()); } void InspectUI::PopulateLists() { std::set tab_rvhs; for (TabContentsIterator it; !it.done(); it.Next()) tab_rvhs.insert(it->GetRenderViewHost()); scoped_ptr target_list(new ListValue()); std::vector rvh_vector = DevToolsAgentHost::GetValidRenderViewHosts(); std::map description_map; std::vector guest_contents; for (std::vector::iterator it(rvh_vector.begin()); it != rvh_vector.end(); it++) { bool is_tab = tab_rvhs.find(*it) != tab_rvhs.end(); RenderViewHost* rvh = (*it); WebContents* web_contents = WebContents::FromRenderViewHost(rvh); if (rvh->GetProcess()->IsGuest()) { if (web_contents) guest_contents.push_back(web_contents); } else { DictionaryValue* dictionary = BuildTargetDescriptor(rvh, is_tab); if (web_contents) description_map[web_contents] = dictionary; target_list->Append(dictionary); } } // Add the list of guest-views to each of its embedders. for (std::vector::iterator it(guest_contents.begin()); it != guest_contents.end(); ++it) { WebContents* guest = (*it); WebContents* embedder = guest->GetEmbedderWebContents(); if (embedder && description_map.count(embedder) > 0) { DictionaryValue* description = description_map[embedder]; ListValue* guests = NULL; if (!description->GetList(kGuestList, &guests)) { guests = new ListValue(); description->Set(kGuestList, guests); } RenderViewHost* rvh = guest->GetRenderViewHost(); if (rvh) guests->Append(BuildTargetDescriptor(rvh, false)); } } web_ui()->CallJavascriptFunction("populateLists", *target_list.get()); } void InspectUI::Observe(int type, const content::NotificationSource& source, const content::NotificationDetails& details) { if (source != content::Source(web_ui()->GetWebContents())) PopulateLists(); else if (type == content::NOTIFICATION_WEB_CONTENTS_DISCONNECTED) StopListeningNotifications(); } void InspectUI::StartListeningNotifications() { if (observer_) return; observer_ = new WorkerCreationDestructionListener(); observer_->Init(this); Profile* profile = Profile::FromWebUI(web_ui()); DevToolsAdbBridge* adb_bridge = DevToolsAdbBridge::Factory::GetForProfile(profile); if (adb_bridge) adb_bridge->AddListener(this); notification_registrar_.Add(this, content::NOTIFICATION_WEB_CONTENTS_CONNECTED, content::NotificationService::AllSources()); notification_registrar_.Add(this, content::NOTIFICATION_WEB_CONTENTS_DISCONNECTED, content::NotificationService::AllSources()); notification_registrar_.Add(this, content::NOTIFICATION_WEB_CONTENTS_DESTROYED, content::NotificationService::AllSources()); pref_change_registrar_.Init(profile->GetPrefs()); pref_change_registrar_.Add(prefs::kDevToolsPortForwardingEnabled, base::Bind(&InspectUI::UpdatePortForwardingEnabled, base::Unretained(this))); pref_change_registrar_.Add(prefs::kDevToolsPortForwardingConfig, base::Bind(&InspectUI::UpdatePortForwardingConfig, base::Unretained(this))); } void InspectUI::StopListeningNotifications() { if (!observer_.get()) return; Profile* profile = Profile::FromWebUI(web_ui()); DevToolsAdbBridge* adb_bridge = DevToolsAdbBridge::Factory::GetForProfile(profile); if (adb_bridge) adb_bridge->RemoveListener(this); observer_->InspectUIDestroyed(); observer_ = NULL; notification_registrar_.RemoveAll(); pref_change_registrar_.RemoveAll(); } content::WebUIDataSource* InspectUI::CreateInspectUIHTMLSource() { content::WebUIDataSource* source = content::WebUIDataSource::Create(chrome::kChromeUIInspectHost); source->AddResourcePath("inspect.css", IDR_INSPECT_CSS); source->AddResourcePath("inspect.js", IDR_INSPECT_JS); source->SetDefaultResource(IDR_INSPECT_HTML); return source; } void InspectUI::RemoteDevicesChanged( DevToolsAdbBridge::RemoteDevices* devices) { Profile* profile = Profile::FromWebUI(web_ui()); PortForwardingController* port_forwarding_controller = PortForwardingController::Factory::GetForProfile(profile); PortForwardingController::DevicesStatus port_forwarding_status; if (port_forwarding_controller) port_forwarding_status = port_forwarding_controller->UpdateDeviceList(*devices); remote_browsers_.clear(); remote_pages_.clear(); ListValue device_list; for (DevToolsAdbBridge::RemoteDevices::iterator dit = devices->begin(); dit != devices->end(); ++dit) { DevToolsAdbBridge::RemoteDevice* device = dit->get(); DictionaryValue* device_data = new DictionaryValue(); device_data->SetString(kAdbModelField, device->model()); device_data->SetString(kAdbSerialField, device->serial()); std::string device_id = base::StringPrintf( "device:%s", device->serial().c_str()); device_data->SetString(kAdbGlobalIdField, device_id); ListValue* browser_list = new ListValue(); device_data->Set(kAdbBrowsersField, browser_list); DevToolsAdbBridge::RemoteBrowsers& browsers = device->browsers(); for (DevToolsAdbBridge::RemoteBrowsers::iterator bit = browsers.begin(); bit != browsers.end(); ++bit) { DevToolsAdbBridge::RemoteBrowser* browser = bit->get(); DictionaryValue* browser_data = new DictionaryValue(); browser_data->SetString(kAdbBrowserProductField, browser->product()); browser_data->SetString(kAdbBrowserPackageField, browser->package()); browser_data->SetString(kAdbBrowserVersionField, browser->version()); std::string browser_id = base::StringPrintf( "browser:%s:%s:%s", device->serial().c_str(), browser->product().c_str(), // Force sorting by product name. browser->socket().c_str()); browser_data->SetString(kAdbGlobalIdField, browser_id); remote_browsers_[browser_id] = browser; ListValue* page_list = new ListValue(); browser_data->Set(kAdbPagesField, page_list); DevToolsAdbBridge::RemotePages& pages = browser->pages(); for (DevToolsAdbBridge::RemotePages::iterator it = pages.begin(); it != pages.end(); ++it) { DevToolsAdbBridge::RemotePage* page = it->get(); DictionaryValue* page_data = BuildTargetDescriptor( kAdbTargetType, page->attached(), GURL(page->url()), page->title(), GURL(page->favicon_url()), page->description(), 0, 0); std::string page_id = base::StringPrintf("page:%s:%s:%s", device->serial().c_str(), browser->socket().c_str(), page->id().c_str()); page_data->SetString(kAdbGlobalIdField, page_id); page_data->SetBoolean(kAdbAttachedForeignField, page->attached() && !page->HasDevToolsWindow()); // Pass the screen size in the page object to make sure that // the caching logic does not prevent the page item from updating // when the screen size changes. gfx::Size screen_size = device->GetScreenSize(); page_data->SetInteger(kAdbScreenWidthField, screen_size.width()); page_data->SetInteger(kAdbScreenHeightField, screen_size.height()); remote_pages_[page_id] = page; page_list->Append(page_data); } browser_list->Append(browser_data); } if (port_forwarding_controller) { PortForwardingController::DevicesStatus::iterator sit = port_forwarding_status.find(device->serial()); if (sit != port_forwarding_status.end()) { DictionaryValue* port_status_dict = new DictionaryValue(); typedef PortForwardingController::PortStatusMap StatusMap; const StatusMap& port_status = sit->second; for (StatusMap::const_iterator it = port_status.begin(); it != port_status.end(); ++it) { port_status_dict->SetInteger( base::StringPrintf("%d", it->first), it->second); } device_data->Set(kAdbPortStatus, port_status_dict); } } device_list.Append(device_data); } web_ui()->CallJavascriptFunction("populateDeviceLists", device_list); } void InspectUI::UpdatePortForwardingEnabled() { web_ui()->CallJavascriptFunction("updatePortForwardingEnabled", *GetPrefValue(prefs::kDevToolsPortForwardingEnabled)); } void InspectUI::UpdatePortForwardingConfig() { web_ui()->CallJavascriptFunction("updatePortForwardingConfig", *GetPrefValue(prefs::kDevToolsPortForwardingConfig)); } void InspectUI::SetPortForwardingDefaults() { Profile* profile = Profile::FromWebUI(web_ui()); PrefService* prefs = profile->GetPrefs(); bool default_set; if (!GetPrefValue(prefs::kDevToolsPortForwardingDefaultSet)-> GetAsBoolean(&default_set) || default_set) { return; } // This is the first chrome://inspect invocation on a fresh profile or after // upgrade from a version that did not have kDevToolsPortForwardingDefaultSet. prefs->SetBoolean(prefs::kDevToolsPortForwardingDefaultSet, true); bool enabled; const base::DictionaryValue* config; if (!GetPrefValue(prefs::kDevToolsPortForwardingEnabled)-> GetAsBoolean(&enabled) || !GetPrefValue(prefs::kDevToolsPortForwardingConfig)-> GetAsDictionary(&config)) { return; } // Do nothing if user already took explicit action. if (enabled || config->size() != 0) return; base::DictionaryValue default_config; default_config.SetString( kPortForwardingDefaultPort, kPortForwardingDefaultLocation); prefs->Set(prefs::kDevToolsPortForwardingConfig, default_config); } const base::Value* InspectUI::GetPrefValue(const char* name) { Profile* profile = Profile::FromWebUI(web_ui()); return profile->GetPrefs()->FindPreference(name)->GetValue(); }