All Downloads are FREE. Search and download functionalities are using the official Maven repository.

functionalj.stream.doublestream.DoubleStreamPlusWithLimit Maven / Gradle / Ivy

There is a newer version: 1.0.17
Show newest version
// ============================================================================
// Copyright (c) 2017-2021 Nawapunth Manusitthipol (NawaMan - http://nawaman.net).
// ----------------------------------------------------------------------------
// MIT License
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
// ============================================================================
package functionalj.stream.doublestream;

import static functionalj.stream.doublestream.DoubleStreamPlusHelper.sequential;

import java.util.Spliterators;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.function.DoubleConsumer;
import java.util.function.DoublePredicate;
import java.util.stream.DoubleStream;
import java.util.stream.StreamSupport;

import functionalj.function.DoubleDoublePredicatePrimitive;
import functionalj.function.aggregator.DoubleAggregationToBoolean;
import functionalj.stream.markers.Sequential;
import lombok.val;


public interface DoubleStreamPlusWithLimit {
    
    public DoubleStreamPlus doubleStreamPlus();
    
    /** Limit the size of the stream to the given size. */
    public default DoubleStreamPlus limit(Long maxSize) {
        val streamPlus = doubleStreamPlus();
        return ((maxSize == null) || (maxSize.longValue() < 0))
                ? streamPlus
                : streamPlus
                    .limit((long)maxSize);
    }
    
    /** Skip to the given offset position. */
    public default DoubleStreamPlus skip(Long offset) {
        val streamPlus = doubleStreamPlus();
        return ((offset == null) || (offset.longValue() < 0))
                ? streamPlus
                : streamPlus
                    .skip((long)offset);
    }
    
    /** Skip any value while the condition is true. */
    @Sequential
    public default DoubleStreamPlus skipWhile(DoublePredicate condition) {
        val streamPlus = doubleStreamPlus();
        return sequential(streamPlus, stream -> {
            val isStillTrue = new AtomicBoolean(true);
            return stream.filter(e -> {
                if (!isStillTrue.get())
                    return true;
                
                if (!condition.test(e))
                    isStillTrue.set(false);
                
                return !isStillTrue.get();
            });
        });
    }
    
    /** Skip any value while the condition is true. */
    @Sequential
    public default DoubleStreamPlus skipWhile(DoubleAggregationToBoolean aggregationCondition) {
        val condition = aggregationCondition.newAggregator();
        return skipWhile(condition);
    }
    
    /** Skip any value while the condition is true. */
    @Sequential
    public default DoubleStreamPlus skipWhile(DoubleDoublePredicatePrimitive condition) {
        val streamPlus = doubleStreamPlus();
        return sequential(streamPlus, stream -> {
            val orgSpliterator = stream.spliterator();
            val newSpliterator = new Spliterators.AbstractDoubleSpliterator(orgSpliterator.estimateSize(), 0) {
                boolean isStillSkipping = true;
                boolean isFirst         = true;
                double  prevValue       = Double.NaN;
                @Override
                public boolean tryAdvance(DoubleConsumer consumer) {
                    DoubleConsumer action = elem -> {
                        if (isStillSkipping) {
                            if (!isFirst) {
                                if (condition.test(prevValue, elem)) {
                                    isStillSkipping = false;
                                }
                            } else {
                                isFirst = false;
                            }
                            if (!isStillSkipping) {
                                consumer.accept(prevValue);
                            }
                            prevValue = elem;
                        } else {
                            consumer.accept(prevValue);
                            prevValue = elem;
                        }
                    };
                    boolean hadNext = orgSpliterator.tryAdvance(action);
                    if (!isStillSkipping && !hadNext) {
                        consumer.accept(prevValue);
                    }
                    return hadNext;
                }
            };
            val newStream = StreamSupport.doubleStream(newSpliterator, false);
            return DoubleStreamPlus.from(newStream);
        });
    }
    
    /** Skip any value until the condition is true. */
    @Sequential
    public default DoubleStreamPlus skipUntil(DoublePredicate condition) {
        val streamPlus = doubleStreamPlus();
        return sequential(streamPlus, stream -> {
            val isStillTrue = new AtomicBoolean(true);
            return stream.filter(e -> {
                if (!isStillTrue.get())
                    return true;
                
                if (condition.test(e))
                    isStillTrue.set(false);
                
                return !isStillTrue.get();
            });
        });
    }
    
    /** Skip any value until the condition is true. */
    @Sequential
    public default DoubleStreamPlus skipUntil(DoubleAggregationToBoolean aggregationCondition) {
        val condition = aggregationCondition.newAggregator();
        return skipUntil(condition);
    }
    
