/*******************************************************************************
 * Copyright (c) 2004, 2008 Sybase, Inc. and others.
 *
 * 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:
 *     Sybase, Inc. - initial API and implementation
 *******************************************************************************/
package org.eclipse.jst.jsf.facesconfig.ui.pageflow.model.impl;

import org.eclipse.draw2d.geometry.Dimension;
import org.eclipse.emf.common.notify.Notification;
import org.eclipse.emf.common.notify.NotificationChain;
import org.eclipse.emf.ecore.EClass;
import org.eclipse.emf.ecore.EStructuralFeature;
import org.eclipse.emf.ecore.InternalEObject;
import org.eclipse.emf.ecore.impl.ENotificationImpl;
import org.eclipse.emf.ecore.impl.EObjectImpl;
import org.eclipse.emf.ecore.util.EcoreUtil;
import org.eclipse.jst.jsf.facesconfig.ui.pageflow.PageflowMessages;
import org.eclipse.jst.jsf.facesconfig.ui.pageflow.model.PageflowLink;
import org.eclipse.jst.jsf.facesconfig.ui.pageflow.model.PageflowLinkBendpoint;
import org.eclipse.jst.jsf.facesconfig.ui.pageflow.model.PageflowPackage;
import org.eclipse.osgi.util.NLS;

/**
 * <!-- begin-user-doc --> An implementation of the model object '<em><b>PF Link Bendpoint</b></em>'.
 * <!-- end-user-doc -->
 * <p>
 * The following features are implemented:
 * <ul>
 * <li>{@link org.eclipse.jst.jsf.facesconfig.ui.pageflow.model.impl.PageflowLinkBendpointImpl#getD1Width <em>D1 Width</em>}</li>
 * <li>{@link org.eclipse.jst.jsf.facesconfig.ui.pageflow.model.impl.PageflowLinkBendpointImpl#getD1Height <em>D1 Height</em>}</li>
 * <li>{@link org.eclipse.jst.jsf.facesconfig.ui.pageflow.model.impl.PageflowLinkBendpointImpl#getD2Width <em>D2 Width</em>}</li>
 * <li>{@link org.eclipse.jst.jsf.facesconfig.ui.pageflow.model.impl.PageflowLinkBendpointImpl#getD2Height <em>D2 Height</em>}</li>
 * <li>{@link org.eclipse.jst.jsf.facesconfig.ui.pageflow.model.impl.PageflowLinkBendpointImpl#getWeight <em>Weight</em>}</li>
 * <li>{@link org.eclipse.jst.jsf.facesconfig.ui.pageflow.model.impl.PageflowLinkBendpointImpl#getLink <em>Link</em>}</li>
 * </ul>
 * </p>
 * 
 * @generated
 */
public class PageflowLinkBendpointImpl extends EObjectImpl implements PageflowLinkBendpoint {
	/**
	 * The default value of the '{@link #getD1Width() <em>D1 Width</em>}'
	 * attribute. <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @see #getD1Width()
	 * @generated
	 * @ordered
	 */
	protected static final int D1_WIDTH_EDEFAULT = 0;

	/**
	 * The cached value of the '{@link #getD1Width() <em>D1 Width</em>}'
	 * attribute. <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @see #getD1Width()
	 * @generated
	 * @ordered
	 */
	protected int d1Width = D1_WIDTH_EDEFAULT;

	/**
	 * The default value of the '{@link #getD1Height() <em>D1 Height</em>}'
	 * attribute. <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @see #getD1Height()
	 * @generated
	 * @ordered
	 */
	protected static final int D1_HEIGHT_EDEFAULT = 0;

	/**
	 * The cached value of the '{@link #getD1Height() <em>D1 Height</em>}'
	 * attribute. <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @see #getD1Height()
	 * @generated
	 * @ordered
	 */
	protected int d1Height = D1_HEIGHT_EDEFAULT;

	/**
	 * The default value of the '{@link #getD2Width() <em>D2 Width</em>}'
	 * attribute. <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @see #getD2Width()
	 * @generated
	 * @ordered
	 */
	protected static final int D2_WIDTH_EDEFAULT = 0;

