| /* |
| * 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; |
| } |
| } |
| } |