001/*
002 * ============================================================================
003 * Copyright © 2014 by Dominic Fox.
004 * All Rights Reserved.
005 * ============================================================================
006 * The MIT License (MIT)
007 *
008 * Permission is hereby granted, free of charge, to any person obtaining a copy
009 * of this software and associated documentation files (the "Software"), to
010 * deal in the Software without restriction, including without limitation the
011 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
012 * sell copies of the Software, and to permit persons to whom the Software is
013 * furnished to do so, subject to the following conditions:
014 *
015 * The above copyright notice and this permission notice shall be included in
016 * all copies or substantial portions of the Software.
017 *
018 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
019 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
020 * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE
021 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
022 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
023 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
024 * IN THE SOFTWARE.
025 */
026
027package org.tquadrat.foundation.stream.internal;
028
029import org.apiguardian.api.API;
030import org.tquadrat.foundation.annotation.ClassVersion;
031
032import java.util.Spliterator;
033import java.util.function.Consumer;
034import java.util.function.Predicate;
035
036import static org.apiguardian.api.API.Status.INTERNAL;
037import static org.tquadrat.foundation.lang.Objects.requireNonNullArgument;
038
039/**
040 *  An implementation of
041 *  {@link Spliterator}
042 *  that skips elements on the stream that do not match the provided condition.
043 *
044 *  @author Dominic Fox
045 *  @modified Thomas Thrien - thomas.thrien@tquadrat.org
046 *  @version $Id: SkipUntilSpliterator.java 1119 2024-03-16 09:03:57Z tquadrat $
047 *  @since 0.0.7
048 *
049 *  @param  <T> The type over which the given streams stream.
050 *
051 *  @UMLGraph.link
052 */
053@ClassVersion( sourceVersion = "$Id: SkipUntilSpliterator.java 1119 2024-03-16 09:03:57Z tquadrat $" )
054@API( status = INTERNAL, since = "0.0.7" )
055public final class SkipUntilSpliterator<T> implements Spliterator<T>
056{
057        /*------------*\
058    ====** Attributes **=======================================================
059        \*------------*/
060    /**
061     *  The condition.
062     */
063    private final Predicate<T> m_Condition;
064
065    /**
066     *  The flag that indicates whether the condition was met or not.
067     */
068    private boolean m_ConditionMet = false;
069
070    /**
071     *  The source.
072     */
073    private final Spliterator<T> m_Source;
074
075        /*--------------*\
076    ====** Constructors **=====================================================
077        \*--------------*/
078    /**
079     *  Creates a new {@code SkipUntilSpliterator} instance.
080     *
081     *  @param  source  The source stream.
082     *  @param  condition   The condition to apply to elements of the source
083     *      stream.
084     */
085    private SkipUntilSpliterator( final Spliterator<T> source, final Predicate<T> condition )
086    {
087        m_Source = requireNonNullArgument( source, "source" );
088        m_Condition = requireNonNullArgument( condition, "condition" );
089    }   //  SkipUntilSpliterator()
090
091        /*---------*\
092    ====** Methods **==========================================================
093        \*---------*/
094    /**
095     *  Factory method for instances of {@code SkipUntilSpliterator}-
096     *
097     *  @param  <T> The type of elements for the source spliterators.
098     *  @param  source  The source stream.
099     *  @param  condition   The condition to apply to elements of the source
100     *      stream.
101     *  @return The instance.
102     */
103    public static final <T> SkipUntilSpliterator<T> over( final Spliterator<T> source, final Predicate<T> condition )
104    {
105        final var retValue = new SkipUntilSpliterator<>( source, condition );
106
107        //---* Done *----------------------------------------------------------
108        return retValue;
109    }   //  over()
110
111    /**
112     *  {@inheritDoc}
113     */
114    @Override
115    public final int characteristics()
116    {
117        @SuppressWarnings( "ConstantExpression" )
118        final var retValue = m_Source.characteristics() & ~SIZED;
119
120        //---* Done *----------------------------------------------------------
121        return retValue;
122    }   //  characteristics()
123
124    /**
125     *  {@inheritDoc}
126     */
127    @Override
128    public final long estimateSize() { return m_ConditionMet ? m_Source.estimateSize() : Long.MAX_VALUE; }
129
130    /**
131     *  {@inheritDoc}
132     */
133    @Override
134    public final void forEachRemaining( final Consumer<? super T> action )
135    {
136        if( !m_ConditionMet )
137        {
138            tryAdvance( action );
139        }
140        if( m_ConditionMet )
141        {
142            m_Source.forEachRemaining( action );
143        }
144    }   //  forEachRemaining()
145
146    /**
147     *  {@inheritDoc}
148     */
149    @Override
150    public final boolean tryAdvance( final Consumer<? super T> action )
151    {
152        final boolean retValue;
153        if( m_ConditionMet )
154        {
155            retValue = m_Source.tryAdvance( action );
156        }
157        else
158        {
159            while( !m_ConditionMet && m_Source.tryAdvance( e ->
160            {
161                //noinspection NestedAssignment,PointlessBooleanExpression
162                if( (m_ConditionMet = m_Condition.test( e )) == true )
163                {
164                    action.accept( e );
165                }
166            }) )
167            { /* Does nothing! */ }
168            retValue = m_ConditionMet;
169        }
170
171        //---* Done *----------------------------------------------------------
172        return retValue;
173    }   //  tryAdvance()
174
175    /**
176     *  {@inheritDoc}
177     */
178    @Override
179    public final Spliterator<T> trySplit() { return null; }
180}
181//  class SkipUntilSpliterator
182
183/*
184 *  End of File
185 */