001/*
002 *  Copyright 2001-2010 Stephen Colebourne
003 *
004 *  Licensed under the Apache License, Version 2.0 (the "License");
005 *  you may not use this file except in compliance with the License.
006 *  You may obtain a copy of the License at
007 *
008 *      http://www.apache.org/licenses/LICENSE-2.0
009 *
010 *  Unless required by applicable law or agreed to in writing, software
011 *  distributed under the License is distributed on an "AS IS" BASIS,
012 *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013 *  See the License for the specific language governing permissions and
014 *  limitations under the License.
015 */
016package org.joda.primitives.collection.impl;
017
018import java.lang.reflect.Array;
019import java.util.Collection;
020import java.util.Iterator;
021
022import org.joda.primitives.BooleanUtils;
023import org.joda.primitives.collection.BooleanCollection;
024import org.joda.primitives.iterator.BooleanIterator;
025
026/**
027 * Abstract base class for collections of primitive <code>boolean</code> elements.
028 * <p>
029 * This class implements {@link java.util.Collection Collection} allowing
030 * seamless integration with other APIs.
031 * <p>
032 * The <code>iterator</code> and <code>size</code> must be implemented by subclases.
033 * To make the subclass modifiable, the <code>add(boolean)</code> and
034 * iterator <code>remove()</code> methods must also be implemented.
035 * Subclasses may override other methods to increase efficiency.
036 *
037 * @author Stephen Colebourne
038 * @author Jason Tiscione
039 * @version CODE GENERATED
040 * @since 1.0
041 */
042public abstract class AbstractBooleanCollection
043        extends AbstractPrimitiveCollectable<Boolean>
044        implements BooleanCollection {
045    // This file is CODE GENERATED. Do not change manually.
046
047    /**
048     * Constructor.
049     */
050    protected AbstractBooleanCollection() {
051        super();
052    }
053    
054    // Mandatory operations
055    //-----------------------------------------------------------------------
056    /**
057     * Checks whether this collection contains a specified primitive value.
058     * <p>
059     * This implementation uses <code>booleanIterator()</code>.
060     *
061     * @param value  the value to search for
062     * @return <code>true</code> if the value is found
063     */
064    public boolean contains(boolean value) {
065        for (BooleanIterator it = iterator(); it.hasNext();) {
066            if (it.nextBoolean() == value) {
067                return true;
068            }
069        }
070        return false;
071    }
072
073    /**
074     * Checks if this collection contains all of the values in the specified array.
075     * If the specified array is empty, <code>true</code> is returned.
076     * <p>
077     * This implementation uses <code>contains(boolean)</code>.
078     *
079     * @param values  the values to search for, null treated as empty array
080     * @return <code>true</code> if all of the values are found
081     */
082    public boolean containsAll(boolean[] values) {
083        if (values != null) {
084            for (int i = 0; i < values.length; i++) {
085                if (contains(values[i]) == false) {
086                    return false;
087                }
088            }
089        }
090        return true;
091    }
092
093    /**
094     * Checks if this collection contains all of the values in the specified collection.
095     * If the specified collection is empty, <code>true</code> is returned.
096     * <p>
097     * This implementation uses <code>contains(boolean)</code>.
098     *
099     * @param values  the values to search for, null treated as empty collection
100     * @return <code>true</code> if all of the values are found
101     */
102    public boolean containsAll(BooleanCollection values) {
103        if (values != null) {
104            for (BooleanIterator it = values.iterator(); it.hasNext(); ) {
105                if (contains(it.nextBoolean()) == false) {
106                    return false;
107                }
108            }
109        }
110        return true;
111    }
112
113    /**
114     * Checks if this collection contains any of the values in the specified array.
115     * If the specified array is empty, <code>false</code> is returned.
116     * <p>
117     * This implementation uses <code>contains(boolean)</code>.
118     *
119     * @param values  the values to search for, null treated as empty array
120     * @return <code>true</code> if at least one of the values is found
121     */
122    public boolean containsAny(boolean[] values) {
123        if (values != null) {
124            for (int i = 0; i < values.length; i++) {
125                if (contains(values[i])) {
126                    return true;
127                }
128            }
129        }
130        return false;
131    }
132
133    /**
134     * Checks if this collection contains any of the values in the specified collection.
135     * If the specified collection is empty, <code>false</code> is returned.
136     * <p>
137     * This implementation uses <code>contains(boolean)</code>.
138     *
139     * @param values  the values to search for, null treated as empty collection
140     * @return <code>true</code> if at least one of the values is found
141     */
142    public boolean containsAny(BooleanCollection values) {
143        if (values != null) {
144            for (BooleanIterator it = values.iterator(); it.hasNext(); ) {
145                if (contains(it.nextBoolean())) {
146                    return true;
147                }
148            }
149        }
150        return false;
151    }
152
153    /**
154     * Gets the elements of this collection as an array.
155     * <p>
156     * This implementation uses <code>arrayCopy</code>.
157     *
158     * @return a new array containing a copy of the elements of this collection
159     */
160    public boolean[] toBooleanArray() {
161        if (size() == 0) {
162            return BooleanUtils.EMPTY_BOOLEAN_ARRAY;
163        }
164        boolean[] result = new boolean[size()];
165        arrayCopy(0, result, 0, size());
166        return result;
167    }
168
169    /**
170     * Copies the elements of this collection into an array at a specified position.
171     * Previous values in the array are overwritten.
172     * <p>
173     * If the array specified is null a new array is created.
174     * If the array specified is large enough, it will be modified.
175     * If the array is not large enough, a new array will be created containing the
176     * values from the specified array before the startIndex plus those from this collection.
177     * <p>
178     * This implementation uses <code>arrayCopy</code>.
179     *
180     * @param array  the array to add the elements to, null treated as empty array
181     * @param startIndex  the position in the array to start setting elements
182     * @return the array with the populated collection
183     * @throws IndexOutOfBoundsException if the index is negative
184     */
185    public boolean[] toBooleanArray(boolean[] array, int startIndex) {
186        if (startIndex < 0) {
187            throw new IndexOutOfBoundsException("Start index must not be negative: " + startIndex);
188        }
189        boolean[] result = null;
190        if (array == null) {
191            // create new
192            result = new boolean[startIndex + size()];
193            
194        } else if (array.length - startIndex - size() >= 0) {
195            // room to fit data
196            result = array;
197            
198        } else {
199            // expand array
200            result = new boolean[startIndex + size()];
201            System.arraycopy(array, 0, result, 0, startIndex);
202        }
203        arrayCopy(0, result, startIndex, size());
204        return result;
205    }
206
207    // Optional operations
208    //-----------------------------------------------------------------------
209    /**
210     * Clears the collection/map of all elements (optional operation).
211     * <p>
212     * The collection/map will have a zero size after this method completes.
213     * This method is optional, throwing an UnsupportedOperationException if the
214     * collection/map cannot be cleared.
215     * <p>
216     * This implementation uses <code>iterator()</code>.
217     *
218     * @throws UnsupportedOperationException if method not supported by this collection
219     */
220    public void clear() {
221        checkRemoveModifiable();
222        for (BooleanIterator it = iterator(); it.hasNext();) {
223            it.nextBoolean();
224            it.remove();
225        }
226    }
227
228    /**
229     * Adds a primitive value to this collection (optional operation).
230     * <p>
231     * This implementation throws UnsupportedOperationException.
232     *
233     * @param value  the value to add to this collection
234     * @return <code>true</code> if this collection was modified by this method call
235     * @throws IllegalArgumentException if value is rejected by this collection
236     * @throws UnsupportedOperationException if not supported by this collection
237     */
238    public boolean add(boolean value) {
239        throw new UnsupportedOperationException("Collection does not support add");
240    }
241
242    /**
243     * Adds an array of primitive values to this collection (optional operation).
244     * <p>
245     * This implementation uses <code>add(boolean)</code>.
246     *
247     * @param values  the values to add to this collection, null treated as empty array
248     * @return <code>true</code> if this collection was modified by this method call
249     * @throws IllegalArgumentException if a value is rejected by this collection
250     * @throws UnsupportedOperationException if not supported by this collection
251     */
252    public boolean addAll(boolean[] values) {
253        checkAddModifiable();
254        boolean changed = false;
255        if (values != null) {
256            for (int i = 0; i < values.length; i++) {
257                changed |= add(values[i]);
258            }
259        }
260        return changed;
261    }
262
263    /**
264     * Adds a collection of primitive values to this collection (optional operation).
265     * <p>
266     * This implementation uses <code>add(boolean)</code>.
267     *
268     * @param values  the values to add to this collection, null treated as empty collection
269     * @return <code>true</code> if this collection was modified by this method call
270     * @throws IllegalArgumentException if a value is rejected by this collection
271     * @throws UnsupportedOperationException if not supported by this collection
272     */
273    public boolean addAll(BooleanCollection values) {
274        checkAddModifiable();
275        boolean changed = false;
276        if (values != null) {
277            for (BooleanIterator it = values.iterator(); it.hasNext(); ) {
278                changed |= add(it.nextBoolean());
279            }
280        }
281        return changed;
282    }
283
284    /**
285     * Removes the first occurrence of the specified primitive value from this collection
286     * <p>
287     * This implementation uses <code>iterator().remove()</code>.
288     *
289     * @param value  the value to remove
290     * @return <code>true</code> if this collection was modified by this method call
291     * @throws UnsupportedOperationException if not supported by this collection
292     */
293    public boolean removeFirst(boolean value) {
294        checkRemoveModifiable();
295        for (BooleanIterator it = iterator(); it.hasNext(); ) {
296            if (it.nextBoolean() == value) {
297                it.remove();
298                return true;
299            }
300        }
301        return false;
302    }
303
304    /**
305     * Removes all occurrences of the specified primitive value from this collection.
306     * <p>
307     * This implementation uses <code>iterator().remove()</code>.
308     *
309     * @param value  the value to remove
310     * @return <code>true</code> if this collection was modified by this method call
311     * @throws UnsupportedOperationException if not supported by this collection
312     */
313    public boolean removeAll(boolean value) {
314        checkRemoveModifiable();
315        boolean changed = false;
316        for (BooleanIterator it = iterator(); it.hasNext(); ) {
317            if (it.nextBoolean() == value) {
318                it.remove();
319                changed = true;
320            }
321        }
322        return changed;
323    }
324
325    /**
326     * Removes all occurrences from this collection of each primitive in the specified array.
327     * <p>
328     * This implementation uses <code>iterator().remove()</code>.
329     *
330     * @param values  the values to remove from this collection, null treated as empty array
331     * @return <code>true</code> if this list was modified by this method call
332     * @throws UnsupportedOperationException if not supported by this collection
333     */
334    public boolean removeAll(boolean[] values) {
335        checkRemoveModifiable();
336        boolean changed = false;
337        if (values != null) {
338            for (BooleanIterator it = iterator(); it.hasNext(); ) {
339                boolean value = it.nextBoolean();
340                for (int i = 0; i < values.length; i++) {
341                    if (values[i] == value) {
342                        it.remove();
343                        changed = true;
344                    }
345                }
346            }
347        }
348        return changed;
349    }
350
351    /**
352     * Removes all occurrences from this collection of each primitive in the specified collection.
353     * <p>
354     * This implementation uses <code>iterator().remove()</code>.
355     *
356     * @param values  the values to remove from this collection, null treated as empty collection
357     * @return <code>true</code> if this list was modified by this method call
358     * @throws UnsupportedOperationException if not supported by this collection
359     */
360    public boolean removeAll(BooleanCollection values) {
361        checkRemoveModifiable();
362        boolean changed = false;
363        if (values != null) {
364            for (BooleanIterator it = iterator(); it.hasNext(); ) {
365                if (values.contains(it.nextBoolean())) {
366                    it.remove();
367                    changed = true;
368                }
369            }
370        }
371        return changed;
372    }
373
374    /**
375     * Retains each element of this collection that is present in the specified array
376     * removing all other values.
377     * <p>
378     * This implementation uses <code>iterator().remove()</code>.
379     *
380     * @param values  the values to remove from this collection, null treated as empty array
381     * @return <code>true</code> if this list was modified by this method call
382     * @throws UnsupportedOperationException if not supported by this collection
383     */
384    public boolean retainAll(boolean[] values) {
385        checkRemoveModifiable();
386        boolean changed = false;
387        if (values == null || values.length == 0) {
388            changed = !isEmpty();
389            clear();
390        } else {
391            for (BooleanIterator it = iterator(); it.hasNext(); ) {
392                boolean next = it.nextBoolean();
393                boolean match = false;
394                for (int i = 0; i < values.length; i++) {
395                    if (values[i] == next) {
396                        match = true;
397                        break;
398                    }
399                }
400                if (match == false) {
401                    it.remove();
402                    changed = true;
403                }
404            }
405        }
406        return changed;
407    }
408
409    /**
410     * Retains each element of this collection that is present in the specified collection
411     * removing all other values.
412     * <p>
413     * This implementation uses <code>iterator().remove()</code>.
414     *
415     * @param values  the values to retain in this collection, null treated as empty collection
416     * @return <code>true</code> if this collection was modified by this method call
417     * @throws UnsupportedOperationException if not supported by this collection
418     */
419    public boolean retainAll(BooleanCollection values) {
420        checkRemoveModifiable();
421        boolean changed = false;
422        if (values == null || values.isEmpty()) {
423            changed = !isEmpty();
424            clear();
425        } else {
426            for (BooleanIterator it = iterator(); it.hasNext(); ) {
427                if (values.contains(it.nextBoolean()) == false) {
428                    it.remove();
429                    changed = true;
430                }
431            }
432        }
433        return changed;
434    }
435
436    // Collection integration
437    //-----------------------------------------------------------------------
438    /**
439     * Checks whether this collection contains a specified <code>Boolean</code> value.
440     * <p>
441     * This implementation uses <code>contains(boolean)</code>.
442     *
443     * @param value  the value to search for
444     * @return <code>true</code> if the value is found
445     */
446    public boolean contains(Object value) {
447        return contains(toPrimitive(value));
448    }
449
450    /**
451     * Checks if the collection contains all of the primitive values.
452     * <p>
453     * This implementation uses <code>containsAll(boolean[])</code>.
454     *
455     * @param coll  the collection of values to search for
456     * @return <code>true</code> if all the values are found
457     */
458    public boolean containsAll(Collection<?> coll) {
459        if (coll == this || coll.size() == 0) {
460            return true;
461        }
462        if (size() == 0) {
463            return false;
464        }
465        return containsAll(toPrimitiveArray(coll));
466    }
467
468    /**
469     * Checks if the collection contains any of the primitive values in the array.
470     * If the specified collection is empty, <code>false</code> is returned.
471     * <p>
472     * This implementation uses <code>containsAny(boolean[])</code>.
473     *
474     * @param coll  the collection of values to search for
475     * @return <code>true</code> if at least one of the values is found
476     */
477    public boolean containsAny(Collection<?> coll) {
478        if (size() == 0 || coll.size() == 0) {
479            return false;
480        }
481        if (coll == this) {
482            return true;
483        }
484        return containsAny(toPrimitiveArray(coll));
485    }
486
487    /**
488     * Gets the collection as an array of <code>Boolean</code>.
489     * 
490     * @return an array of <code>Boolean</code>
491     */
492    public Object[] toArray() {
493        Object[] result = new Boolean[size()];
494        BooleanIterator it = iterator();
495        for (int i = 0; it.hasNext(); i++) {
496            result[i] = it.next();
497        }
498        return result;
499    }
500
501    /**
502     * Gets the collection as an array, using the array provided.
503     * 
504     * @param array  the array to populate
505     * @return an array of <code>Boolean</code>
506     */
507    @SuppressWarnings("unchecked")
508    public <T> T[] toArray(T[] array) {
509        int size = size();
510        if (array.length < size) {
511          array = (T[]) Array.newInstance(array.getClass().getComponentType(), size);
512        }
513
514        Iterator<Boolean> it = iterator();
515        for (int i = 0; i < size; i++) {
516            array[i] = (T)it.next();
517        }
518
519        if (array.length > size) {
520            array[size] = null;
521        }
522        
523        return array;
524    }
525
526    /**
527     * Adds the <code>Boolean</code> value to this collection (optional operation).
528     * <p>
529     * This method is optional, throwing an UnsupportedOperationException if the
530     * collection cannot be added to.
531     * <p>
532     * This implementation uses <code>add(boolean)</code>.
533     *
534     * @param value  the value to add to this collection
535     * @return <code>true</code> if this collection was modified by this method call
536     * @throws IllegalArgumentException if value is rejected by this collection
537     * @throws UnsupportedOperationException if not supported by this collection
538     */
539    public boolean add(Boolean value) {
540        checkAddModifiable();
541        return add(toPrimitive(value));
542    }
543
544    /**
545     * Adds a collection of <code>Boolean</code> values to this collection (optional operation).
546     * <p>
547     * This method is optional, throwing an UnsupportedOperationException if the
548     * collection cannot be added to.
549     * <p>
550     * This implementation uses <code>addAll(boolean[])</code>.
551     *
552     * @param coll  the values to add to this collection
553     * @return <code>true</code> if this list was modified by this method call
554     * @throws IndexOutOfBoundsException if the index is invalid
555     * @throws ClassCastException if any object is not <code>Boolean</code>
556     * @throws IllegalArgumentException if value is rejected by this collection
557     * @throws UnsupportedOperationException if not supported by this collection
558     */
559    public boolean addAll(Collection<? extends Boolean> coll) {
560        checkAddModifiable();
561        return addAll(toPrimitiveArray(coll));
562    }
563
564    /**
565     * Removes the first occurrance of the specified <code>Boolean</code> value from
566     * this collection (optional operation).
567     * <p>
568     * This method is optional, throwing an UnsupportedOperationException if the
569     * collection cannot be removed from.
570     * <p>
571     * This implementation uses <code>removeFirst(boolean)</code>.
572     *
573     * @param value  the value to remove
574     * @return <code>true</code> if this collection was modified by this method call
575     * @throws UnsupportedOperationException if not supported by this collection
576     */
577    public boolean remove(Object value) {
578        checkRemoveModifiable();
579        return removeFirst(toPrimitive(value));
580    }
581
582    /**
583     * Removes each of a collection of <code>Boolean</code> values from this collection (optional operation).
584     * <p>
585     * This method is optional, throwing an UnsupportedOperationException if the
586     * collection cannot be added to.
587     * <p>
588     * This implementation uses <code>removeAll(boolean[])</code>.
589     *
590     * @param coll  the values to remove from this collection
591     * @return <code>true</code> if this list was modified by this method call
592     * @throws UnsupportedOperationException if not supported by this collection
593     */
594    public boolean removeAll(Collection<?> coll) {
595        checkRemoveModifiable();
596        if (coll == this) {
597            int size = size();
598            clear();
599            return (size() != size);
600        }
601        return removeAll(toPrimitiveArray(coll));
602    }
603
604    /**
605     * Retains each of a collection of <code>Boolean</code> values, removing other
606     * values (optional operation).
607     * <p>
608     * This method is optional, throwing an UnsupportedOperationException if the
609     * collection cannot be added to.
610     * <p>
611     * This implementation uses <code>retainAll(boolean[])</code>.
612     *
613     * @param coll  the values to retain in this collection
614     * @return <code>true</code> if this list was modified by this method call
615     * @throws UnsupportedOperationException if not supported by this collection
616     */
617    public boolean retainAll(Collection<?> coll) {
618        checkRemoveModifiable();
619        if (coll == this) {
620            return false;
621        }
622        return retainAll(toPrimitiveArray(coll));
623    }
624
625    // Basics
626    //-----------------------------------------------------------------------
627    /**
628     * Gets a string representing this collection.
629     * <p>
630     * The format used is as per <code>Collection</code>.
631     * 
632     * @return collection as a String
633     */
634    public String toString() {
635        StringBuffer buf = new StringBuffer();
636        buf.append("[");
637
638        BooleanIterator it = iterator();
639        boolean hasNext = it.hasNext();
640        while (hasNext) {
641            buf.append(it.nextBoolean());
642            hasNext = it.hasNext();
643            if (hasNext) {
644                buf.append(", ");
645            }
646        }
647
648        buf.append("]");
649        return buf.toString();
650    }
651    
652    // Internals
653    //-----------------------------------------------------------------------
654    /**
655     * Copies data from this collection into the specified array.
656     * This method is pre-validated.
657     * 
658     * @param fromIndex  the index to start from
659     * @param dest  the destination array
660     * @param destIndex  the destination start index
661     * @param size  the number of items to copy
662     */
663    protected void arrayCopy(int fromIndex, boolean[] dest, int destIndex, int size) {
664        BooleanIterator it = iterator();
665        for (int i = 0; it.hasNext() && i < size; i++) {
666            dest[destIndex + i] = it.nextBoolean();
667        }
668    }
669
670    /**
671     * Are the add methods supported.
672     * <p>
673     * This implementation returns false.
674     *
675     * @return true if supported
676     */
677    protected boolean isAddModifiable() {
678        return false;
679    }
680
681    /**
682     * Are the remove methods supported.
683     * <p>
684     * This implementation returns false.
685     *
686     * @return true if supported
687     */
688    protected boolean isRemoveModifiable() {
689        return false;
690    }
691
692    /**
693     * Is the collection modifiable in any way.
694     *
695     * @return true if supported
696     */
697    public boolean isModifiable() {
698        return isAddModifiable() || isRemoveModifiable();
699    }
700
701    /**
702     * Check whether add is suported and throw an exception.
703     */
704    protected void checkAddModifiable() {
705        if (isAddModifiable() == false) {
706            throw new UnsupportedOperationException("Collection does not support add");
707        }
708    }
709
710    /**
711     * Check whether remove is suported and throw an exception.
712     */
713    protected void checkRemoveModifiable() {
714        if (isRemoveModifiable() == false) {
715            throw new UnsupportedOperationException("Collection does not support remove");
716        }
717    }
718
719    /**
720     * Wraps an <code>boolean</code> with an Object wrapper.
721     * 
722     * @param value  the primitive value
723     * @return the Object wrapper
724     */
725    protected Boolean toObject(boolean value) {
726        return BooleanUtils.toObject(value);
727    }
728
729    /**
730     * Checks if the object can be converted to a primitive successfully.
731     * <p>
732     * This implementation only allows non-null Boolean objects.
733     * 
734     * @param value  the Object wrapper
735     * @return true if a primitive value can be successfully extracted
736     */
737    protected boolean isToPrimitivePossible(Object value) {
738        return (value instanceof Boolean);
739    }
740
741    /**
742     * Unwraps the <code>Boolean</code> to retrieve the primitive <code>boolean</code>.
743     * <p>
744     * This implementation only allows non-null Boolean objects.
745     * 
746     * @param value  the Object to convert to a primitive
747     * @return the primitive value
748     * @throws NullPointerException if the value is null and this is unacceptable
749     * @throws ClassCastException if the object is of an unsuitable type
750     */
751    protected boolean toPrimitive(Object value) {
752        return BooleanUtils.toPrimitive(value);
753    }
754
755    /**
756     * Unwraps a <code>Collection</code> to retrieve the primitive <code>boolean</code>.
757     * <p>
758     * This implementation only allows non-null Boolean objects.
759     * 
760     * @param coll  the Collection to convert to primitives
761     * @return the primitive value
762     * @throws NullPointerException if the value is null and this is unacceptable
763     * @throws ClassCastException if any object is of an unsuitable type
764     */
765    protected boolean[] toPrimitiveArray(Collection<?> coll) {
766        return BooleanUtils.toPrimitiveArray(coll);
767    }
768
769}