    /** Skip any value until the condition is true. */
    @Sequential
    public default DoubleStreamPlus skipUntil(DoubleDoublePredicatePrimitive condition) {
        val streamPlus = doubleStreamPlus();
        return sequential(streamPlus, stream -> {
            val orgSpliterator = stream.spliterator();
            val newSpliterator = new Spliterators.AbstractDoubleSpliterator(orgSpliterator.estimateSize(), 0) {
                boolean isStillSkipping = true;
                boolean isFirst         = true;
                double  prevValue       = Integer.MIN_VALUE;
                @Override
                public boolean tryAdvance(DoubleConsumer consumer) {
                    DoubleConsumer action = elem -> {
                        if (isStillSkipping) {
                            if (!isFirst) {
                                if (!condition.test(prevValue, elem)) {
                                    isStillSkipping = false;
                                }
                            } else {
                                isFirst = false;
                            }
                            if (!isStillSkipping) {
                                consumer.accept(prevValue);
                            }
                            prevValue = elem;
                        } else {
                            consumer.accept(prevValue);
                            prevValue = elem;
                        }
                    };
                    boolean hadNext = orgSpliterator.tryAdvance(action);
                    if (!isStillSkipping && !hadNext) {
                        consumer.accept(prevValue);
                    }
                    return hadNext;
                }
            };
            val newStream = StreamSupport.doubleStream(newSpliterator, false);
            return DoubleStreamPlus.from(newStream);
        });
    }
    
    /** Accept any value while the condition is true. */
    @Sequential
    public default DoubleStreamPlus takeWhile(DoublePredicate condition) {
        val streamPlus = doubleStreamPlus();
        return sequential(streamPlus, stream -> {
            val orgSpliterator = stream.spliterator();
            val newSpliterator = new Spliterators.AbstractDoubleSpliterator(orgSpliterator.estimateSize(), 0) {
                boolean stillGoing = true;
                @Override
                public boolean tryAdvance(DoubleConsumer consumer) {
                    if (stillGoing) {
                        DoubleConsumer action = elem -> {
                            if (condition.test(elem)) {
                                consumer.accept(elem);
                            } else {
                                stillGoing = false;
                            }
                        };
                        boolean hadNext = orgSpliterator.tryAdvance(action);
                        return hadNext && stillGoing;
                    }
                    return false;
                }
            };
            val newStream = StreamSupport.doubleStream(newSpliterator, false);
            return DoubleStreamPlus.from(newStream);
        });
    }
    
    /** Accept any value while the condition is true. */
    @Sequential
    public default DoubleStreamPlus takeWhile(DoubleAggregationToBoolean aggregationCondition) {
        val condition = aggregationCondition.newAggregator();
        return takeWhile(condition);
    }
    
    /** Accept any value while the condition is true. */
    @Sequential
    public default DoubleStreamPlus takeWhile(DoubleDoublePredicatePrimitive condition) {
        val streamPlus = doubleStreamPlus();
        return sequential(streamPlus, stream -> {
            val orgSpliterator = stream.spliterator();
            val newSpliterator = new Spliterators.AbstractDoubleSpliterator(orgSpliterator.estimateSize(), 0) {
                boolean stillGoing = true;
                boolean isFirst    = true;
                double  prevValue  = Double.NaN;
                @Override
                public boolean tryAdvance(DoubleConsumer consumer) {
                    if (stillGoing) {
                        DoubleConsumer action = elem -> {
                            if (!isFirst) {
                                if (condition.test(prevValue, elem)) {
                                    consumer.accept(elem);
                                } else {
                                    stillGoing = false;
                                }
                            } else {
                                consumer.accept(elem);
                                isFirst = false;
                            }
                            prevValue = elem;
                        };
                        boolean hadNext = orgSpliterator.tryAdvance(action);
                        return hadNext && stillGoing;
                    }
                    return false;
                }
            };
            val newStream = StreamSupport.doubleStream(newSpliterator, false);
            return DoubleStreamPlus.from(newStream);
        });
    }
    
    /** Accept any value until the condition is true. */
    @Sequential
    public default DoubleStreamPlus takeUntil(DoublePredicate condition) {
        val streamPlus = doubleStreamPlus();
        return sequential(streamPlus, stream -> {
            val orgSpliterator = stream.spliterator();
            val newSpliterator = new Spliterators.AbstractDoubleSpliterator(orgSpliterator.estimateSize(), 0) {
                boolean stillGoing = true;
                @Override
                public boolean tryAdvance(DoubleConsumer consumer) {
                    if (stillGoing) {
                        DoubleConsumer action = elem -> {
                            if (!condition.test(elem)) {
                                consumer.accept(elem);
                            } else {
                                stillGoing = false;
                            }
                        };
                        boolean hadNext = orgSpliterator.tryAdvance(action);
                        return hadNext && stillGoing;
                    }
                    return false;
                }
            };
            val newStream = StreamSupport.doubleStream(newSpliterator, false);
            return DoubleStreamPlus.from(newStream);
        });
    }
    
