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 */