blob: 69f3a6b6f024a1b8b574dbe994f6d4df29012dd8 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2006 IBM Corporation and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.debug.internal.ui.views.console;
import com.ibm.icu.text.MessageFormat;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IConfigurationElement;
import org.eclipse.core.runtime.IExtensionPoint;
import org.eclipse.core.runtime.Platform;
import org.eclipse.debug.core.DebugPlugin;
import org.eclipse.debug.core.ILaunch;
import org.eclipse.debug.core.ILaunchConfiguration;
import org.eclipse.debug.core.ILaunchListener;
import org.eclipse.debug.core.ILaunchManager;
import org.eclipse.debug.core.model.IProcess;
import org.eclipse.debug.internal.ui.DebugUIPlugin;
import org.eclipse.debug.ui.IDebugUIConstants;
import org.eclipse.debug.ui.console.ConsoleColorProvider;
import org.eclipse.debug.ui.console.IConsoleColorProvider;
import org.eclipse.debug.ui.console.IConsoleLineTracker;
import org.eclipse.jface.text.IDocument;
import org.eclipse.ui.console.ConsolePlugin;
import org.eclipse.ui.console.IConsole;
import org.eclipse.ui.console.IConsoleManager;
/**
* Creates documents for processes as they are registered with a launch.
* The singleton manager is accessible from the debug UI plugin.
*/
public class ProcessConsoleManager implements ILaunchListener {
/**
* Console document content provider extensions, keyed by extension id
*/
private Map fColorProviders;
/**
* The default color provider. Used if no color provider is contributed
* for the given process type.
*/
private IConsoleColorProvider fDefaultColorProvider;
/**
* Console line trackers; keyed by process type to list of trackers (1:N)
*/
private Map fLineTrackers;
/**
* Map of processes for a launch to compute removed processes
*/
private Map fProcesses;
/**
* @see ILaunchListener#launchRemoved(ILaunch)
*/
public void launchRemoved(ILaunch launch) {
removeLaunch(launch);
}
protected void removeLaunch(ILaunch launch) {
IProcess[] processes= launch.getProcesses();
for (int i= 0; i < processes.length; i++) {
IProcess iProcess = processes[i];
removeProcess(iProcess);
}
if (fProcesses != null) {
fProcesses.remove(launch);
}
}
/**
* Removes the console and document associated with the given process.
*
* @param iProcess process to clean up
*/
private void removeProcess(IProcess iProcess) {
IConsole console = getConsole(iProcess);
if (console != null) {
IConsoleManager manager = ConsolePlugin.getDefault().getConsoleManager();
manager.removeConsoles(new IConsole[]{console});
}
}
/**
* Returns the console for the given process, or <code>null</code> if none.
*
* @param process
* @return the console for the given process, or <code>null</code> if none
*/
public IConsole getConsole(IProcess process) {
IConsoleManager manager = ConsolePlugin.getDefault().getConsoleManager();
IConsole[] consoles = manager.getConsoles();
for (int i = 0; i < consoles.length; i++) {
IConsole console = consoles[i];
if (console instanceof ProcessConsole) {
ProcessConsole pc = (ProcessConsole)console;
if (pc.getProcess().equals(process)) {
return pc;
}
}
}
return null;
}
/**
* @see ILaunchListener#launchAdded(ILaunch)
*/
public void launchAdded(ILaunch launch) {
launchChanged(launch);
}
/**
* @see ILaunchListener#launchChanged(ILaunch)
*/
public void launchChanged(final ILaunch launch) {
IProcess[] processes= launch.getProcesses();
for (int i= 0; i < processes.length; i++) {
if (getConsoleDocument(processes[i]) == null) {
IProcess process = processes[i];
if (process.getStreamsProxy() == null) {
continue;
}
ILaunchConfiguration launchConfiguration = launch.getLaunchConfiguration();
//create a new console.
IConsoleColorProvider colorProvider = getColorProvider(process.getAttribute(IProcess.ATTR_PROCESS_TYPE));
String encoding = null;
try {
if (launchConfiguration != null) {
encoding = launchConfiguration.getAttribute(IDebugUIConstants.ATTR_CONSOLE_ENCODING, (String)null);
}
} catch (CoreException e) {
}
ProcessConsole pc = new ProcessConsole(process, colorProvider, encoding);
pc.setAttribute(IDebugUIConstants.ATTR_CONSOLE_PROCESS, process);
//add new console to console manager.
ConsolePlugin.getDefault().getConsoleManager().addConsoles(new IConsole[]{pc});
}
}
List removed = getRemovedProcesses(launch);
if (removed != null) {
Iterator iterator = removed.iterator();
while (iterator.hasNext()) {
IProcess p = (IProcess) iterator.next();
removeProcess(p);
}
}
}
/**
* Returns the document for the process, or <code>null</code>
* if none.
*/
public IDocument getConsoleDocument(IProcess process) {
ProcessConsole console = (ProcessConsole) getConsole(process);
return (console != null ? console.getDocument() : null);
}
/**
* Called by the debug ui plug-in on startup.
* The console document manager starts listening for
* launches to be registered and initializes if any launches
* already exist.
*/
public void startup() {
ILaunchManager launchManager= DebugPlugin.getDefault().getLaunchManager();
launchManager.addLaunchListener(this);
//set up the docs for launches already registered
ILaunch[] launches= launchManager.getLaunches();
for (int i = 0; i < launches.length; i++) {
launchAdded(launches[i]);
}
}
/**
* Called by the debug ui plug-in on shutdown.
* The console document manager de-registers as a
* launch listener and kills all existing console documents.
*/
public void shutdown() {
ILaunchManager launchManager= DebugPlugin.getDefault().getLaunchManager();
ILaunch[] launches = launchManager.getLaunches();
for (int i = 0; i < launches.length; i++) {
ILaunch launch = launches[i];
removeLaunch(launch);
}
launchManager.removeLaunchListener(this);
if (fProcesses != null) {
fProcesses.clear();
}
}
/**
* Returns a new console document color provider extension for the given
* process type, or <code>null</code> if none.
*
* @param type corresponds to <code>IProcess.ATTR_PROCESS_TYPE</code>
* @return IConsoleColorProvider
*/
public IConsoleColorProvider getColorProvider(String type) {
if (fColorProviders == null) {
fColorProviders = new HashMap();
IExtensionPoint extensionPoint= Platform.getExtensionRegistry().getExtensionPoint(DebugUIPlugin.getUniqueIdentifier(), IDebugUIConstants.EXTENSION_POINT_CONSOLE_COLOR_PROVIDERS);
IConfigurationElement[] elements = extensionPoint.getConfigurationElements();
for (int i = 0; i < elements.length; i++) {
IConfigurationElement extension = elements[i];
fColorProviders.put(extension.getAttributeAsIs("processType"), extension); //$NON-NLS-1$
}
}
IConfigurationElement extension = (IConfigurationElement)fColorProviders.get(type);
if (extension != null) {
try {
Object colorProvider = extension.createExecutableExtension("class"); //$NON-NLS-1$
if (colorProvider instanceof IConsoleColorProvider) {
return (IConsoleColorProvider)colorProvider;
}
DebugUIPlugin.logErrorMessage(MessageFormat.format(
"Extension {0} must specify an instanceof IConsoleColorProvider for class attribute.", //$NON-NLS-1$
new String[]{extension.getDeclaringExtension().getUniqueIdentifier()}));
} catch (CoreException e) {
DebugUIPlugin.log(e);
}
}
//no color provider found of specified type, return default color provider.
if (fDefaultColorProvider == null) {
fDefaultColorProvider = new ConsoleColorProvider();
}
return fDefaultColorProvider;
}
/**
* Returns the Line Trackers for a given process type.
* @param process The process for which line trackers are required.
* @return An array of line trackers which match the given process type.
*/
public IConsoleLineTracker[] getLineTrackers(IProcess process) {
String type = process.getAttribute(IProcess.ATTR_PROCESS_TYPE);
if (fLineTrackers == null) {
fLineTrackers = new HashMap();
IExtensionPoint extensionPoint= Platform.getExtensionRegistry().getExtensionPoint(DebugUIPlugin.getUniqueIdentifier(), IDebugUIConstants.EXTENSION_POINT_CONSOLE_LINE_TRACKERS);
IConfigurationElement[] elements = extensionPoint.getConfigurationElements();
for (int i = 0; i < elements.length; i++) {
IConfigurationElement extension = elements[i];
String processType = extension.getAttributeAsIs("processType"); //$NON-NLS-1$
List list = (List)fLineTrackers.get(processType);
if (list == null) {
list = new ArrayList();
fLineTrackers.put(processType, list);
}
list.add(extension);
}
}
ArrayList trackers = new ArrayList();
if (type != null) {
List lineTrackerExtensions = (List) fLineTrackers.get(type);
if(lineTrackerExtensions != null) {
for(Iterator i = lineTrackerExtensions.iterator(); i.hasNext(); ) {
IConfigurationElement element = (IConfigurationElement) i.next();
try {
trackers.add(element.createExecutableExtension("class")); //$NON-NLS-1$
} catch (CoreException e) {
DebugUIPlugin.log(e);
}
}
}
}
return (IConsoleLineTracker[]) trackers.toArray(new IConsoleLineTracker[0]);
}
/**
* Returns the processes that have been removed from the given
* launch, or <code>null</code> if none.
*
* @param launch launch that has changed
* @return removed processes or <code>null</code>
*/
private List getRemovedProcesses(ILaunch launch) {
List removed = null;
if (fProcesses == null) {
fProcesses = new HashMap();
}
IProcess[] old = (IProcess[]) fProcesses.get(launch);
IProcess[] curr = launch.getProcesses();
if (old != null) {
for (int i = 0; i < old.length; i++) {
IProcess process = old[i];
if (!contains(curr, process)) {
if (removed == null) {
removed = new ArrayList();
}
removed.add(process);
}
}
}
// update cache with current processes
fProcesses.put(launch, curr);
return removed;
}
/**
* Returns whether the given object is contained in the list.
*
* @param list list to search
* @param object object to search for
* @return whether the given object is contained in the list
*/
private boolean contains(Object[] list, Object object) {
for (int i = 0; i < list.length; i++) {
Object object2 = list[i];
if (object2.equals(object)) {
return true;
}
}
return false;
}
}