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

org.elasticsearch.index.search.geo.GeoPolygonQuery Maven / Gradle / Ivy

There is a newer version: 8.15.1
Show newest version
/*
 * Licensed to Elasticsearch under one or more contributor
 * license agreements. See the NOTICE file distributed with
 * this work for additional information regarding copyright
 * ownership. Elasticsearch 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.elasticsearch.index.search.geo;

import org.apache.lucene.index.LeafReaderContext;
import org.apache.lucene.search.IndexSearcher;
import org.apache.lucene.search.Query;
import org.apache.lucene.search.RandomAccessWeight;
import org.apache.lucene.search.Weight;
import org.apache.lucene.util.Bits;
import org.elasticsearch.common.geo.GeoPoint;
import org.elasticsearch.index.fielddata.IndexGeoPointFieldData;
import org.elasticsearch.index.fielddata.MultiGeoPointValues;

import java.io.IOException;
import java.util.Arrays;

/**
 *
 */
public class GeoPolygonQuery extends Query {

    private final GeoPoint[] points;

    private final IndexGeoPointFieldData indexFieldData;

    public GeoPolygonQuery(IndexGeoPointFieldData indexFieldData, GeoPoint...points) {
        this.points = points;
        this.indexFieldData = indexFieldData;
    }

    public GeoPoint[] points() {
        return points;
    }

    public String fieldName() {
        return indexFieldData.getFieldNames().indexName();
    }

    @Override
    public Weight createWeight(IndexSearcher searcher, boolean needsScores) throws IOException {
        return new RandomAccessWeight(this) {
            @Override
            protected Bits getMatchingDocs(LeafReaderContext context) throws IOException {
                final int maxDoc = context.reader().maxDoc();
                final MultiGeoPointValues values = indexFieldData.load(context).getGeoPointValues();
                return new Bits() {

                    private boolean pointInPolygon(GeoPoint[] points, double lat, double lon) {
                        boolean inPoly = false;

                        for (int i = 1; i < points.length; i++) {
                            if (points[i].lon() < lon && points[i-1].lon() >= lon
                                    || points[i-1].lon() < lon && points[i].lon() >= lon) {
                                if (points[i].lat() + (lon - points[i].lon()) /
                                        (points[i-1].lon() - points[i].lon()) * (points[i-1].lat() - points[i].lat()) < lat) {
                                    inPoly = !inPoly;
                                }
                            }
                        }
                        return inPoly;
                    }

                    @Override
                    public boolean get(int doc) {
                        values.setDocument(doc);
                        final int length = values.count();
                        for (int i = 0; i < length; i++) {
                            GeoPoint point = values.valueAt(i);
                            if (pointInPolygon(points, point.lat(), point.lon())) {
                                return true;
                            }
                        }
                        return false;
                    }

                    @Override
                    public int length() {
                        return maxDoc;
                    }

                };
            }
        };
    }

    @Override
    public String toString(String field) {
        StringBuilder sb = new StringBuilder("GeoPolygonQuery(");
        sb.append(indexFieldData.getFieldNames().indexName());
        sb.append(", ").append(Arrays.toString(points)).append(')');
        return sb.toString();
    }

    @Override
    public boolean equals(Object obj) {
        if (super.equals(obj) == false) {
            return false;
        }
        GeoPolygonQuery that = (GeoPolygonQuery) obj;
        return indexFieldData.getFieldNames().indexName().equals(that.indexFieldData.getFieldNames().indexName())
                && Arrays.equals(points, that.points);
    }

    @Override
    public int hashCode() {
        int h = super.hashCode();
        h = 31 * h + indexFieldData.getFieldNames().indexName().hashCode();
        h = 31 * h + Arrays.hashCode(points);
        return h;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy