/*
 * Copyright (c) 2005, 2018 IBM Corporation, CEA, and others.
 * All rights reserved.   This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v2.0
 * which accompanies this distribution, and is available at
 * http://www.eclipse.org/legal/epl-v20.html
 *
 * Contributors:
 *   IBM - initial API and implementation
 *   Kenn Hussey (CEA) - 327039, 418466, 451350, 485756
 *   Kenn Hussey - 535301
 *
 */
package org.eclipse.uml2.uml.cdo.internal.impl;

import java.lang.reflect.InvocationTargetException;
import java.util.Collection;
import java.util.Map;
import org.eclipse.emf.common.notify.Notification;

import org.eclipse.emf.common.util.DiagnosticChain;
import org.eclipse.emf.common.util.EList;

import org.eclipse.emf.ecore.EAnnotation;
import org.eclipse.emf.ecore.EClass;
import org.eclipse.emf.ecore.InternalEObject;

import org.eclipse.emf.ecore.impl.ENotificationImpl;

import org.eclipse.emf.ecore.util.EObjectResolvingEList;

import org.eclipse.uml2.uml.Comment;
import org.eclipse.uml2.uml.Event;
import org.eclipse.uml2.uml.NamedElement;
import org.eclipse.uml2.uml.Namespace;
import org.eclipse.uml2.uml.Port;
import org.eclipse.uml2.uml.Stereotype;
import org.eclipse.uml2.uml.StringExpression;
import org.eclipse.uml2.uml.Trigger;
import org.eclipse.uml2.uml.Type;
import org.eclipse.uml2.uml.UMLPackage;
import org.eclipse.uml2.uml.VisibilityKind;
import org.eclipse.uml2.uml.internal.operations.TriggerOperations;

/**
 * <!-- begin-user-doc -->
 * An implementation of the model object '<em><b>Trigger</b></em>'.
 * <!-- end-user-doc -->
 * <p>
 * The following features are implemented:
 * </p>
 * <ul>
 *   <li>{@link org.eclipse.uml2.uml.cdo.internal.impl.TriggerImpl#getEvent <em>Event</em>}</li>
 *   <li>{@link org.eclipse.uml2.uml.cdo.internal.impl.TriggerImpl#getPorts <em>Port</em>}</li>
 * </ul>
 *
 * @generated
 */
