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

org.davidmoten.hilbert.Box Maven / Gradle / Ivy

package org.davidmoten.hilbert;

import java.util.Arrays;
import java.util.function.Consumer;

import com.github.davidmoten.guavamini.Preconditions;
import com.github.davidmoten.guavamini.annotations.VisibleForTesting;

final class Box {

    final long[] a;
    final long[] b;

    Box(long[] a, long[] b) {
        Preconditions.checkArgument(a.length == b.length);
        this.a = a;
        this.b = b;
    }

    int dimensions() {
        return a.length;
    }

    @Override
    public String toString() {
        return "Box [" + Arrays.toString(a) + ", " + Arrays.toString(b) + "]";
    }

    void visitCells(Consumer visitor) {
        long[] mins = mins(a, b);
        long[] maxes = maxes(a, b);
        long[] x = Arrays.copyOf(mins, mins.length);
        while (true) {
            visitor.accept(x);
            if (equals(x, maxes)) {
                break;
            } else {
                addOne(x, mins, maxes);
            }
        }
    }

    void visitPerimeter(Consumer visitor) {
        long[] mins = mins(a, b);
        long[] maxes = maxes(a, b);
        for (int specialIndex = dimensions() - 1; specialIndex >= 0; specialIndex--) {
            long[] x = Arrays.copyOf(mins, mins.length);
            // visit for the minimum at specialIndex
            visitPerimeter(mins, maxes, x, specialIndex, visitor);
            if (mins[specialIndex] != maxes[specialIndex]) {
                // visit for the maximum at specialIndex
                long[] y = Arrays.copyOf(mins, mins.length);
                y[specialIndex] = maxes[specialIndex];
                visitPerimeter(mins, maxes, y, specialIndex, visitor);
            } else {
                break;
            }
        }
    }

    @VisibleForTesting
    static void visitPerimeter(long[] mins, long[] maxes, long[] x, int specialIndex,
            Consumer visitor) {
        long[] y = Arrays.copyOf(x, x.length);
        for (int i = specialIndex + 1; i < y.length; i++) {
            if (mins[i] >= maxes[i] - 1) {
                return;
            }
            y[i] = mins[i] + 1;
        }
        visitor.accept(y);
        while (true) {
            // try to increment once
            for (int i = y.length - 1; i >= 0; i--) {
                if (i > specialIndex) {
                    // to the right of specialIndex we only allow values between min + 1 and max -1
                    // inclusive
                    if (y[i] == maxes[i] - 1) {
                        y[i] = mins[i] + 1;
                        // continue looping to increment at the next index to the left
                    } else {
                        // increment happened without carryover so we break and report y
                        y[i] += 1;
                        break;
                    }
                } else if (i < specialIndex) {
                    // to the left of specialIndex we allow all values
                    if (y[i] == maxes[i]) {
                        if (i == 0) {
                            return;
                        } else {
                            y[i] = mins[i];
                        }
                    } else {
                        y[i] += 1;
                        break;
                    }
                } else if (i == specialIndex && i == 0) {
                    return;
                }
            }
            visitor.accept(y);
        }
    }

    @VisibleForTesting
    static void addOne(long[] x, long[] mins, long[] maxes) {
        for (int i = x.length - 1; i >= 0; i--) {
            if (x[i] != maxes[i]) {
                x[i]++;
                break;
            } else {
                x[i] = mins[i];
            }
        }
    }

    @VisibleForTesting
    static boolean equals(long[] a, long[] b) {
        for (int i = 0; i < a.length; i++) {
            if (a[i] != b[i]) {
                return false;
            }
        }
        return true;
    }

    private static long[] mins(long[] a, long[] b) {
        long[] c = new long[a.length];
        for (int i = 0; i < a.length; i++) {
            c[i] = Math.min(a[i], b[i]);
        }
        return c;
    }

    private static long[] maxes(long[] a, long[] b) {
        long[] c = new long[a.length];
        for (int i = 0; i < a.length; i++) {
            c[i] = Math.max(a[i], b[i]);
        }
        return c;
    }

    boolean contains(long[] point) {
        Preconditions.checkArgument(a.length == point.length);
        for (int i = 0; i < a.length; i++) {
            if (point[i] < Math.min(a[i], b[i]) || point[i] > Math.max(a[i], b[i])) {
                return false;
            }
        }
        return true;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy