BoundarySourceLinecaster2D.java
- /*
- * Licensed to the Apache Software Foundation (ASF) under one or more
- * contributor license agreements. See the NOTICE file distributed with
- * this work for additional information regarding copyright ownership.
- * The ASF licenses this file to You under the Apache License, Version 2.0
- * (the "License"); you may not use this file except in compliance with
- * the License. You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- package org.apache.commons.geometry.euclidean.twod;
- import java.util.ArrayList;
- import java.util.List;
- import java.util.Objects;
- import java.util.stream.Collectors;
- import java.util.stream.Stream;
- /** Class that performs linecast operations against arbitrary {@link BoundarySource2D}
- * instances. This class performs a brute-force computation of the intersections of the
- * line or line subset against all boundaries. Some data structures may support more
- * efficient algorithms and should therefore prefer those instead.
- */
- final class BoundarySourceLinecaster2D implements Linecastable2D {
- /** The boundary source instance providing boundaries for the linecast operation. */
- private final BoundarySource2D boundarySrc;
- /** Construct a new instance for linecasting against the given boundary source.
- * @param boundarySrc boundary source to linecast against.
- */
- BoundarySourceLinecaster2D(final BoundarySource2D boundarySrc) {
- this.boundarySrc = boundarySrc;
- }
- /** {@inheritDoc} */
- @Override
- public List<LinecastPoint2D> linecast(final LineConvexSubset subset) {
- try (Stream<LinecastPoint2D> stream = getIntersectionStream(subset)) {
- final List<LinecastPoint2D> results = stream.collect(Collectors.toCollection(ArrayList::new));
- LinecastPoint2D.sortAndFilter(results);
- return results;
- }
- }
- /** {@inheritDoc} */
- @Override
- public LinecastPoint2D linecastFirst(final LineConvexSubset subset) {
- try (Stream<LinecastPoint2D> stream = getIntersectionStream(subset)) {
- return stream.min(LinecastPoint2D.ABSCISSA_ORDER)
- .orElse(null);
- }
- }
- /** Return a stream containing intersections between the boundary source and the
- * given line subset.
- * @param subset line subset to intersect
- * @return a stream containing linecast intersections
- */
- private Stream<LinecastPoint2D> getIntersectionStream(final LineConvexSubset subset) {
- return boundarySrc.boundaryStream()
- .map(boundary -> computeIntersection(boundary, subset))
- .filter(Objects::nonNull);
- }
- /** Compute the intersection between a boundary line subset and linecast intersecting line subset. Null is
- * returned if no intersection is discovered.
- * @param boundary boundary from the boundary source
- * @param subset line subset to intersect with
- * @return the linecast intersection between the two arguments or null if there is no such
- * intersection
- */
- private LinecastPoint2D computeIntersection(final LineConvexSubset boundary, final LineConvexSubset subset) {
- final Vector2D intersectionPt = boundary.intersection(subset);
- if (intersectionPt != null) {
- final Vector2D normal = boundary.getLine().getOffsetDirection();
- return new LinecastPoint2D(intersectionPt, normal, subset.getLine());
- }
- return null; // no intersection
- }
- }