	/**
	 * The cached value of the '{@link #getD2Width() <em>D2 Width</em>}'
	 * attribute. <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @see #getD2Width()
	 * @generated
	 * @ordered
	 */
	protected int d2Width = D2_WIDTH_EDEFAULT;

	/**
	 * The default value of the '{@link #getD2Height() <em>D2 Height</em>}'
	 * attribute. <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @see #getD2Height()
	 * @generated
	 * @ordered
	 */
	protected static final int D2_HEIGHT_EDEFAULT = 0;

	/**
	 * The cached value of the '{@link #getD2Height() <em>D2 Height</em>}'
	 * attribute. <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @see #getD2Height()
	 * @generated
	 * @ordered
	 */
	protected int d2Height = D2_HEIGHT_EDEFAULT;

	/**
	 * The default value of the '{@link #getWeight() <em>Weight</em>}'
	 * attribute. <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @see #getWeight()
	 * @generated
	 * @ordered
	 */
	protected static final float WEIGHT_EDEFAULT = 0.5F;

	/**
	 * The cached value of the '{@link #getWeight() <em>Weight</em>}'
	 * attribute. <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @see #getWeight()
	 * @generated
	 * @ordered
	 */
	protected float weight = WEIGHT_EDEFAULT;

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated NOT
	 */
	private Dimension dimStart, dimEnd;

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

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	protected EClass eStaticClass() {
		return PageflowPackage.eINSTANCE.getPFLinkBendpoint();
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public int getD1Width() {
		return d1Width;
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public void setD1Width(int newD1Width) {
		int oldD1Width = d1Width;
		d1Width = newD1Width;
		if (eNotificationRequired())
			eNotify(new ENotificationImpl(this, Notification.SET,
					PageflowPackage.PF_LINK_BENDPOINT__D1_WIDTH, oldD1Width,
					d1Width));
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public int getD1Height() {
		return d1Height;
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public void setD1Height(int newD1Height) {
		int oldD1Height = d1Height;
		d1Height = newD1Height;
		if (eNotificationRequired())
			eNotify(new ENotificationImpl(this, Notification.SET,
					PageflowPackage.PF_LINK_BENDPOINT__D1_HEIGHT, oldD1Height,
					d1Height));
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public int getD2Width() {
		return d2Width;
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public void setD2Width(int newD2Width) {
		int oldD2Width = d2Width;
		d2Width = newD2Width;
		if (eNotificationRequired())
			eNotify(new ENotificationImpl(this, Notification.SET,
					PageflowPackage.PF_LINK_BENDPOINT__D2_WIDTH, oldD2Width,
					d2Width));
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public int getD2Height() {
		return d2Height;
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public void setD2Height(int newD2Height) {
		int oldD2Height = d2Height;
		d2Height = newD2Height;
		if (eNotificationRequired())
			eNotify(new ENotificationImpl(this, Notification.SET,
					PageflowPackage.PF_LINK_BENDPOINT__D2_HEIGHT, oldD2Height,
					d2Height));
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public float getWeight() {
		return weight;
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public void setWeight(float newWeight) {
		float oldWeight = weight;
		weight = newWeight;
		if (eNotificationRequired())
			eNotify(new ENotificationImpl(this, Notification.SET,
					PageflowPackage.PF_LINK_BENDPOINT__WEIGHT, oldWeight,
					weight));
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public PageflowLink getLink() {
		if (eContainerFeatureID != PageflowPackage.PF_LINK_BENDPOINT__LINK)
			return null;
		return (PageflowLink) eContainer;
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public void setLink(PageflowLink newLink) {
		if (newLink != eContainer
				|| (eContainerFeatureID != PageflowPackage.PF_LINK_BENDPOINT__LINK && newLink != null)) {
			if (EcoreUtil.isAncestor(this, newLink))
				throw new IllegalArgumentException(
						NLS.bind(PageflowMessages.PageflowNode_NoRecursiveContainment,toString()));
			NotificationChain msgs = null;
			if (eContainer != null)
				msgs = eBasicRemoveFromContainer(msgs);
			if (newLink != null)
				msgs = ((InternalEObject) newLink).eInverseAdd(this,
						PageflowPackage.PF_LINK__BEND_POINTS, PageflowLink.class,
						msgs);
			msgs = eBasicSetContainer((InternalEObject) newLink,
					PageflowPackage.PF_LINK_BENDPOINT__LINK, msgs);
			if (msgs != null)
				msgs.dispatch();
		} else if (eNotificationRequired())
			eNotify(new ENotificationImpl(this, Notification.SET,
					PageflowPackage.PF_LINK_BENDPOINT__LINK, newLink, newLink));
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public NotificationChain eInverseAdd(InternalEObject otherEnd,
			int featureID, Class baseClass, NotificationChain msgs) {
		if (featureID >= 0) {
			switch (eDerivedStructuralFeatureID(featureID, baseClass)) {
			case PageflowPackage.PF_LINK_BENDPOINT__LINK:
				if (eContainer != null)
					msgs = eBasicRemoveFromContainer(msgs);
				return eBasicSetContainer(otherEnd,
						PageflowPackage.PF_LINK_BENDPOINT__LINK, msgs);
			default:
				return eDynamicInverseAdd(otherEnd, featureID, baseClass, msgs);
			}
		}
		if (eContainer != null)
			msgs = eBasicRemoveFromContainer(msgs);
		return eBasicSetContainer(otherEnd, featureID, msgs);
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public NotificationChain eInverseRemove(InternalEObject otherEnd,
			int featureID, Class baseClass, NotificationChain msgs) {
		if (featureID >= 0) {
			switch (eDerivedStructuralFeatureID(featureID, baseClass)) {
			case PageflowPackage.PF_LINK_BENDPOINT__LINK:
				return eBasicSetContainer(null,
						PageflowPackage.PF_LINK_BENDPOINT__LINK, msgs);
			default:
				return eDynamicInverseRemove(otherEnd, featureID, baseClass,
						msgs);
			}
		}
		return eBasicSetContainer(null, featureID, msgs);
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public NotificationChain eBasicRemoveFromContainer(NotificationChain msgs) {
		if (eContainerFeatureID >= 0) {
			switch (eContainerFeatureID) {
			case PageflowPackage.PF_LINK_BENDPOINT__LINK:
				return eContainer.eInverseRemove(this,
						PageflowPackage.PF_LINK__BEND_POINTS, PageflowLink.class,
						msgs);
			default:
				return eDynamicBasicRemoveFromContainer(msgs);
			}
		}
		return eContainer.eInverseRemove(this,
				EOPPOSITE_FEATURE_BASE - eContainerFeatureID, null, msgs);
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public Object eGet(EStructuralFeature eFeature, boolean resolve) {
		switch (eDerivedStructuralFeatureID(eFeature)) {
		case PageflowPackage.PF_LINK_BENDPOINT__D1_WIDTH:
			return new Integer(getD1Width());
		case PageflowPackage.PF_LINK_BENDPOINT__D1_HEIGHT:
			return new Integer(getD1Height());
		case PageflowPackage.PF_LINK_BENDPOINT__D2_WIDTH:
			return new Integer(getD2Width());
		case PageflowPackage.PF_LINK_BENDPOINT__D2_HEIGHT:
			return new Integer(getD2Height());
		case PageflowPackage.PF_LINK_BENDPOINT__WEIGHT:
			return new Float(getWeight());
		case PageflowPackage.PF_LINK_BENDPOINT__LINK:
			return getLink();
		}
		return eDynamicGet(eFeature, resolve);
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public void eSet(EStructuralFeature eFeature, Object newValue) {
		switch (eDerivedStructuralFeatureID(eFeature)) {
		case PageflowPackage.PF_LINK_BENDPOINT__D1_WIDTH:
			setD1Width(((Integer) newValue).intValue());
			return;
		case PageflowPackage.PF_LINK_BENDPOINT__D1_HEIGHT:
			setD1Height(((Integer) newValue).intValue());
			return;
		case PageflowPackage.PF_LINK_BENDPOINT__D2_WIDTH:
			setD2Width(((Integer) newValue).intValue());
			return;
		case PageflowPackage.PF_LINK_BENDPOINT__D2_HEIGHT:
			setD2Height(((Integer) newValue).intValue());
			return;
		case PageflowPackage.PF_LINK_BENDPOINT__WEIGHT:
			setWeight(((Float) newValue).floatValue());
			return;
		case PageflowPackage.PF_LINK_BENDPOINT__LINK:
			setLink((PageflowLink) newValue);
			return;
		}
		eDynamicSet(eFeature, newValue);
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public void eUnset(EStructuralFeature eFeature) {
		switch (eDerivedStructuralFeatureID(eFeature)) {
		case PageflowPackage.PF_LINK_BENDPOINT__D1_WIDTH:
			setD1Width(D1_WIDTH_EDEFAULT);
			return;
		case PageflowPackage.PF_LINK_BENDPOINT__D1_HEIGHT:
			setD1Height(D1_HEIGHT_EDEFAULT);
			return;
		case PageflowPackage.PF_LINK_BENDPOINT__D2_WIDTH:
			setD2Width(D2_WIDTH_EDEFAULT);
			return;
		case PageflowPackage.PF_LINK_BENDPOINT__D2_HEIGHT:
			setD2Height(D2_HEIGHT_EDEFAULT);
			return;
		case PageflowPackage.PF_LINK_BENDPOINT__WEIGHT:
			setWeight(WEIGHT_EDEFAULT);
			return;
		case PageflowPackage.PF_LINK_BENDPOINT__LINK:
			setLink((PageflowLink) null);
			return;
		}
		eDynamicUnset(eFeature);
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public boolean eIsSet(EStructuralFeature eFeature) {
		switch (eDerivedStructuralFeatureID(eFeature)) {
		case PageflowPackage.PF_LINK_BENDPOINT__D1_WIDTH:
			return d1Width != D1_WIDTH_EDEFAULT;
		case PageflowPackage.PF_LINK_BENDPOINT__D1_HEIGHT:
			return d1Height != D1_HEIGHT_EDEFAULT;
		case PageflowPackage.PF_LINK_BENDPOINT__D2_WIDTH:
			return d2Width != D2_WIDTH_EDEFAULT;
		case PageflowPackage.PF_LINK_BENDPOINT__D2_HEIGHT:
			return d2Height != D2_HEIGHT_EDEFAULT;
		case PageflowPackage.PF_LINK_BENDPOINT__WEIGHT:
			return weight != WEIGHT_EDEFAULT;
		case PageflowPackage.PF_LINK_BENDPOINT__LINK:
			return getLink() != null;
		}
		return eDynamicIsSet(eFeature);
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated
	 */
	public String toString() {
		if (eIsProxy())
			return super.toString();

		StringBuffer result = new StringBuffer(super.toString());
		result.append(" (d1Width: "); //$NON-NLS-1$
		result.append(d1Width);
		result.append(", d1Height: "); //$NON-NLS-1$
		result.append(d1Height);
		result.append(", d2Width: "); //$NON-NLS-1$
		result.append(d2Width);
		result.append(", d2Height: "); //$NON-NLS-1$
		result.append(d2Height);
		result.append(", weight: "); //$NON-NLS-1$
		result.append(weight);
		result.append(')');
		return result.toString();
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated NOT
	 */
	public Dimension getFirstRelativeDimension() {
		dimStart = new Dimension(getD1Width(), getD1Height());
		return dimStart;
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated NOT
	 */
	public Dimension getSecondRelativeDimension() {
		dimEnd = new Dimension(getD2Width(), getD2Height());
		return dimEnd;
	}

	/**
	 * <!-- begin-user-doc --> <!-- end-user-doc -->
	 * 
	 * @generated NOT
	 */
	public void setRelativeDimensions(Dimension dim1, Dimension dim2) {
		dimStart = dim1;
		dimEnd = dim2;
		setD1Width(dimStart.width);
		setD1Height(dimStart.height);

		setD2Width(dimEnd.width);
		setD2Height(dimEnd.height);
	}

}
// PFLinkBendpointImpl
