| /******************************************************************************* |
| * Copyright (c) 2009,2010 IBM Corporation and others. |
| * 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: |
| * IBM Corporation - initial API and implementation |
| *******************************************************************************/ |
| package org.eclipse.stem.diseasemodels.multipopulation.impl; |
| |
| import org.eclipse.emf.common.notify.Notification; |
| import org.eclipse.emf.common.notify.NotificationChain; |
| import org.eclipse.emf.common.util.EList; |
| import org.eclipse.emf.ecore.EClass; |
| import org.eclipse.emf.ecore.InternalEObject; |
| import org.eclipse.emf.ecore.impl.ENotificationImpl; |
| import org.eclipse.stem.core.common.DoubleValue; |
| import org.eclipse.stem.core.common.DoubleValueList; |
| import org.eclipse.stem.core.common.StringValue; |
| import org.eclipse.stem.core.graph.Node; |
| import org.eclipse.stem.core.graph.NodeLabel; |
| import org.eclipse.stem.core.model.STEMTime; |
| import org.eclipse.stem.diseasemodels.multipopulation.MultiPopulationSIRDiseaseModel; |
| import org.eclipse.stem.diseasemodels.multipopulation.MultipopulationPackage; |
| import org.eclipse.stem.diseasemodels.standard.DiseaseModelLabel; |
| import org.eclipse.stem.diseasemodels.standard.DiseaseModelLabelValue; |
| import org.eclipse.stem.diseasemodels.standard.SIRLabel; |
| import org.eclipse.stem.diseasemodels.standard.SIRLabelValue; |
| import org.eclipse.stem.diseasemodels.standard.StandardDiseaseModelLabel; |
| import org.eclipse.stem.diseasemodels.standard.StandardDiseaseModelLabelValue; |
| import org.eclipse.stem.diseasemodels.standard.StandardFactory; |
| import org.eclipse.stem.diseasemodels.standard.StandardPackage; |
| import org.eclipse.stem.diseasemodels.standard.impl.SIRLabelValueImpl; |
| |
| /** |
| * <!-- begin-user-doc --> |
| * An implementation of the model object '<em><b>Multi Population SIR Disease Model</b></em>'. |
| * <!-- end-user-doc --> |
| * <p> |
| * The following features are implemented: |
| * <ul> |
| * <li>{@link org.eclipse.stem.diseasemodels.multipopulation.impl.MultiPopulationSIRDiseaseModelImpl#getImmunityLossRate <em>Immunity Loss Rate</em>}</li> |
| * </ul> |
| * </p> |
| * |
| * @generated |
| */ |
| public class MultiPopulationSIRDiseaseModelImpl extends MultiPopulationSIDiseaseModelImpl implements MultiPopulationSIRDiseaseModel { |
| /** |
| * The cached value of the '{@link #getImmunityLossRate() <em>Immunity Loss Rate</em>}' containment reference. |
| * <!-- begin-user-doc --> |
| * <!-- end-user-doc --> |
| * @see #getImmunityLossRate() |
| * @generated |
| * @ordered |
| */ |
| protected DoubleValueList immunityLossRate; |
| |
| /** |
| * <!-- begin-user-doc --> |
| * <!-- end-user-doc --> |
| * @generated NOT |
| */ |
| public MultiPopulationSIRDiseaseModelImpl() { |
| super(); |
| } |
| |
| /** |
| * <!-- begin-user-doc --> |
| * <!-- end-user-doc --> |
| * @generated |
| */ |
| @Override |
| protected EClass eStaticClass() { |
| return MultipopulationPackage.Literals.MULTI_POPULATION_SIR_DISEASE_MODEL; |
| } |
| |
| /** |
| * <!-- begin-user-doc --> |
| * <!-- end-user-doc --> |
| * @generated |
| */ |
| public DoubleValueList getImmunityLossRate() { |
| return immunityLossRate; |
| } |
| |
| /** |
| * <!-- begin-user-doc --> |
| * <!-- end-user-doc --> |
| * @generated |
| */ |
| public NotificationChain basicSetImmunityLossRate(DoubleValueList newImmunityLossRate, NotificationChain msgs) { |
| DoubleValueList oldImmunityLossRate = immunityLossRate; |
| immunityLossRate = newImmunityLossRate; |
| if (eNotificationRequired()) { |
| ENotificationImpl notification = new ENotificationImpl(this, Notification.SET, MultipopulationPackage.MULTI_POPULATION_SIR_DISEASE_MODEL__IMMUNITY_LOSS_RATE, oldImmunityLossRate, newImmunityLossRate); |
| if (msgs == null) msgs = notification; else msgs.add(notification); |
| } |
| return msgs; |
| } |
| |
| /** |
| * <!-- begin-user-doc --> |
| * <!-- end-user-doc --> |
| * @generated |
| */ |
| public void setImmunityLossRate(DoubleValueList newImmunityLossRate) { |
| if (newImmunityLossRate != immunityLossRate) { |
| NotificationChain msgs = null; |
| if (immunityLossRate != null) |
| msgs = ((InternalEObject)immunityLossRate).eInverseRemove(this, EOPPOSITE_FEATURE_BASE - MultipopulationPackage.MULTI_POPULATION_SIR_DISEASE_MODEL__IMMUNITY_LOSS_RATE, null, msgs); |
| if (newImmunityLossRate != null) |
| msgs = ((InternalEObject)newImmunityLossRate).eInverseAdd(this, EOPPOSITE_FEATURE_BASE - MultipopulationPackage.MULTI_POPULATION_SIR_DISEASE_MODEL__IMMUNITY_LOSS_RATE, null, msgs); |
| msgs = basicSetImmunityLossRate(newImmunityLossRate, msgs); |
| if (msgs != null) msgs.dispatch(); |
| } |
| else if (eNotificationRequired()) |
| eNotify(new ENotificationImpl(this, Notification.SET, MultipopulationPackage.MULTI_POPULATION_SIR_DISEASE_MODEL__IMMUNITY_LOSS_RATE, newImmunityLossRate, newImmunityLossRate)); |
| } |
| |
| |
| |
| /** |
| * <!-- begin-user-doc --> |
| * <!-- end-user-doc --> |
| * @generated |
| */ |
| @Override |
| public NotificationChain eInverseRemove(InternalEObject otherEnd, int featureID, NotificationChain msgs) { |
| switch (featureID) { |
| case MultipopulationPackage.MULTI_POPULATION_SIR_DISEASE_MODEL__IMMUNITY_LOSS_RATE: |
| return basicSetImmunityLossRate(null, msgs); |
| } |
| return super.eInverseRemove(otherEnd, featureID, msgs); |
| } |
| |
| @Override |
| public StandardDiseaseModelLabelValue computeDiseaseDeltas(STEMTime time, |
| StandardDiseaseModelLabelValue currentState, |
| StandardDiseaseModelLabel diseaseLabel, long timeDelta, |
| DiseaseModelLabelValue returnValue) { |
| |
| // THIS method gets called multiple times, once for each population identifier |
| // in the population model. |
| // which population is "this".... |
| final SIRLabelValue currentSIR = (SIRLabelValue) currentState; |
| String thisPopulation = diseaseLabel.getPopulationModelLabel().getPopulationIdentifier(); |
| |
| // Compute deaths |
| final double adjustedInfectiousMortalityRate = getAdjustedInfectiousMortalityRate(timeDelta, thisPopulation); |
| final double diseaseDeaths = adjustedInfectiousMortalityRate |
| * currentSIR.getI(); |
| |
| // next get it's INDEX in the model |
| int populationIndex = this.getPopulationIndex(thisPopulation); |
| |
| // now we know the index of the current population being integrated. |
| // Get the correct transmission rate list from the MATRIX |
| EList<DoubleValue> transmissionVector = getTransmissionRate().getValueLists().get(populationIndex).getValues(); |
| |
| // ALL the other disease parameters are also DoubleValueLists. We now iterate through all populations |
| // get the specific rate parameters from EACH list based on this population index |
| double thisRecoveryRate = 0.0; |
| if(getRecoveryRate() != null) thisRecoveryRate = getRecoveryRate().getValues().get(populationIndex).getValue(); |
| double thisImmunityLossRate = 0.0; |
| if(getImmunityLossRate() != null) thisImmunityLossRate = getImmunityLossRate().getValues().get(populationIndex).getValue(); |
| |
| |
| // NOW iterate over every population (including this one) to compute new infections |
| // this includes infections within a population group |
| // and all the cross terms |
| double numberOfSusceptibleToInfected = 0.0; |
| double numberSusceptible = currentSIR.getS(); |
| Node thisNode = diseaseLabel.getNode(); |
| |
| EList<StringValue> groupList = getPopulationGroups().getValues(); |
| |
| for(int i = 0; i< transmissionVector.size(); i ++) { |
| // We need to get the identifier of the ith population model |
| String nextPop = groupList.get(i).getValue(); |
| |
| ////////////////// |
| // 1. compute BETA for transmission from the ith population to this population |
| double specificTransmission = transmissionVector.get(i).getValue(); |
| double adjustedTransmission = getAdjustedTransmissionRate(specificTransmission, timeDelta); |
| if(!this.isFrequencyDependent()) adjustedTransmission *= getTransmissionRateScaleFactor(diseaseLabel); |
| |
| |
| // we need to get the disease label for the ith population as well. |
| // to get the on site number of infectios individuals of type i |
| EList<NodeLabel> nodeLabels = thisNode.getLabels(); |
| for(int j = 0; j < nodeLabels.size(); j ++) { |
| NodeLabel label = nodeLabels.get(j); |
| if(label instanceof SIRLabel) { |
| if (this == ((SIRLabel)label).getDecorator()) { |
| StandardDiseaseModelLabel otherDiseaseLabel = (StandardDiseaseModelLabel) label; |
| // now check the popualtion type |
| String otherPopulation = otherDiseaseLabel.getPopulationModelLabel().getPopulationIdentifier(); |
| if(otherPopulation.equals(nextPop)) { |
| // Yes? |
| // then we found the label for the correct next population |
| |
| |
| // for this population we need to get the EFFECTIVE Infectious including |
| // ALL neighboring nodes |
| double onsiteInfectious = ((SIRLabelValue) otherDiseaseLabel.getTempValue()).getI(); |
| final double effectiveInfectious = getNormalizedEffectiveInfectious( |
| thisNode, otherDiseaseLabel, onsiteInfectious, |
| StandardPackage.Literals.SI_LABEL_VALUE__I); |
| |
| // ADD up the new incidence |
| numberOfSusceptibleToInfected += adjustedTransmission * numberSusceptible * effectiveInfectious; |
| } |
| } |
| }// if it's an SI label |
| }//for all labels in THIS node |
| }// For all population in the model |
| |
| double numberOfInfectedToRecovered = getAdjustedRecoveryRate(thisRecoveryRate, timeDelta) * currentSIR.getI(); |
| double numberOfRecoveredToSusceptible = getAdjustedImmunityLossRate(thisImmunityLossRate, timeDelta) * currentSIR.getR(); |
| |
| // Determine delta S |
| final double deltaS = - numberOfSusceptibleToInfected + numberOfRecoveredToSusceptible; |
| // Determine delta I |
| final double deltaI = numberOfSusceptibleToInfected - numberOfInfectedToRecovered - diseaseDeaths; |
| // Determine delta R |
| final double deltaR = numberOfInfectedToRecovered - numberOfRecoveredToSusceptible; |
| |
| SIRLabelValueImpl ret = (SIRLabelValueImpl)returnValue; |
| ret.setS(deltaS); |
| ret.setI(deltaI); |
| ret.setR(deltaR); |
| ret.setIncidence(numberOfSusceptibleToInfected); |
| ret.setDiseaseDeaths(diseaseDeaths); |
| return ret; |
| |
| } |
| |
| public double getAdjustedImmunityLossRate(double specificLossRate, long timeDelta) { |
| return (specificLossRate * ((double) timeDelta / (double) getTimePeriod()) ); |
| } // getAdjustedImmunityLossRate |
| |
| |
| @Override |
| public DiseaseModelLabel createDiseaseModelLabel(String populationIdentifier) { |
| return StandardFactory.eINSTANCE.createSIRLabel(); |
| } |
| |
| @Override |
| public DiseaseModelLabelValue createDiseaseModelLabelValue(String populationIdentifier) { |
| return StandardFactory.eINSTANCE.createSIRLabelValue(); |
| } |
| |
| |
| |
| /** |
| * <!-- begin-user-doc --> |
| * <!-- end-user-doc --> |
| * @generated |
| */ |
| @Override |
| public Object eGet(int featureID, boolean resolve, boolean coreType) { |
| switch (featureID) { |
| case MultipopulationPackage.MULTI_POPULATION_SIR_DISEASE_MODEL__IMMUNITY_LOSS_RATE: |
| return getImmunityLossRate(); |
| } |
| return super.eGet(featureID, resolve, coreType); |
| } |
| |
| /** |
| * <!-- begin-user-doc --> |
| * <!-- end-user-doc --> |
| * @generated |
| */ |
| @Override |
| public void eSet(int featureID, Object newValue) { |
| switch (featureID) { |
| case MultipopulationPackage.MULTI_POPULATION_SIR_DISEASE_MODEL__IMMUNITY_LOSS_RATE: |
| setImmunityLossRate((DoubleValueList)newValue); |
| return; |
| } |
| super.eSet(featureID, newValue); |
| } |
| |
| /** |
| * <!-- begin-user-doc --> |
| * <!-- end-user-doc --> |
| * @generated |
| */ |
| @Override |
| public void eUnset(int featureID) { |
| switch (featureID) { |
| case MultipopulationPackage.MULTI_POPULATION_SIR_DISEASE_MODEL__IMMUNITY_LOSS_RATE: |
| setImmunityLossRate((DoubleValueList)null); |
| return; |
| } |
| super.eUnset(featureID); |
| } |
| |
| /** |
| * <!-- begin-user-doc --> |
| * <!-- end-user-doc --> |
| * @generated |
| */ |
| @Override |
| public boolean eIsSet(int featureID) { |
| switch (featureID) { |
| case MultipopulationPackage.MULTI_POPULATION_SIR_DISEASE_MODEL__IMMUNITY_LOSS_RATE: |
| return immunityLossRate != null; |
| } |
| return super.eIsSet(featureID); |
| } |
| |
| } //MultiPopulationSIRDiseaseModelImpl |