001/*
002 *                    BioJava development code
003 *
004 * This code may be freely distributed and modified under the
005 * terms of the GNU Lesser General Public Licence.  This should
006 * be distributed with the code.  If you do not have a copy,
007 * see:
008 *
009 *      http://www.gnu.org/copyleft/lesser.html
010 *
011 * Copyright for this code is held jointly by the individual
012 * authors.  These should be listed in @author doc comments.
013 *
014 * For more information on the BioJava project and its aims,
015 * or to join the biojava-l mailing list, visit the home page
016 * at:
017 *
018 *      http://www.biojava.org/
019 *
020 */
021package org.biojava.nbio.structure.symmetry.geometry;
022
023import javax.vecmath.Point3d;
024import java.util.*;
025
026/**
027 *
028 * @author Peter
029 */
030public class DistanceBox<T> {
031        private Map<Long, List<T>> map;
032        private Map<Long, List<T>> layerMap;
033        private boolean modified;
034        private double inverseBinWidth;
035        // relative locations of 26 boxes surrounding a box at (0, 0, 0)
036        private static final long[] offset = new long[] {
037                0 + ( 0 * 10000) + ( 0 * 1000000000L),
038           -1 + (-1 * 10000) + (-1 * 1000000000L),
039           -1 + (-1 * 10000) + ( 0 * 1000000000L),
040           -1 + (-1 * 10000) + ( 1 * 1000000000L),
041           -1 + ( 0 * 10000) + (-1 * 1000000000L),
042           -1 + ( 0 * 10000) + ( 0 * 1000000000L),
043           -1 + ( 0 * 10000) + ( 1 * 1000000000L),
044           -1 + ( 1 * 10000) + (-1 * 1000000000L),
045           -1 + ( 1 * 10000) + ( 0 * 1000000000L),
046           -1 + ( 1 * 10000) + ( 1 * 1000000000L),
047                0 + (-1 * 10000) + (-1 * 1000000000L),
048                0 + (-1 * 10000) + ( 0 * 1000000000L),
049                0 + (-1 * 10000) + ( 1 * 1000000000L),
050                0 + ( 0 * 10000) + (-1 * 1000000000L),
051                0 + ( 0 * 10000) + ( 1 * 1000000000L),
052                0 + ( 1 * 10000) + (-1 * 1000000000L),
053                0 + ( 1 * 10000) + ( 0 * 1000000000L),
054                0 + ( 1 * 10000) + ( 1 * 1000000000L),
055                1 + (-1 * 10000) + (-1 * 1000000000L),
056                1 + (-1 * 10000) + ( 0 * 1000000000L),
057                1 + (-1 * 10000) + ( 1 * 1000000000L),
058                1 + ( 0 * 10000) + (-1 * 1000000000L),
059                1 + ( 0 * 10000) + ( 0 * 1000000000L),
060                1 + ( 0 * 10000) + ( 1 * 1000000000L),
061                1 + ( 1 * 10000) + (-1 * 1000000000L),
062                1 + ( 1 * 10000) + ( 0 * 1000000000L),
063                1 + ( 1 * 10000) + ( 1 * 1000000000L)
064        };
065
066        private List<T> tempBox = new ArrayList<>(offset.length);
067
068        /** Creates a new instance of DistanceBox */
069        public DistanceBox(double binWidth) {
070                map = new HashMap<>();
071                layerMap = new HashMap<>();
072                this.inverseBinWidth = 1.0f/binWidth;
073                this.modified = true;
074        }
075
076        public void addPoint(Point3d point, T object) {
077                long i = (long) StrictMath.rint(point.x * inverseBinWidth);
078                long j = (long) StrictMath.rint(point.y * inverseBinWidth);
079                long k = (long) StrictMath.rint(point.z * inverseBinWidth);
080                long location = i + (j * 10000L) + (k * 1000000000L);
081
082                List<T> box = map.get(location);
083
084                if (box == null) {
085                        box = new ArrayList<>();
086                        map.put(location, box);
087                }
088
089                box.add(object);
090                modified = true;
091        }
092
093        public List<T> getNeighborsWithCache(Point3d point) {
094                if (modified) {
095                        layerMap.clear();
096                        modified = false;
097                }
098
099                long i = (long) StrictMath.rint(point.x * inverseBinWidth);
100                long j = (long) StrictMath.rint(point.y * inverseBinWidth);
101                long k = (long) StrictMath.rint(point.z * inverseBinWidth);
102                long location = i + (j * 10000L) + (k * 1000000000L);
103
104                List<T> box = layerMap.get(location);
105
106                if (box == null) {
107                        box = getBoxTwo(location);
108                        layerMap.put(location, box);
109                }
110
111                return box;
112        }
113
114        public List<T> getNeighbors(Point3d point) {
115                if (modified) {
116                        layerMap.clear();
117                        modified = false;
118                }
119
120                long i = (long) StrictMath.rint(point.x * inverseBinWidth);
121                long j = (long) StrictMath.rint(point.y * inverseBinWidth);
122                long k = (long) StrictMath.rint(point.z * inverseBinWidth);
123                long location = i + (j * 10000L) + (k * 1000000000L);
124
125                List<T> box = getBoxTwo(location);
126                return box;
127        }
128
129        public List<T> getIntersection(DistanceBox<T> distanceBox) {
130                List<T> intersection = new ArrayList<>();
131                HashSet<Long> checkedLocations = new HashSet<>();
132
133                for (Iterator<Long> iter = map.keySet().iterator(); iter.hasNext();) {
134                        long location = iter.next();
135                        boolean overlap = false;
136                        for (int i = 0, n = offset.length; i < n; i++) {
137                                long loc = location + offset[i];
138                                if (distanceBox.contains(loc)) {
139                                        overlap = true;
140                                        break;
141                                }
142                        }
143                        if (overlap) {
144                                for (int i = 0, n = offset.length; i < n; i++) {
145                                        long loc = location + offset[i];
146                                        if (checkedLocations.contains(loc))
147                                                continue;
148
149                                        checkedLocations.add(loc);
150                                        if (contains(loc)) {
151                                                intersection.addAll(map.get(loc));
152                                        }
153                                }
154                        }
155                }
156                return intersection;
157        }
158
159        private List<T> getBoxTwo(long location) {
160                tempBox.clear();
161                for (int i = 0, n = offset.length; i < n; i++) {
162                        List<T> box = map.get(location + offset[i]);
163                        if (box != null) {
164                                tempBox.addAll(box);
165                        }
166                }
167                // ensure that boxTwo has no empty element by copying from tempBox of defined size
168                List<T> boxTwo = null;
169                if (tempBox.size() == 0) {
170                        boxTwo = Collections.emptyList();
171                } else if (tempBox.size() == 1) {
172                        boxTwo = Collections.singletonList(tempBox.get(0));
173                } else {
174                        boxTwo = new ArrayList<>(tempBox);
175                }
176                return boxTwo;
177        }
178
179        private boolean contains(long location) {
180                return map.containsKey(location);
181        }
182}