public class TriggerImpl
		extends NamedElementImpl
		implements Trigger {

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	protected TriggerImpl() {
		super();
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	@Override
	protected EClass eStaticClass() {
		return UMLPackage.Literals.TRIGGER;
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	public Event getEvent() {
		return (Event) eDynamicGet(UMLPackage.TRIGGER__EVENT - ESTATIC_FEATURE_COUNT, UMLPackage.Literals.TRIGGER__EVENT, true, true);
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	public Event basicGetEvent() {
		return (Event) eDynamicGet(UMLPackage.TRIGGER__EVENT - ESTATIC_FEATURE_COUNT, UMLPackage.Literals.TRIGGER__EVENT, false, true);
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	public void setEvent(Event newEvent) {
		eDynamicSet(UMLPackage.TRIGGER__EVENT - ESTATIC_FEATURE_COUNT, UMLPackage.Literals.TRIGGER__EVENT, newEvent);
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	@SuppressWarnings("unchecked")
	public EList<Port> getPorts() {
		return (EList<Port>) eDynamicGet(UMLPackage.TRIGGER__PORT - ESTATIC_FEATURE_COUNT, UMLPackage.Literals.TRIGGER__PORT, true, true);
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	public Port getPort(String name, Type type) {
		return getPort(name, type, false);
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	public Port getPort(String name, Type type, boolean ignoreCase) {
		portLoop: for (Port port : getPorts()) {
			if (name != null && !(ignoreCase ? name.equalsIgnoreCase(port.getName()) : name.equals(port.getName())))
				continue portLoop;
			if (type != null && !type.equals(port.getType()))
				continue portLoop;
			return port;
		}
		return null;
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	public boolean validateTriggerWithPorts(DiagnosticChain diagnostics,
			Map<Object, Object> context) {
		return TriggerOperations.validateTriggerWithPorts(this, diagnostics, context);
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	@Override
	public Object eGet(int featureID, boolean resolve, boolean coreType) {
		switch (featureID) {
		case UMLPackage.TRIGGER__EANNOTATIONS:
			return getEAnnotations();
		case UMLPackage.TRIGGER__OWNED_COMMENT:
			return getOwnedComments();
		case UMLPackage.TRIGGER__OWNED_ELEMENT:
			return getOwnedElements();
		case UMLPackage.TRIGGER__OWNER:
			if (resolve)
				return getOwner();
			return basicGetOwner();
		case UMLPackage.TRIGGER__CLIENT_DEPENDENCY:
			return getClientDependencies();
		case UMLPackage.TRIGGER__NAME:
			return getName();
		case UMLPackage.TRIGGER__NAME_EXPRESSION:
			if (resolve)
				return getNameExpression();
			return basicGetNameExpression();
		case UMLPackage.TRIGGER__NAMESPACE:
			if (resolve)
				return getNamespace();
			return basicGetNamespace();
		case UMLPackage.TRIGGER__QUALIFIED_NAME:
			return getQualifiedName();
		case UMLPackage.TRIGGER__VISIBILITY:
			return getVisibility();
		case UMLPackage.TRIGGER__EVENT:
			if (resolve)
				return getEvent();
			return basicGetEvent();
		case UMLPackage.TRIGGER__PORT:
			return getPorts();
		}
		return eDynamicGet(featureID, resolve, coreType);
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	@SuppressWarnings("unchecked")
	@Override
	public void eSet(int featureID, Object newValue) {
		switch (featureID) {
		case UMLPackage.TRIGGER__EANNOTATIONS:
			getEAnnotations().clear();
			getEAnnotations().addAll((Collection<? extends EAnnotation>) newValue);
			return;
		case UMLPackage.TRIGGER__OWNED_COMMENT:
			getOwnedComments().clear();
			getOwnedComments().addAll((Collection<? extends Comment>) newValue);
			return;
		case UMLPackage.TRIGGER__NAME:
			setName((String) newValue);
			return;
		case UMLPackage.TRIGGER__NAME_EXPRESSION:
			setNameExpression((StringExpression) newValue);
			return;
		case UMLPackage.TRIGGER__VISIBILITY:
			setVisibility((VisibilityKind) newValue);
			return;
		case UMLPackage.TRIGGER__EVENT:
			setEvent((Event) newValue);
			return;
		case UMLPackage.TRIGGER__PORT:
			getPorts().clear();
			getPorts().addAll((Collection<? extends Port>) newValue);
			return;
		}
		eDynamicSet(featureID, newValue);
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	@Override
	public void eUnset(int featureID) {
		switch (featureID) {
		case UMLPackage.TRIGGER__EANNOTATIONS:
			getEAnnotations().clear();
			return;
		case UMLPackage.TRIGGER__OWNED_COMMENT:
			getOwnedComments().clear();
			return;
		case UMLPackage.TRIGGER__NAME:
			unsetName();
			return;
		case UMLPackage.TRIGGER__NAME_EXPRESSION:
			setNameExpression((StringExpression) null);
			return;
		case UMLPackage.TRIGGER__VISIBILITY:
			unsetVisibility();
			return;
		case UMLPackage.TRIGGER__EVENT:
			setEvent((Event) null);
			return;
		case UMLPackage.TRIGGER__PORT:
			getPorts().clear();
			return;
		}
		eDynamicUnset(featureID);
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	@Override
	public boolean eIsSet(int featureID) {
		switch (featureID) {
		case UMLPackage.TRIGGER__EANNOTATIONS:
			return !getEAnnotations().isEmpty();
		case UMLPackage.TRIGGER__OWNED_COMMENT:
			return !getOwnedComments().isEmpty();
		case UMLPackage.TRIGGER__OWNED_ELEMENT:
			return isSetOwnedElements();
		case UMLPackage.TRIGGER__OWNER:
			return isSetOwner();
		case UMLPackage.TRIGGER__CLIENT_DEPENDENCY:
			return !getClientDependencies().isEmpty();
		case UMLPackage.TRIGGER__NAME:
			return isSetName();
		case UMLPackage.TRIGGER__NAME_EXPRESSION:
			return basicGetNameExpression() != null;
		case UMLPackage.TRIGGER__NAMESPACE:
			return isSetNamespace();
		case UMLPackage.TRIGGER__QUALIFIED_NAME:
			return QUALIFIED_NAME_EDEFAULT == null ? getQualifiedName() != null : !QUALIFIED_NAME_EDEFAULT.equals(getQualifiedName());
		case UMLPackage.TRIGGER__VISIBILITY:
			return isSetVisibility();
		case UMLPackage.TRIGGER__EVENT:
			return basicGetEvent() != null;
		case UMLPackage.TRIGGER__PORT:
			return !getPorts().isEmpty();
		}
		return eDynamicIsSet(featureID);
	}

	/**
	 * <!-- begin-user-doc -->
	 * <!-- end-user-doc -->
	 * @generated
	 */
	@Override
	@SuppressWarnings("unchecked")
	public Object eInvoke(int operationID, EList<?> arguments)
			throws InvocationTargetException {
		switch (operationID) {
		case UMLPackage.TRIGGER___GET_EANNOTATION__STRING:
			return getEAnnotation((String) arguments.get(0));
		case UMLPackage.TRIGGER___VALIDATE_HAS_OWNER__DIAGNOSTICCHAIN_MAP:
			return validateHasOwner((DiagnosticChain) arguments.get(0), (Map<Object, Object>) arguments.get(1));
		case UMLPackage.TRIGGER___VALIDATE_NOT_OWN_SELF__DIAGNOSTICCHAIN_MAP:
			return validateNotOwnSelf((DiagnosticChain) arguments.get(0), (Map<Object, Object>) arguments.get(1));
		case UMLPackage.TRIGGER___ADD_KEYWORD__STRING:
			return addKeyword((String) arguments.get(0));
		case UMLPackage.TRIGGER___APPLY_STEREOTYPE__STEREOTYPE:
			return applyStereotype((Stereotype) arguments.get(0));
		case UMLPackage.TRIGGER___CREATE_EANNOTATION__STRING:
			return createEAnnotation((String) arguments.get(0));
		case UMLPackage.TRIGGER___DESTROY:
			destroy();
			return null;
		case UMLPackage.TRIGGER___GET_KEYWORDS:
			return getKeywords();
		case UMLPackage.TRIGGER___GET_APPLICABLE_STEREOTYPE__STRING:
			return getApplicableStereotype((String) arguments.get(0));
		case UMLPackage.TRIGGER___GET_APPLICABLE_STEREOTYPES:
			return getApplicableStereotypes();
		case UMLPackage.TRIGGER___GET_APPLIED_STEREOTYPE__STRING:
			return getAppliedStereotype((String) arguments.get(0));
		case UMLPackage.TRIGGER___GET_APPLIED_STEREOTYPES:
			return getAppliedStereotypes();
		case UMLPackage.TRIGGER___GET_APPLIED_SUBSTEREOTYPE__STEREOTYPE_STRING:
			return getAppliedSubstereotype((Stereotype) arguments.get(0), (String) arguments.get(1));
		case UMLPackage.TRIGGER___GET_APPLIED_SUBSTEREOTYPES__STEREOTYPE:
			return getAppliedSubstereotypes((Stereotype) arguments.get(0));
		case UMLPackage.TRIGGER___GET_MODEL:
			return getModel();
		case UMLPackage.TRIGGER___GET_NEAREST_PACKAGE:
			return getNearestPackage();
		case UMLPackage.TRIGGER___GET_RELATIONSHIPS:
			return getRelationships();
		case UMLPackage.TRIGGER___GET_RELATIONSHIPS__ECLASS:
			return getRelationships((EClass) arguments.get(0));
		case UMLPackage.TRIGGER___GET_REQUIRED_STEREOTYPE__STRING:
			return getRequiredStereotype((String) arguments.get(0));
		case UMLPackage.TRIGGER___GET_REQUIRED_STEREOTYPES:
			return getRequiredStereotypes();
		case UMLPackage.TRIGGER___GET_SOURCE_DIRECTED_RELATIONSHIPS:
			return getSourceDirectedRelationships();
		case UMLPackage.TRIGGER___GET_SOURCE_DIRECTED_RELATIONSHIPS__ECLASS:
			return getSourceDirectedRelationships((EClass) arguments.get(0));
		case UMLPackage.TRIGGER___GET_STEREOTYPE_APPLICATION__STEREOTYPE:
			return getStereotypeApplication((Stereotype) arguments.get(0));
		case UMLPackage.TRIGGER___GET_STEREOTYPE_APPLICATIONS:
			return getStereotypeApplications();
		case UMLPackage.TRIGGER___GET_TARGET_DIRECTED_RELATIONSHIPS:
			return getTargetDirectedRelationships();
		case UMLPackage.TRIGGER___GET_TARGET_DIRECTED_RELATIONSHIPS__ECLASS:
			return getTargetDirectedRelationships((EClass) arguments.get(0));
		case UMLPackage.TRIGGER___GET_VALUE__STEREOTYPE_STRING:
			return getValue((Stereotype) arguments.get(0), (String) arguments.get(1));
		case UMLPackage.TRIGGER___HAS_KEYWORD__STRING:
			return hasKeyword((String) arguments.get(0));
		case UMLPackage.TRIGGER___HAS_VALUE__STEREOTYPE_STRING:
			return hasValue((Stereotype) arguments.get(0), (String) arguments.get(1));
		case UMLPackage.TRIGGER___IS_STEREOTYPE_APPLICABLE__STEREOTYPE:
			return isStereotypeApplicable((Stereotype) arguments.get(0));
		case UMLPackage.TRIGGER___IS_STEREOTYPE_APPLIED__STEREOTYPE:
			return isStereotypeApplied((Stereotype) arguments.get(0));
		case UMLPackage.TRIGGER___IS_STEREOTYPE_REQUIRED__STEREOTYPE:
			return isStereotypeRequired((Stereotype) arguments.get(0));
		case UMLPackage.TRIGGER___REMOVE_KEYWORD__STRING:
			return removeKeyword((String) arguments.get(0));
		case UMLPackage.TRIGGER___SET_VALUE__STEREOTYPE_STRING_OBJECT:
			setValue((Stereotype) arguments.get(0), (String) arguments.get(1), arguments.get(2));
			return null;
		case UMLPackage.TRIGGER___UNAPPLY_STEREOTYPE__STEREOTYPE:
			return unapplyStereotype((Stereotype) arguments.get(0));
		case UMLPackage.TRIGGER___ALL_OWNED_ELEMENTS:
			return allOwnedElements();
		case UMLPackage.TRIGGER___MUST_BE_OWNED:
			return mustBeOwned();
		case UMLPackage.TRIGGER___VALIDATE_VISIBILITY_NEEDS_OWNERSHIP__DIAGNOSTICCHAIN_MAP:
			return validateVisibilityNeedsOwnership((DiagnosticChain) arguments.get(0), (Map<Object, Object>) arguments.get(1));
		case UMLPackage.TRIGGER___VALIDATE_HAS_QUALIFIED_NAME__DIAGNOSTICCHAIN_MAP:
			return validateHasQualifiedName((DiagnosticChain) arguments.get(0), (Map<Object, Object>) arguments.get(1));
		case UMLPackage.TRIGGER___VALIDATE_HAS_NO_QUALIFIED_NAME__DIAGNOSTICCHAIN_MAP:
			return validateHasNoQualifiedName((DiagnosticChain) arguments.get(0), (Map<Object, Object>) arguments.get(1));
		case UMLPackage.TRIGGER___CREATE_DEPENDENCY__NAMEDELEMENT:
			return createDependency((NamedElement) arguments.get(0));
		case UMLPackage.TRIGGER___CREATE_USAGE__NAMEDELEMENT:
			return createUsage((NamedElement) arguments.get(0));
		case UMLPackage.TRIGGER___GET_LABEL:
			return getLabel();
		case UMLPackage.TRIGGER___GET_LABEL__BOOLEAN:
			return getLabel((Boolean) arguments.get(0));
		case UMLPackage.TRIGGER___GET_NAMESPACE:
			return getNamespace();
		case UMLPackage.TRIGGER___ALL_NAMESPACES:
			return allNamespaces();
		case UMLPackage.TRIGGER___ALL_OWNING_PACKAGES:
			return allOwningPackages();
		case UMLPackage.TRIGGER___IS_DISTINGUISHABLE_FROM__NAMEDELEMENT_NAMESPACE:
			return isDistinguishableFrom((NamedElement) arguments.get(0), (Namespace) arguments.get(1));
		case UMLPackage.TRIGGER___GET_QUALIFIED_NAME:
			return getQualifiedName();
		case UMLPackage.TRIGGER___SEPARATOR:
			return separator();
		case UMLPackage.TRIGGER___GET_CLIENT_DEPENDENCIES:
			return getClientDependencies();
		case UMLPackage.TRIGGER___VALIDATE_TRIGGER_WITH_PORTS__DIAGNOSTICCHAIN_MAP:
			return validateTriggerWithPorts((DiagnosticChain) arguments.get(0), (Map<Object, Object>) arguments.get(1));
		}
		return eDynamicInvoke(operationID, arguments);
	}

} //TriggerImpl
