checkpoint
[sbp.git] / src / edu / berkeley / sbp / Union.java
index 3348742..204404d 100644 (file)
@@ -7,11 +7,19 @@ import java.util.*;
 import java.lang.reflect.*;
 import java.lang.ref.*;
 
-/** <font color=green>an element which can produce one of several alternatives</font> */
+/**
+ *  <font color=green>an element which can produce one of several alternatives</font>.
+ *  <p>
+ *
+ *  Unlike the other Elements, Union is not immutable once
+ *  constructed.  To simulate this desirable feature, it is immutable
+ *  <i>once examined</i> by taking its iterator or calling contains().
+ */
 public class Union extends Element implements Iterable<Sequence> {
 
     private final String name;
     private final boolean synthetic;
+    private boolean viewed = false;
 
     private final List<Sequence> alternatives = new ArrayList<Sequence>();
 
@@ -31,11 +39,20 @@ public class Union extends Element implements Iterable<Sequence> {
         this.synthetic = synthetic;
     }
 
-    public boolean contains(Sequence s) { return alternatives.contains(s); }
-    public Iterator<Sequence> iterator() { return alternatives.iterator(); }
+    public boolean contains(Sequence s) {
+        viewed = true;
+        return alternatives.contains(s);
+    }
+
+    public Iterator<Sequence> iterator() {
+        viewed = true;
+        return alternatives.iterator();
+    }
 
     /** adds an alternative */
     public void add(Sequence s) {
+        if (viewed)
+            throw new RuntimeException("attempt to add a Sequence to a Union that has already been examined");
         if (alternatives.contains(s)) return;
         alternatives.add(s);
     }
@@ -64,6 +81,7 @@ public class Union extends Element implements Iterable<Sequence> {
         return "(anon_union)";
     }
     public String toString() {
+        viewed = true;
         if (name != null) return name;
         StringBuffer sb = new StringBuffer();
         sb.append("(");
@@ -71,7 +89,10 @@ public class Union extends Element implements Iterable<Sequence> {
         sb.append(")");
         return sb.toString();
     }
+
+    /** display this union in long/expanded form */
     public StringBuffer toString(StringBuffer sb) {
+        viewed = true;
         if (synthetic) return sb;
         boolean first = true;
         String before = StringUtil.pad(15, getName()) + " = ";
@@ -87,6 +108,7 @@ public class Union extends Element implements Iterable<Sequence> {
     }
     
     private void bodyToString(StringBuffer sb, String before, String between) {
+        viewed = true;
         boolean first = true;
         for(Sequence s : this) {
             // FIXME: what to do here about printing out negated sequences?