blob: b5bbd1d939f7f8d2917dd3f962a0dbee10c5df9c [file] [log] [blame]
package org.eclipse.emf.edapt.declaration.generalization;
import java.util.ArrayList;
import java.util.List;
import org.eclipse.emf.ecore.EClass;
import org.eclipse.emf.ecore.EReference;
import org.eclipse.emf.edapt.declaration.EdaptConstraint;
import org.eclipse.emf.edapt.declaration.EdaptOperation;
import org.eclipse.emf.edapt.declaration.EdaptParameter;
import org.eclipse.emf.edapt.declaration.OperationImplementation;
import org.eclipse.emf.edapt.spi.migration.Instance;
import org.eclipse.emf.edapt.spi.migration.Metamodel;
import org.eclipse.emf.edapt.spi.migration.Model;
/**
* {@description}
*
* @author herrmama
* @author $Author$
* @version $Rev$
* @levd.rating YELLOW Hash: 06CD6AB4B386554B62F64EF5019B7203
*/
@EdaptOperation(identifier = "specializeReference", label = "Specialize Reference", description = "In the metamodel, either the type or the multiplicity of a reference is specialized. In the model, values no longer conforming to the new type or multiplicity are removed.")
public class SpecializeReference extends OperationImplementation {
/** {@description} */
@EdaptParameter(main = true, description = "The reference to be generalized")
public EReference reference;
/** {@description} */
@EdaptParameter(description = "The new type of the reference")
public EClass type;
/** {@description} */
@EdaptConstraint(restricts = "type", description = "The type must be the same or more special")
public boolean checkType(EClass type) {
return reference.getEReferenceType().isSuperTypeOf(type);
}
/** {@description} */
@EdaptParameter(description = "The new lower bound of the reference")
public int lowerBound;
/** {@description} */
@EdaptParameter(description = "The new upper bound of the reference")
public int upperBound;
/** {@description} */
@EdaptConstraint(description = "The multiplicity must be the same or more special")
public boolean checkReferenceMultiplicityRestricted() {
return lowerBound >= reference.getLowerBound()
&& (upperBound <= reference.getUpperBound() || reference
.getUpperBound() == -1);
}
/** {@inheritDoc} */
@Override
public void initialize(Metamodel metamodel) {
type = reference.getEReferenceType();
lowerBound = reference.getLowerBound();
upperBound = reference.getUpperBound();
}
/** {@inheritDoc} */
@Override
public void execute(Metamodel metamodel, Model model) {
// metamodel adaptation
if (reference.getEType() != type) {
reference.setEType(type);
if (reference.getEOpposite() != null) {
type.getEStructuralFeatures().add(reference.getEOpposite());
}
}
// model migration
for (final Instance instance : model.getAllInstances(reference
.getEContainingClass())) {
filterValueByType(instance, reference, type, model);
filterValueByMultiplicity(instance, reference, upperBound, model);
}
// metamodel adaptation
reference.setLowerBound(lowerBound);
reference.setUpperBound(upperBound);
}
/** Remove all values that do not conform to a certain type. */
private void filterValueByType(Instance instance, EReference reference,
EClass type, Model model) {
if (reference.isMany()) {
final List<Instance> values = new ArrayList<Instance>(instance
.getLinks(reference));
for (final Instance value : values) {
if (!value.instanceOf(type)) {
instance.remove(reference, value);
if (reference.isContainment()) {
model.delete(value);
}
}
}
} else {
final Instance value = instance.get(reference);
if (value != null) {
if (!value.instanceOf(type)) {
instance.unset(reference);
if (reference.isContainment()) {
model.delete(value);
}
}
}
}
}
/** Remove all values that do not conform to a certain upper bound. */
private void filterValueByMultiplicity(Instance instance,
EReference reference, int upperBound, Model model) {
if (reference.isMany()) {
final List<Instance> values = new ArrayList<Instance>(instance
.getLinks(reference));
if (upperBound == 1 && values.size() > 1) {
int i = 0;
for (final Instance value : values) {
if (i >= upperBound) {
instance.remove(reference, value);
if (reference.isContainment()) {
model.delete(value);
}
}
i++;
}
}
}
}
}