blob: 6dae1d6b1efa429db0c97d71233ed504de8dc6e8 [file] [log] [blame]
/*
* Copyright 2000-2010 JetBrains s.r.o.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.tools.idea.monitor;
import com.android.ddmlib.AndroidDebugBridge;
import com.android.ddmlib.ClientData;
import com.android.ddmlib.Log;
import com.android.tools.idea.ddms.*;
import com.android.tools.idea.ddms.actions.*;
import com.android.tools.idea.ddms.adb.AdbService;
import com.android.tools.idea.monitor.cpu.CpuMonitorView;
import com.android.tools.idea.monitor.gpu.GpuMonitorView;
import com.android.tools.idea.monitor.memory.MemoryMonitorView;
import com.android.tools.idea.monitor.network.NetworkMonitorView;
import com.google.common.util.concurrent.FutureCallback;
import com.google.common.util.concurrent.Futures;
import com.google.common.util.concurrent.ListenableFuture;
import com.intellij.ProjectTopics;
import com.intellij.execution.ExecutionManager;
import com.intellij.execution.filters.HyperlinkInfo;
import com.intellij.execution.impl.ConsoleViewImpl;
import com.intellij.execution.ui.ConsoleView;
import com.intellij.execution.ui.ConsoleViewContentType;
import com.intellij.execution.ui.RunnerLayoutUi;
import com.intellij.execution.ui.layout.PlaceInGrid;
import com.intellij.facet.ProjectFacetManager;
import com.intellij.openapi.actionSystem.ActionGroup;
import com.intellij.openapi.actionSystem.ActionPlaces;
import com.intellij.openapi.actionSystem.DefaultActionGroup;
import com.intellij.openapi.actionSystem.Separator;
import com.intellij.openapi.application.ApplicationManager;
import com.intellij.openapi.diagnostic.Logger;
import com.intellij.openapi.module.Module;
import com.intellij.openapi.project.DumbAware;
import com.intellij.openapi.project.Project;
import com.intellij.openapi.roots.ModuleRootAdapter;
import com.intellij.openapi.roots.ModuleRootEvent;
import com.intellij.openapi.ui.Messages;
import com.intellij.openapi.util.Comparing;
import com.intellij.openapi.util.Key;
import com.intellij.openapi.wm.ToolWindow;
import com.intellij.openapi.wm.ToolWindowFactory;
import com.intellij.openapi.wm.ToolWindowManager;
import com.intellij.openapi.wm.ex.ToolWindowManagerAdapter;
import com.intellij.openapi.wm.ex.ToolWindowManagerEx;
import com.intellij.ui.IdeBorderFactory;
import com.intellij.ui.SideBorder;
import com.intellij.ui.components.JBLoadingPanel;
import com.intellij.ui.content.Content;
import com.intellij.ui.content.ContentManager;
import com.intellij.util.messages.MessageBusConnection;
import icons.AndroidIcons;
import org.jetbrains.android.facet.AndroidFacet;
import org.jetbrains.android.logcat.AdbErrors;
import org.jetbrains.android.logcat.AndroidLogcatConstants;
import org.jetbrains.android.logcat.AndroidLogcatView;
import org.jetbrains.android.maven.AndroidMavenUtil;
import org.jetbrains.android.run.AndroidDebugRunner;
import org.jetbrains.android.sdk.AndroidPlatform;
import org.jetbrains.android.sdk.AndroidSdkUtils;
import org.jetbrains.android.util.AndroidBundle;
import org.jetbrains.annotations.NonNls;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import javax.swing.*;
import java.awt.*;
import java.io.File;
import java.util.List;
/**
* @author Eugene.Kudelevsky
*/
public class AndroidToolWindowFactory implements ToolWindowFactory, DumbAware {
public static final String TOOL_WINDOW_ID = AndroidBundle.message("android.logcat.title");
@NonNls private static final String ADBLOGS_CONTENT_ID = "AdbLogsContent";
public static final Key<DevicePanel> DEVICES_PANEL_KEY = Key.create("DevicePanel");
@Override
public void createToolWindowContent(@NotNull final Project project, @NotNull final ToolWindow toolWindow) {
// In order to use the runner layout ui, the runner infrastructure needs to be initialized.
// Otherwise it is not possible to for example drag one of the tabs out of the tool window.
// The object that needs to be created is the content manager of the execution manager for this project.
ExecutionManager.getInstance(project).getContentManager();
RunnerLayoutUi layoutUi = RunnerLayoutUi.Factory.getInstance(project).create("Android", "Android", "Android", project);
toolWindow.setIcon(AndroidIcons.AndroidToolWindow);
toolWindow.setAvailable(true, null);
toolWindow.setToHideOnEmptyContent(true);
toolWindow.setTitle(TOOL_WINDOW_ID);
DeviceContext deviceContext = new DeviceContext();
// TODO Remove global handlers. These handlers are global, but are set per project
// if there are two projects opened, things go very wrong.
ClientData.setMethodProfilingHandler(new OpenVmTraceHandler(project));
ClientData.setAllocationTrackingHandler(new ShowAllocationsHandler(project));
Content logcatContent = createLogcatContent(layoutUi, project, deviceContext);
final AndroidLogcatView logcatView = logcatContent.getUserData(AndroidLogcatView.ANDROID_LOGCAT_VIEW_KEY);
assert logcatView != null;
logcatContent.setSearchComponent(logcatView.createSearchComponent());
layoutUi.addContent(logcatContent, 0, PlaceInGrid.center, false);
Content adbLogsContent = createAdbLogsContent(layoutUi, project);
layoutUi.addContent(adbLogsContent, 1, PlaceInGrid.center, false);
Content memoryContent = createMemoryContent(layoutUi, project, deviceContext);
layoutUi.addContent(memoryContent, 2, PlaceInGrid.center, false);
Content cpuContent = createCpuContent(layoutUi, project, deviceContext);
layoutUi.addContent(cpuContent, 3, PlaceInGrid.center, false);
Content gpuContent = createGpuContent(layoutUi, project, deviceContext);
layoutUi.addContent(gpuContent, 3, PlaceInGrid.center, false);
Content networkContent = createNetworkContent(layoutUi, project, deviceContext);
layoutUi.addContent(networkContent, 4, PlaceInGrid.center, false);
layoutUi.getOptions().setLeftToolbar(getToolbarActions(project, deviceContext), ActionPlaces.UNKNOWN);
final JBLoadingPanel loadingPanel = new JBLoadingPanel(new BorderLayout(), project);
DevicePanel devicePanel = new DevicePanel(project, deviceContext);
JPanel panel = devicePanel.getComponent();
panel.setBorder(IdeBorderFactory.createBorder(SideBorder.BOTTOM));
loadingPanel.add(panel, BorderLayout.NORTH);
loadingPanel.add(layoutUi.getComponent(), BorderLayout.CENTER);
final ContentManager contentManager = toolWindow.getContentManager();
Content c = contentManager.getFactory().createContent(loadingPanel, "", true);
// Store references to the logcat & device panel views, so that these views can be retrieved directly from
// the DDMS tool window. (e.g. to clear logcat before a launch, select a particular device, etc)
c.putUserData(AndroidLogcatView.ANDROID_LOGCAT_VIEW_KEY, logcatView);
c.putUserData(DEVICES_PANEL_KEY, devicePanel);
contentManager.addContent(c);
ApplicationManager.getApplication().invokeLater(new Runnable() {
@Override
public void run() {
logcatView.activate();
final ToolWindow window = ToolWindowManager.getInstance(project).getToolWindow(TOOL_WINDOW_ID);
if (window != null && window.isVisible()) {
ConsoleView console = logcatView.getLogConsole().getConsole();
if (console != null) {
checkFacetAndSdk(project, console);
}
}
}
}, project.getDisposed());
final File adb = AndroidSdkUtils.getAdb(project);
if (adb == null) {
return;
}
loadingPanel.setLoadingText("Initializing ADB");
loadingPanel.startLoading();
ListenableFuture<AndroidDebugBridge> future = AdbService.getInstance().getDebugBridge(adb);
Futures.addCallback(future, new FutureCallback<AndroidDebugBridge>() {
@Override
public void onSuccess(@Nullable AndroidDebugBridge bridge) {
Logger.getInstance(AndroidToolWindowFactory.class).info("Successfully obtained debug bridge");
loadingPanel.stopLoading();
}
@Override
public void onFailure(Throwable t) {
loadingPanel.stopLoading();
// If we cannot connect to ADB in a reasonable amount of time (10 seconds timeout in AdbService), then something is seriously
// wrong. The only identified reason so far is that some machines have incompatible versions of adb that were already running.
// e.g. Genymotion, some HTC flashing software, Ubuntu's adb package may all conflict with the version of adb in the SDK.
Logger.getInstance(AndroidToolWindowFactory.class).info("Unable to obtain debug bridge", t);
String msg = String.format("Unable to establish a connection to adb.\n\n" +
"This usually happens if you have an incompatible version of adb running already.\n" +
"Try re-opening Studio after killing any existing adb daemons.\n\n" +
"If this happens repeatedly, please file a bug at http://b.android.com including the following:\n" +
" 1. Output of the command: '%1$s devices'\n" +
" 2. Your idea.log file (Help | Show Log in Explorer)\n", adb.getAbsolutePath());
Messages.showErrorDialog(msg, "ADB Connection Error");
}
}, EdtExecutor.INSTANCE);
}
private static Content createMemoryContent(@NotNull RunnerLayoutUi layoutUi,
@NotNull Project project,
@NotNull DeviceContext deviceContext) {
MemoryMonitorView view = new MemoryMonitorView(project, deviceContext);
Content content = layoutUi.createContent("Memory", view.createComponent(), "Memory", AndroidIcons.MemoryMonitor, null);
content.setCloseable(false);
return content;
}
private static Content createCpuContent(@NotNull RunnerLayoutUi layoutUi,
@NotNull Project project,
@NotNull DeviceContext deviceContext) {
CpuMonitorView view = new CpuMonitorView(project, deviceContext);
Content content = layoutUi.createContent("CPU", view.createComponent(), "CPU", AndroidIcons.CpuMonitor, null);
content.setCloseable(false);
return content;
}
private static Content createGpuContent(@NotNull RunnerLayoutUi layoutUi,
@NotNull Project project,
@NotNull DeviceContext deviceContext) {
GpuMonitorView view = new GpuMonitorView(project, deviceContext);
Content content = layoutUi.createContent("GPU", view.createComponent(), "GPU", AndroidIcons.GpuMonitor, null);
content.setCloseable(false);
return content;
}
private static Content createNetworkContent(@NotNull RunnerLayoutUi layoutUi,
@NotNull Project project,
@NotNull DeviceContext deviceContext) {
NetworkMonitorView view = new NetworkMonitorView(project, deviceContext);
Content content = layoutUi.createContent("Network", view.createComponent(), "Network", AndroidIcons.NetworkMonitor, null);
content.setCloseable(false);
return content;
}
@NotNull
public ActionGroup getToolbarActions(Project project, DeviceContext deviceContext) {
DefaultActionGroup group = new DefaultActionGroup();
group.add(new ScreenshotAction(project, deviceContext));
group.add(new ScreenRecorderAction(project, deviceContext));
group.add(DumpSysActions.create(project, deviceContext));
//group.add(new MyFileExplorerAction());
group.add(new Separator());
group.add(new TerminateVMAction(deviceContext));
//group.add(new MyAllocationTrackerAction());
//group.add(new Separator());
return group;
}
private static Content createLogcatContent(RunnerLayoutUi layoutUi, final Project project, DeviceContext deviceContext) {
final AndroidLogcatView logcatView = new AndroidLogcatView(project, deviceContext) {
@Override
protected boolean isActive() {
ToolWindow window = ToolWindowManager.getInstance(project).getToolWindow(TOOL_WINDOW_ID);
return window.isVisible();
}
};
ToolWindowManagerEx.getInstanceEx(project).addToolWindowManagerListener(new ToolWindowManagerAdapter() {
boolean myToolWindowVisible;
@Override
public void stateChanged() {
ToolWindow window = ToolWindowManager.getInstance(project).getToolWindow(TOOL_WINDOW_ID);
if (window != null) {
boolean visible = window.isVisible();
if (visible != myToolWindowVisible) {
myToolWindowVisible = visible;
logcatView.activate();
if (visible) {
ConsoleView console = logcatView.getLogConsole().getConsole();
if (console != null) {
checkFacetAndSdk(project, console);
}
}
}
}
}
});
final MessageBusConnection connection = project.getMessageBus().connect(project);
connection.subscribe(ProjectTopics.PROJECT_ROOTS, new MyAndroidPlatformListener(logcatView));
JPanel logcatContentPanel = logcatView.getContentPanel();
final Content logcatContent =
layoutUi.createContent(AndroidDebugRunner.ANDROID_LOGCAT_CONTENT_ID, logcatContentPanel, "logcat", AndroidIcons.Ddms.Logcat, null);
logcatContent.putUserData(AndroidLogcatView.ANDROID_LOGCAT_VIEW_KEY, logcatView);
logcatContent.setDisposer(logcatView);
logcatContent.setCloseable(false);
logcatContent.setPreferredFocusableComponent(logcatContentPanel);
return logcatContent;
}
private Content createAdbLogsContent(RunnerLayoutUi layoutUi, Project project) {
final ConsoleView console = new ConsoleViewImpl(project, false);
Content adbLogsContent =
layoutUi.createContent(ADBLOGS_CONTENT_ID, console.getComponent(), AndroidBundle.message("android.adb.logs.tab.title"), null, null);
adbLogsContent.setCloseable(false);
//noinspection UnnecessaryFullyQualifiedName
com.android.ddmlib.Log.setLogOutput(new Log.ILogOutput() {
@Override
public void printLog(Log.LogLevel logLevel, String tag, String message) {
reportAdbLogMessage(logLevel, tag, message, console);
}
@Override
public void printAndPromptLog(Log.LogLevel logLevel, String tag, String message) {
// todo: should we show dialog?
reportAdbLogMessage(logLevel, tag, message, console);
}
});
return adbLogsContent;
}
private static void reportAdbLogMessage(Log.LogLevel logLevel, String tag, String message, @NotNull ConsoleView consoleView) {
if (message == null) {
return;
}
if (logLevel == null) {
logLevel = Log.LogLevel.INFO;
}
if (logLevel == Log.LogLevel.ERROR || logLevel == Log.LogLevel.ASSERT) {
AdbErrors.reportError(message, tag);
}
final ConsoleViewContentType contentType = toConsoleViewContentType(logLevel);
if (contentType == null) {
return;
}
final String fullMessage = tag != null ? tag + ": " + message : message;
consoleView.print(fullMessage + '\n', contentType);
}
@Nullable
private static ConsoleViewContentType toConsoleViewContentType(@NotNull Log.LogLevel logLevel) {
switch (logLevel) {
case VERBOSE:
return null;
case DEBUG:
return null;
case INFO:
return ConsoleViewContentType.getConsoleViewType(AndroidLogcatConstants.INFO);
case WARN:
return ConsoleViewContentType.getConsoleViewType(AndroidLogcatConstants.WARNING);
case ERROR:
return ConsoleViewContentType.getConsoleViewType(AndroidLogcatConstants.ERROR);
case ASSERT:
return ConsoleViewContentType.getConsoleViewType(AndroidLogcatConstants.ASSERT);
default:
assert false : "Unknown log level " + logLevel;
}
return null;
}
private static void checkFacetAndSdk(Project project, @NotNull final ConsoleView console) {
final List<AndroidFacet> facets = ProjectFacetManager.getInstance(project).getFacets(AndroidFacet.ID);
if (facets.size() == 0) {
console.clear();
console.print(AndroidBundle.message("android.logcat.no.android.facets.error"), ConsoleViewContentType.ERROR_OUTPUT);
return;
}
final AndroidFacet facet = facets.get(0);
AndroidPlatform platform = facet.getConfiguration().getAndroidPlatform();
if (platform == null) {
console.clear();
final Module module = facet.getModule();
if (!AndroidMavenUtil.isMavenizedModule(module)) {
console.print("Please ", ConsoleViewContentType.ERROR_OUTPUT);
console.printHyperlink("configure", new HyperlinkInfo() {
@Override
public void navigate(Project project) {
AndroidSdkUtils.openModuleDependenciesConfigurable(module);
}
});
console.print(" Android SDK\n", ConsoleViewContentType.ERROR_OUTPUT);
}
else {
console.print(AndroidBundle.message("android.maven.cannot.parse.android.sdk.error", module.getName()) + '\n',
ConsoleViewContentType.ERROR_OUTPUT);
}
}
}
private static class MyAndroidPlatformListener extends ModuleRootAdapter {
private final Project myProject;
private final AndroidLogcatView myView;
private AndroidPlatform myPrevPlatform;
private MyAndroidPlatformListener(@NotNull AndroidLogcatView view) {
myProject = view.getProject();
myView = view;
myPrevPlatform = getPlatform();
}
@Override
public void rootsChanged(ModuleRootEvent event) {
final ToolWindow window = ToolWindowManager.getInstance(myProject).getToolWindow(TOOL_WINDOW_ID);
if (window == null) {
return;
}
if (window.isDisposed() || !window.isVisible()) {
return;
}
AndroidPlatform newPlatform = getPlatform();
if (!Comparing.equal(myPrevPlatform, newPlatform)) {
myPrevPlatform = newPlatform;
ApplicationManager.getApplication().invokeLater(new Runnable() {
@Override
public void run() {
if (!window.isDisposed() && window.isVisible()) {
myView.activate();
}
}
});
}
}
@Nullable
private AndroidPlatform getPlatform() {
AndroidPlatform newPlatform = null;
final List<AndroidFacet> facets = ProjectFacetManager.getInstance(myProject).getFacets(AndroidFacet.ID);
if (facets.size() > 0) {
final AndroidFacet facet = facets.get(0);
newPlatform = facet.getConfiguration().getAndroidPlatform();
}
return newPlatform;
}
}
}