    /** Accept any value until the condition is true. */
    @Sequential
    public default DoubleStreamPlus takeUntil(DoubleAggregationToBoolean aggregationCondition) {
        val condition = aggregationCondition.newAggregator();
        return takeWhile(condition);
    }
    
    /** Accept any value until the condition is true. */
    @Sequential
    public default DoubleStreamPlus takeUntil(DoubleDoublePredicatePrimitive condition) {
        val streamPlus = doubleStreamPlus();
        return sequential(streamPlus, stream -> {
            val orgSpliterator = stream.spliterator();
            val newSpliterator = new Spliterators.AbstractDoubleSpliterator(orgSpliterator.estimateSize(), 0) {
                boolean stillGoing = true;
                boolean isFirst    = true;
                double  prevValue  = -1;
                @Override
                public boolean tryAdvance(DoubleConsumer consumer) {
                    if (stillGoing) {
                        DoubleConsumer action = elem -> {
                            if (!isFirst) {
                                if (!condition.test(prevValue, elem)) {
                                    consumer.accept(elem);
                                } else {
                                    stillGoing = false;
                                }
                            } else {
                                consumer.accept(elem);
                                isFirst = false;
                            }
                            prevValue = elem;
                        };
                        boolean hadNext = orgSpliterator.tryAdvance(action);
                        return hadNext && stillGoing;
                    }
                    return false;
                }
            };
            val newStream = StreamSupport.doubleStream(newSpliterator, false);
            return DoubleStreamPlus.from(newStream);
        });
    }
    
    /** Accept any value until the condition is false - include the item that the condition is false. */
    @Sequential
    public default DoubleStreamPlus dropAfter(DoublePredicate condition) {
        val streamPlus = doubleStreamPlus();
        return sequential(streamPlus, stream -> {
            val orgSpliterator = stream.spliterator();
            val newSpliterator = new Spliterators.AbstractDoubleSpliterator(orgSpliterator.estimateSize(), 0) {
                boolean stillGoing = true;
                
                @Override
                public boolean tryAdvance(DoubleConsumer consumer) {
                    if (stillGoing) {
                        DoubleConsumer action = elem -> {
                            consumer.accept(elem);
                            if (condition.test(elem)) {
                                stillGoing = false;
                            }
                        };
                        boolean hadNext = orgSpliterator.tryAdvance(action);
                        return hadNext && stillGoing;
                    }
                    return false;
                }
            };
            val newStream = StreamSupport.doubleStream(newSpliterator, false);
            return DoubleStreamPlus.from(newStream);
        });
    }
    
    /** Accept any value until the condition is false - include the item that the condition is false. */
    @Sequential
    public default DoubleStreamPlus dropAfter(DoubleAggregationToBoolean aggregationCondition) {
        val condition = aggregationCondition.newAggregator();
        return dropAfter(condition);
    }
    
    /** Accept any value until the condition is false - include the item that the condition is false. */
    @Sequential
    public default DoubleStreamPlus dropAfter(DoubleDoublePredicatePrimitive condition) {
        val streamPlus = doubleStreamPlus();
        return sequential(streamPlus, stream -> {
            val orgSpliterator = stream.spliterator();
            val newSpliterator = new Spliterators.AbstractDoubleSpliterator(orgSpliterator.estimateSize(), 0) {
                boolean stillGoing = true;
                boolean isFirst    = true;
                double  prevValue  = -1;
                @Override
                public boolean tryAdvance(DoubleConsumer consumer) {
                    if (stillGoing) {
                        DoubleConsumer action = elem -> {
                            if (!isFirst) {
                                consumer.accept(elem);
                                if (condition.test(prevValue, elem)) {
                                    stillGoing = false;
                                }
                            } else {
                                consumer.accept(elem);
                                isFirst = false;
                            }
                            prevValue = elem;
                        };
                        boolean hadNext = orgSpliterator.tryAdvance(action);
                        return hadNext && stillGoing;
                    }
                    return false;
                }
            };
            DoubleStream newStream = StreamSupport.doubleStream(newSpliterator, false);
            return DoubleStreamPlus.from(newStream);
        });
    }
    
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy