001/*
002 *                    BioJava development code
003 *
004 * This code may be freely distributed and modified under the
005 * terms of the GNU Lesser General Public Licence.  This should
006 * be distributed with the code.  If you do not have a copy,
007 * see:
008 *
009 *      http://www.gnu.org/copyleft/lesser.html
010 *
011 * Copyright for this code is held jointly by the individual
012 * authors.  These should be listed in @author doc comments.
013 *
014 * For more information on the BioJava project and its aims,
015 * or to join the biojava-l mailing list, visit the home page
016 * at:
017 *
018 *      http://www.biojava.org/
019 *
020 * Created on 01-21-2010
021 */
022
023package org.biojava.nbio.core.sequence.features;
024
025import org.biojava.nbio.core.sequence.location.template.AbstractLocation;
026import org.biojava.nbio.core.sequence.template.AbstractSequence;
027import org.biojava.nbio.core.sequence.template.Compound;
028
029import java.util.ArrayList;
030import java.util.Comparator;
031import java.util.HashMap;
032import java.util.List;
033import java.util.Map;
034
035/**
036 * A feature is currently any descriptive item that can be associated with a sequence position(s)
037 * A feature has a type and a source which is currently a string to allow flexibility for the user
038 * Ideally well defined features should have a class to describe attributes of that feature
039 * @author Scooter Willis 
040 */
041public abstract class AbstractFeature<S extends AbstractSequence<C>, C extends Compound>
042                implements FeatureInterface<S, C> {
043        List<FeatureInterface<S, C>> childrenFeatures = new ArrayList<>();
044        FeatureInterface<S, C> parentFeature;
045        AbstractLocation sequenceLocation;
046        String type = "";
047        String source = "";
048        private String description = "";
049        private String shortDescription = "";
050        private Object userObject = null;
051        private Map<String, List<Qualifier>> Qualifiers = new HashMap<>();
052
053        /**
054         * A feature has a type and a source
055         * @param type
056         * @param source
057         */
058        public AbstractFeature(String type,String source){
059                this.type = type;
060                this.source = source;
061        }
062
063        /**
064         * A feature could be a single sequence position like a mutation or a post translational modification of an amino acid.
065         * It could also be the docking interface of N number of amino acids on the surface. The location wold then be a collection
066         * of sequence positions instead of a single sequence position or the begin and end of a sequence segment.
067         * @return
068         */
069
070        @Override
071        public AbstractLocation getLocations() {
072                return sequenceLocation;
073        }
074
075        /**
076         *  A feature could be a single sequence position like a mutation or a post translational modification of an amino acid.
077         * It could also be the docking interface of N number of amino acids on the surface. The location wold then be a collection
078         * of sequence positions instead of a single sequence position or the begin and end of a sequence segment.
079         * @param loc
080         */
081        @Override
082        public void setLocation(AbstractLocation loc) {
083                sequenceLocation = loc;
084        }
085
086        /**
087         * The feature type
088         * @return
089         */
090        @Override
091        public String getType() {
092                return type;
093        }
094
095        /**
096         * Set the feature type
097         * @param type
098         */
099        @Override
100        public void setType(String type) {
101                this.type = type;
102        }
103
104        /**
105         * The feature source
106         * @return
107         */
108
109        @Override
110        public String getSource() {
111                return source;
112        }
113
114        /**
115         * Set the feature source
116         * @param source
117         */
118        @Override
119        public void setSource(String source) {
120                this.source = source;
121        }
122
123        /**
124         * A feature can be the child or contained by a parent feature. An example is a Helix feature could contain
125         * children features. A PFAM domain could contain secondary structures.
126         * @param feature
127         */
128        @Override
129        public void setParentFeature(FeatureInterface<S, C> feature) {
130                parentFeature = feature;
131        }
132
133        /**
134         * Get the parent Feature
135         * @return
136         */
137        @Override
138        public FeatureInterface<S, C> getParentFeature() {
139           return parentFeature;
140        }
141
142        /**
143         * Get the children features
144         * @return
145         */
146        @Override
147        public List<FeatureInterface<S, C>> getChildrenFeatures() {
148                return childrenFeatures;
149        }
150
151        /**
152         * Set the children features
153         * @param features
154         */
155        @Override
156        public void setChildrenFeatures(List<FeatureInterface<S, C>> features) {
157                childrenFeatures = features;
158
159        }
160
161        /**
162         * @return the description
163         */
164        @Override
165        public String getDescription() {
166                return description;
167        }
168
169        /**
170         * @param description the description to set
171         */
172        @Override
173        public void setDescription(String description) {
174                this.description = description;
175        }
176
177        /**
178         * @return the shortDescription
179         */
180        @Override
181        public String getShortDescription() {
182                return shortDescription;
183        }
184
185        /**
186         * @param shortDescription the shortDescription to set
187         */
188        @Override
189        public void setShortDescription(String shortDescription) {
190                this.shortDescription = shortDescription;
191        }
192
193        /**
194         * Sort features by start position and then longest length. When features are added
195         * having them sorted by start position and then longest length helps on the layout
196         * of overlapping features so they are delivered in a proper order.
197         */
198
199        public static final Comparator<FeatureInterface<?, ?>> LOCATION_LENGTH = new Comparator<FeatureInterface<?, ?>>() {
200
201                @Override
202                public int compare(FeatureInterface<?, ?> e1, FeatureInterface<?, ?> e2) {
203                        double v1 = e1.getLocations().getStart().getPosition();
204                        double v2 = e2.getLocations().getStart().getPosition();
205                        if (v1 < v2) {
206                                return -1;
207                        } else if (v1 > v2) {
208                                return 1;
209                        } else {
210                                double end1 = e1.getLocations().getEnd().getPosition();
211                                double end2 = e2.getLocations().getEnd().getPosition();
212                                if(end1 > end2)
213                                        return -1;
214                                else if(end1 < end2)
215                                        return 1;
216                                else
217                                return 0;
218                        }
219
220                }
221        };
222
223         /**
224         * Sort features by length. //TODO need to handle cases where features have multiple locations, strand etc
225         *
226         */
227
228        static public final Comparator<FeatureInterface<?, ?>> LENGTH = new Comparator<FeatureInterface<?, ?>>() {
229
230                @Override
231                public int compare(FeatureInterface<?, ?> e1, FeatureInterface<?, ?> e2) {
232                        double v1 = Math.abs(e1.getLocations().getEnd().getPosition()- e1.getLocations().getStart().getPosition());
233                        double v2 = Math.abs(e2.getLocations().getEnd().getPosition() -  e2.getLocations().getStart().getPosition());
234                        if (v1 < v2) {
235                                return -1;
236                        } else if (v1 > v2) {
237                                return 1;
238                        } else {
239                                return 0;
240                        }
241
242                }
243        };
244
245        /**
246         * Sort features by type
247         */
248        public static final Comparator<FeatureInterface<?, ?>> TYPE = new Comparator<FeatureInterface<?, ?>>() {
249
250                @Override
251                public int compare(FeatureInterface<?, ?> o1, FeatureInterface<?, ?> o2) {
252                        return o1.getType().compareTo(o2.getType());
253                }
254        };
255
256        /**
257         * @return the userObject
258         */
259        @Override
260        public Object getUserObject() {
261                return userObject;
262        }
263
264        /**
265         * Allow the user to associate an object with the feature. This way if a feature which is displayed in a GUI
266         * is clicked on the application can then get a user defined object associated with the feature.
267         * @param userObject the userObject to set
268         */
269        @Override
270        public void setUserObject(Object userObject) {
271                this.userObject = userObject;
272        }
273
274        @Override
275        public Map<String, List<Qualifier>> getQualifiers() {
276                // TODO Auto-generated method stub
277                return Qualifiers;
278        }
279
280        @Override
281        public void setQualifiers(Map<String, List<Qualifier>> qualifiers) {
282                // TODO Auto-generated method stub
283                Qualifiers = qualifiers;
284
285        }
286
287        @Override
288        public void addQualifier(String key, Qualifier qualifier) {
289                // Check for key. Update list of values
290                if (Qualifiers.containsKey(key)){
291                        List<Qualifier> vals = Qualifiers.get(key);
292                        vals.add(qualifier);
293                        Qualifiers.put(key, vals);
294                } else {
295                        List<Qualifier> vals = new ArrayList<>();
296                        vals.add(qualifier);
297                        Qualifiers.put(key, vals);
298                }
299
300        }
301
302}