blob: 7e7ef8d4d0058adc1cc082b72793b5fbf07508e3 [file] [log] [blame]
/*****************************************************************************
* Copyright (c) 2013 CEA LIST.
*
*
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License 2.0
* which accompanies this distribution, and is available at
* https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* Ansgar Radermacher ansgar.radermacher@cea.fr
*
*****************************************************************************/
package org.eclipse.papyrus.designer.transformation.core;
import java.util.HashMap;
import java.util.Map;
import org.eclipse.papyrus.designer.languages.common.base.StringUtils;
import org.eclipse.papyrus.designer.languages.cpp.profile.C_Cpp.CppInit;
import org.eclipse.papyrus.designer.transformation.core.transformations.TransformationContext;
import org.eclipse.papyrus.uml.tools.utils.StereotypeUtil;
import org.eclipse.uml2.uml.Class;
import org.eclipse.uml2.uml.Classifier;
import org.eclipse.uml2.uml.Dependency;
import org.eclipse.uml2.uml.Element;
import org.eclipse.uml2.uml.Enumeration;
import org.eclipse.uml2.uml.EnumerationLiteral;
import org.eclipse.uml2.uml.NamedElement;
import org.eclipse.uml2.uml.Package;
import org.eclipse.uml2.uml.UMLPackage;
/**
* Manage enumerations within xtend templates
*/
public class EnumService {
private static final String GLOBALENUMS = "globalenums"; //$NON-NLS-1$
public static void init() {
enumHash.clear();
}
/**
* @return the qualified name of enum package which is used to prefix enumerations (namespace)
*/
public static String enumSvcPrefix() {
return enumPkg.getQualifiedName();
}
public static String quoteLiteral(Element dummy, String enumName, String literal) {
return StringUtils.quote(literal(enumName, literal));
}
/**
* Create a literal within an enumeration. Both, the literal and the enumeration may be an
* Xtend template. If the name of the enumeration starts with "L", it is considered as a
* local enumeration, i.e. a nested classifier within the classifier (it has to be a class)
* from the transformation context.
*
* @param enumName
* the name of an enumeration
* @param literal
* the name of a literal within that enumeration.
* @return the created literal
*/
public static String literal(String enumName, String literal) {
return literal(enumName, literal, -1);
}
/**
* Create a literal within an enumeration. Both, the literal and the enumeration may be an
* Xtend template. If the name of the enumeration starts with "L", it is considered as a
* local enumeration, i.e. a nested classifier within the classifier (it has to be a class)
* from the transformation context.
*
* @param enumName
* the name of an enumeration
* @param literal
* the name of a literal within that enumeration.
* @return the created literal
*/
public static String literal(String enumName, String literal, int initialValue) {
Enumeration enumeration;
boolean first = false;
if (enumName.startsWith("L")) { //$NON-NLS-1$
// magic prefix for class local (only allowed for local classes)
if (TransformationContext.current.classifier instanceof Class) {
Class clazz = (Class) TransformationContext.current.classifier;
enumeration = (Enumeration) clazz.getNestedClassifier(enumName);
if (enumeration == null) {
enumeration = (Enumeration) clazz.createNestedClassifier(enumName, UMLPackage.eINSTANCE.getEnumeration());
}
}
else {
throw new RuntimeException(String.format(Messages.EnumService_LOCAL_ENUM_UNUSED, enumName));
}
}
else {
enumeration = enumHash.get(enumName);
if (enumPkg == null) {
// enumeration can not be created
throw new RuntimeException(String.format(Messages.EnumService_GLOBAL_ENUM_NOT_INITIALIZED, enumName));
}
if (enumeration == null) {
enumeration = enumPkg.createOwnedEnumeration(enumName);
enumHash.put(enumName, enumeration);
first = true;
}
}
if (enumeration.getOwnedLiteral(literal) == null) {
EnumerationLiteral umlLiteral = enumeration.createOwnedLiteral(literal);
if (first && initialValue != -1) {
CppInit cppInit = StereotypeUtil.applyApp(umlLiteral, CppInit.class);
cppInit.setValue(initialValue);
}
}
// declare a dependency to the enumeration from the current classifier
checkAndCreateDependency(TransformationContext.current.classifier, enumeration);
if (enumName.startsWith("L")) { //$NON-NLS-1$
return literal;
}
else {
return GLOBALENUMS + "::" + literal; //$NON-NLS-1$
}
}
/**
* Create a dependency between the passed classifier, target pair. The objective
* of this function is that code generators do the necessary to assure that the
* target is known within the classifier (e.g. include directives)
*
* @param classifier
* a classifier
* @param target
* a target, on which the classifier or its code depends.
*/
public static void checkAndCreateDependency(Classifier classifier, NamedElement target) {
boolean found = false;
for (Dependency dep : classifier.getClientDependencies()) {
if (dep.getSuppliers().contains(target)) {
found = true;
}
}
if (!found) {
Dependency dep = classifier.createDependency(target);
dep.setName(String.format("from %s to %s", classifier.getName(), target.getName())); //$NON-NLS-1$
}
}
public static void createEnumPackage(Package root) {
init();
enumPkg = root.createNestedPackage(GLOBALENUMS);
}
public static Package enumPkg;
public static void createEnums() {
}
private static Map<String, Enumeration> enumHash = new HashMap<String, Enumeration>();
}