/*
* Copyright (c) 2008 Borland Software Corporation
*
* 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:
* Tatiana Fesenko (Borland) - initial API and implementation
*/
package org.eclipse.uml2.diagram.common.parser.valuespec;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import org.eclipse.core.runtime.IAdaptable;
import org.eclipse.emf.common.notify.Notification;
import org.eclipse.emf.ecore.EObject;
import org.eclipse.emf.ecore.EStructuralFeature;
import org.eclipse.gmf.runtime.common.core.command.ICommand;
import org.eclipse.gmf.runtime.common.core.command.UnexecutableCommand;
import org.eclipse.gmf.runtime.common.core.util.StringStatics;
import org.eclipse.gmf.runtime.common.ui.services.parser.IParserEditStatus;
import org.eclipse.gmf.runtime.common.ui.services.parser.ParserEditStatus;
import org.eclipse.gmf.runtime.emf.type.core.commands.SetValueCommand;
import org.eclipse.gmf.runtime.emf.type.core.requests.SetRequest;
import org.eclipse.gmf.runtime.emf.ui.services.parser.ISemanticParser;
import org.eclipse.jface.text.contentassist.IContentAssistProcessor;
import org.eclipse.uml2.uml.Constraint;
import org.eclipse.uml2.uml.Expression;
import org.eclipse.uml2.uml.LiteralInteger;
import org.eclipse.uml2.uml.LiteralString;
import org.eclipse.uml2.uml.OpaqueExpression;
import org.eclipse.uml2.uml.UMLPackage;
import org.eclipse.uml2.uml.ValueSpecification;
import org.eclipse.uml2.uml.util.UMLSwitch;
public class ValueSpecificationParser implements ISemanticParser {
private static final Set<EStructuralFeature> ourFeatures;
private static final String BODIES_SEPARATOR = System.getProperty("line.separator") + System.getProperty("line.separator"); //$NON-NLS-1$ //$NON-NLS-2$
public boolean areSemanticElementsAffected(EObject listener, Object notification) {
return isAffectingEvent(notification, 0);
}
public boolean isAffectingEvent(Object notification, int flags) {
if (notification instanceof Notification) {
Object feature = ((Notification) notification).getFeature();
return ourFeatures.contains(feature);
}
return false;
}
public List<?> getSemanticElementsBeingParsed(EObject element) {
if (element == null || false == element instanceof ValueSpecification) {
return Collections.emptyList();
}
return Collections.singletonList(element);
}
public IContentAssistProcessor getCompletionProcessor(IAdaptable element) {
return null;
}
public String getEditString(IAdaptable element, int flags) {
ValueSpecification specification = getValueSpecification(element);
if (specification == null) {
return StringStatics.BLANK;
}
UMLSwitch<String> valueSwitch = new UMLSwitch<String>() {
@Override
public String caseLiteralString(LiteralString object) {
return object.getValue();
}
@Override
public String caseLiteralInteger(LiteralInteger object) {
return Integer.toString(object.getValue());
}
@Override
public String caseExpression(Expression object) {
return object.getSymbol();
}
@Override
public String caseOpaqueExpression(OpaqueExpression object) {
StringBuilder result = new StringBuilder();
for (String nextBody : object.getBodies()) {
if (nextBody == null) {
continue;
}
if (result.length() > 0) {
result.append(BODIES_SEPARATOR);
}
result.append(nextBody);
}
return result.toString();
}
@Override
public String defaultCase(EObject object) {
return StringStatics.BLANK;
}
};
return valueSwitch.doSwitch(specification);
}
public ICommand getParseCommand(IAdaptable element, final String newString, int flags) {
final ValueSpecification vs = getValueSpecification(element);
if (vs == null) {
return UnexecutableCommand.INSTANCE;
}
if (newString == null) {
return UnexecutableCommand.INSTANCE;
}
UMLSwitch<ICommand> valueSwitch = new UMLSwitch<ICommand>() {
@Override
public ICommand caseLiteralString(LiteralString object) {
EStructuralFeature feature = UMLPackage.eINSTANCE.getLiteralString_Value();
return new SetValueCommand(new SetRequest(vs, feature, newString));
}
@Override
public ICommand caseLiteralInteger(LiteralInteger object) {
try {
Integer intValue = Integer.parseInt(newString);
EStructuralFeature feature = UMLPackage.eINSTANCE.getLiteralInteger_Value();
return new SetValueCommand(new SetRequest(vs, feature, intValue));
} catch (NumberFormatException e) {
return UnexecutableCommand.INSTANCE;
}
}
@Override
public ICommand caseExpression(Expression object) {
EStructuralFeature feature = UMLPackage.eINSTANCE.getExpression_Symbol();
return new SetValueCommand(new SetRequest(vs, feature, newString));
}
@Override
public ICommand caseOpaqueExpression(OpaqueExpression object) {
String[] bodies = newString.split("(\\r|\\n|\\u0085|\\u2028|\\u2029){2,}"); //$NON-NLS-1$
List<String> bodiesList = new ArrayList<String>(bodies.length);
for (String next : bodies) {
if (next != null && next.length() != 0) {
bodiesList.add(next);
}
}
return new SetValueCommand(new SetRequest(vs, UMLPackage.eINSTANCE.getOpaqueExpression_Body(), bodiesList));
}
@Override
public ICommand defaultCase(EObject object) {
return UnexecutableCommand.INSTANCE;
}
};
return valueSwitch.doSwitch(vs);
}
public String getPrintString(IAdaptable element, int flags) {
return getEditString(element, flags);
}
public IParserEditStatus isValidEditString(IAdaptable element, String editString) {
return ParserEditStatus.UNEDITABLE_STATUS;
}
protected ValueSpecification getValueSpecification(IAdaptable adaptable) {
return (ValueSpecification) adaptable.getAdapter(EObject.class);
}
static {
HashSet<EStructuralFeature> features = new HashSet<EStructuralFeature>();
features.add(UMLPackage.eINSTANCE.getLiteralInteger_Value());
features.add(UMLPackage.eINSTANCE.getLiteralString_Value());
features.add(UMLPackage.eINSTANCE.getExpression_Symbol());
features.add(UMLPackage.eINSTANCE.getOpaqueExpression_Body());
ourFeatures = Collections.unmodifiableSet(features);
}
public static class ConstraintParser extends ValueSpecificationParser {
public boolean isAffectingEvent(Object notification, int flags) {
if (notification instanceof Notification) {
Object feature = ((Notification) notification).getFeature();
return UMLPackage.eINSTANCE.getConstraint_Specification().equals(feature) || super.isAffectingEvent(notification, flags);
}
return false;
}
public List<?> getSemanticElementsBeingParsed(EObject element) {
if (false == element instanceof Constraint) {
return Collections.emptyList();
}
ValueSpecification spec = ((Constraint) element).getSpecification();
return spec == null ? Collections.emptyList() : Collections.singletonList(spec);
}
protected ValueSpecification getValueSpecification(IAdaptable adaptable) {
Constraint is = (Constraint) adaptable.getAdapter(EObject.class);
return is.getSpecification();
}
}
}