/* * BioJava development code * * This code may be freely distributed and modified under the * terms of the GNU Lesser General Public Licence. This should * be distributed with the code. If you do not have a copy, * see: * * http://www.gnu.org/copyleft/lesser.html * * Copyright for this code is held jointly by the individual * authors. These should be listed in @author doc comments. * * For more information on the BioJava project and its aims, * or to join the biojava-l mailing list, visit the home page * at: * * http://www.biojava.org/ * */ package org.biojava.bio.program.tagvalue; import java.util.ArrayList; import java.util.List; import org.biojava.bio.Annotation; import org.biojava.bio.AnnotationType; import org.biojava.bio.CollectionConstraint; import org.biojava.bio.PropertyConstraint; import org.biojava.bio.SmallAnnotation; import org.biojava.utils.AssertionFailure; import org.biojava.utils.ChangeVetoException; /** *

* Builds an Annotation tree from TagValue events using an AnnotationType to * work out which fields are of what type. *

* * @since 1.2 * @author Matthew Pocock */ public class AnnotationBuilder implements TagValueListener { private List annotationStack; private AnnotationType type; private Annotation last; /** *

* Make a new AnnotationBuilder that will build Annotation instances of a * given type. *

* *

* The type is used to provide appropriate accessors for properties. As tag * -value events stream through this TagValueListener, they will be matched * against the properties of the annotation type. As sub-trees of events are * pushed, child annotation bundles will be pushed into the appropriate * properties. If any of the tag-value events are of a type that are not * accepted by the annotation type, a ClassCastException will be thrown. *

* * @param type the AnnotationType stating what will be built and how * @throws ClassCastException if any of the tag-value events are of * inappropriate type */ public AnnotationBuilder(AnnotationType type) { this.type = type; this.annotationStack = new ArrayList(); } /** *

* Get the last complete annotation built. *

* *

* The value of this is undefined before the first annotation has been built * and during the parsing of an event stream. *

* * @return the Annotation that was last built */ public Annotation getLast() { return last; } public void startRecord() { Frame top = new Frame(); top.annotation = new SmallAnnotation(); if(annotationStack.isEmpty()) { top.type = this.type; } else { Frame old = peek(annotationStack); CollectionConstraint cc = old.type.getConstraint(old.tag); PropertyConstraint pc = null; if (cc instanceof CollectionConstraint.AllValuesIn) { pc = ((CollectionConstraint.AllValuesIn) cc).getPropertyConstraint(); } if(pc instanceof PropertyConstraint.ByAnnotationType) { PropertyConstraint.ByAnnotationType pcat = (PropertyConstraint.ByAnnotationType) pc; top.type = pcat.getAnnotationType(); } else { top.type = AnnotationType.ANY; } } push(annotationStack, top); } public void endRecord() { Frame top = pop(annotationStack); last = top.annotation; if(!annotationStack.isEmpty()) { try { Frame old = peek(annotationStack); old.type.setProperty(old.annotation, old.tag, last); } catch (ChangeVetoException cve) { throw new AssertionFailure(cve); } } } public void startTag(Object tag) { peek(annotationStack).tag = tag; } public void value(TagValueContext ctxt, Object value) { try { Frame top = peek(annotationStack); top.type.setProperty(top.annotation, top.tag, value); } catch (ChangeVetoException cve) { throw new AssertionFailure(cve); } } public void endTag() {} private void push(List list, Frame frame) { list.add(frame); } private Frame peek(List list) { return (Frame) list.get(list.size() - 1); } private Frame pop(List list) { return (Frame) list.remove(list.size() - 1); } private static class Frame { public AnnotationType type; public Annotation annotation; public Object tag; } }