| // 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 <set> |
| |
| #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 kDiscoverUsbDevicesEnabledCommand[] = |
| "set-discover-usb-devices-enabled"; |
| 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 kAdbConnectedField[] = "adbConnected"; |
| const char kAdbModelField[] = "adbModel"; |
| const char kAdbSerialField[] = "adbSerial"; |
| 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<DevToolsAgentHost> 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<DevToolsAgentHost> 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 HandleBooleanPrefChanged(const char* pref_name, |
| 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(kDiscoverUsbDevicesEnabledCommand, |
| base::Bind(&InspectMessageHandler::HandleBooleanPrefChanged, |
| base::Unretained(this), |
| &prefs::kDevToolsDiscoverUsbDevicesEnabled[0])); |
| web_ui()->RegisterMessageCallback(kPortForwardingEnabledCommand, |
| base::Bind(&InspectMessageHandler::HandleBooleanPrefChanged, |
| base::Unretained(this), |
| &prefs::kDevToolsPortForwardingEnabled[0])); |
| 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<DevToolsAgentHost> 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::HandleBooleanPrefChanged( |
| const char* pref_name, |
| 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(pref_name, 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<WorkerCreationDestructionListener> { |
| 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<WorkerCreationDestructionListener>; |
| 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<WorkerService::WorkerInfo>& 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(); |
| UpdateDiscoverUsbDevicesEnabled(); |
| 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<RenderViewHost*> tab_rvhs; |
| for (TabContentsIterator it; !it.done(); it.Next()) |
| tab_rvhs.insert(it->GetRenderViewHost()); |
| |
| scoped_ptr<ListValue> target_list(new ListValue()); |
| |
| std::vector<RenderViewHost*> rvh_vector = |
| DevToolsAgentHost::GetValidRenderViewHosts(); |
| |
| std::map<WebContents*, DictionaryValue*> description_map; |
| std::vector<WebContents*> guest_contents; |
| |
| for (std::vector<RenderViewHost*>::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<WebContents*>::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<WebContents>(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::kDevToolsDiscoverUsbDevicesEnabled, |
| base::Bind(&InspectUI::UpdateDiscoverUsbDevicesEnabled, |
| base::Unretained(this))); |
| 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->GetModel()); |
| device_data->SetString(kAdbSerialField, device->GetSerial()); |
| device_data->SetBoolean(kAdbConnectedField, device->IsConnected()); |
| std::string device_id = base::StringPrintf( |
| "device:%s", |
| device->GetSerial().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->GetSerial().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->GetSerial().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->screen_size(); |
| 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->GetSerial()); |
| 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::UpdateDiscoverUsbDevicesEnabled() { |
| const Value* value = GetPrefValue(prefs::kDevToolsDiscoverUsbDevicesEnabled); |
| web_ui()->CallJavascriptFunction("updateDiscoverUsbDevicesEnabled", *value); |
| |
| // Configure adb bridge. |
| Profile* profile = Profile::FromWebUI(web_ui()); |
| DevToolsAdbBridge* adb_bridge = |
| DevToolsAdbBridge::Factory::GetForProfile(profile); |
| if (adb_bridge) { |
| bool enabled = false; |
| value->GetAsBoolean(&enabled); |
| adb_bridge->set_discover_usb_devices(enabled); |
| } |
| } |
| |
| 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(); |
| } |