/*******************************************************************************
* Copyright (c) 2009, 2016 Wind River Systems, Inc. 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:
* Wind River Systems - initial API and implementation
*******************************************************************************/
package org.eclipse.tcf.internal.debug.ui.commands;
import org.eclipse.jface.dialogs.IInputValidator;
import org.eclipse.jface.dialogs.InputDialog;
import org.eclipse.jface.window.Window;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Shell;
import org.eclipse.tcf.internal.debug.ui.Activator;
import org.eclipse.tcf.internal.debug.ui.ImageCache;
import org.eclipse.tcf.internal.debug.ui.model.ICastToType;
import org.eclipse.tcf.internal.debug.ui.model.TCFNode;
import org.eclipse.tcf.internal.debug.ui.model.TCFNodeExpression;
import org.eclipse.tcf.protocol.Protocol;
import org.eclipse.tcf.services.ISymbols;
import org.eclipse.tcf.util.TCFDataCache;
import org.eclipse.ui.IWorkbenchWindow;
public class CastToArrayCommand extends AbstractActionDelegate {
private static class CastToTypeInputValidator implements IInputValidator {
public CastToTypeInputValidator() {
}
public String isValid(String new_text) {
try {
if (new_text.length() == 0) return "";
int i = Integer.parseInt(new_text);
if (i < 1) return "Array length must be >= 1";
}
catch (Exception x) {
return "Invalid number";
}
return null;
}
}
private static class CastToTypeDialog extends InputDialog {
public CastToTypeDialog(Shell shell, String initial_value) {
super(shell, "Cast To Array", "Enter array length",
initial_value, new CastToTypeInputValidator() );
}
@Override
protected void configureShell(Shell shell) {
super.configureShell(shell);
shell.setImage(ImageCache.getImage(ImageCache.IMG_TCF));
}
}
private static class GetBaseTypeName implements Runnable {
final TCFNode node;
String base_type_name;
String cur_length;
GetBaseTypeName(TCFNode n) {
node = n instanceof ICastToType ? n : null;
if (node == null) done();
else Protocol.invokeLater(this);
}
public void run() {
cur_length = null;
if (node instanceof TCFNodeExpression && !((TCFNodeExpression)node).isEnabled()) {
done(null);
return;
}
String cast = node.getModel().getCastToType(node.getID());
if (cast != null) {
if (cast.endsWith("]")) {
int i = cast.lastIndexOf('[');
if (i > 0) {
cur_length = cast.substring(i + 1, cast.length() - 1);
done(cast.substring(0, i));
return;
}
}
done(null);
return;
}
TCFDataCache<ISymbols.Symbol> type_cache = ((ICastToType)node).getType();
if (!type_cache.validate(this)) return;
ISymbols.Symbol type_data = type_cache.getData();
if (type_data == null || type_data.getTypeClass() != ISymbols.TypeClass.pointer) {
done(null);
}
else {
for (int i = 0;; i++) {
TCFDataCache<ISymbols.Symbol> base_type_cache = node.getModel().getSymbolInfoCache(type_data.getBaseTypeID());
if (!base_type_cache.validate(this)) return;
ISymbols.Symbol base_type_data = base_type_cache.getData();
if (base_type_data == null) {
done(null);
return;
}
else {
if (base_type_data.getName() != null) {
done(makePtrTypeName(base_type_data.getName(), i));
return;
}
else if (base_type_data.getTypeClass() == ISymbols.TypeClass.pointer) {
type_data = base_type_data;
}
else if (!base_type_data.getID().equals(base_type_data.getTypeID())) {
// modified type without name, like "volatile int"
base_type_cache = node.getModel().getSymbolInfoCache(base_type_data.getTypeID());
if (!base_type_cache.validate(this)) return;
base_type_data = base_type_cache.getData();
if (base_type_data != null && base_type_data.getName() != null) {
done(makePtrTypeName(base_type_data.getName(), i));
}
else {
done(null);
}
return;
}
else {
done(null);
return;
}
}
}
}
}
private String makePtrTypeName(String base, int cnt) {
StringBuffer bf = new StringBuffer();
bf.append(base);
if (cnt > 0) {
bf.append(' ');
while (cnt > 0) {
bf.append('*');
cnt--;
}
}
return bf.toString();
}
private void done(String base_type_name) {
this.base_type_name = base_type_name;
Display.getDefault().asyncExec(new Runnable() {
public void run() {
done();
}
});
}
void done() {
}
}
@Override
protected void run() {
new GetBaseTypeName(getSelectedNode()) {
void done() {
if (base_type_name == null) return;
final IWorkbenchWindow window = Activator.getDefault().getWorkbench().getActiveWorkbenchWindow();
if (window == null) return;
CastToTypeDialog dialog = new CastToTypeDialog(window.getShell(), cur_length);
if (dialog.open() != Window.OK) return;
final String new_type = dialog.getValue().trim();
Protocol.invokeLater(new Runnable() {
public void run() {
node.getModel().setCastToType(node.getID(), base_type_name + "[" + new_type + "]");
}
});
}
};
}
@Override
protected void selectionChanged() {
new GetBaseTypeName(getSelectedNode()) {
void done() {
setEnabled(base_type_name != null);
}
};
}
}