[SPIGOT-5240] Bukkit's Vector#angle(Vector) is broken Created: 02/Aug/19 Updated: 02/Aug/19 Resolved: 02/Aug/19 |
|
Status: | Resolved |
Project: | Spigot |
Component/s: | None |
Affects Version/s: | None |
Fix Version/s: | None |
Type: | Bug | Priority: | Minor |
Reporter: | Islandscout | Assignee: | Unassigned |
Resolution: | Fixed | Votes: | 0 |
Labels: | bukkit, util |
Version: | git-Spigot-cbcc8e8-06efc9e (MC: 1.14.4) (Implementing API version 1.14.4-R0.1-SNAPSHOT) |
Guidelines Read: | Yes |
Description |
It appears that for certain vectors, Vector#angle(Vector) returns NaN. This doesn't make sense since there is always an angle between two non-zero vectors which should always be 0 to 180 degrees inclusive. public float angle(@NotNull final Vector other) { final double dot = this.dot(other) / (this.length() * other.length()); return (float)Math.acos(dot); } public float angle(@NotNull final Vector other) { double dot = Math.min(Math.max(this.dot(other) / (this.length() * other.length()), -1), 1); return (float)Math.acos(dot); } |
Comments |
Comment by md_5 [ 02/Aug/19 ] |
> Isn't Double.NaN == Double.NaN always false?
Right, thanks |
Comment by Islandscout [ 02/Aug/19 ] |
Yeah, the normalization is done before the min/max. md_5, what is that assertNotEquals method? Isn't Double.NaN == Double.NaN always false? Wouldn't that mean doing Double.NaN == (a number that is NaN) will also always be false? Try doing Double.isNaN(a.angle(b)) I also want to mention that this fix will not break the condition where at least one of the vectors is zero, which should be one of the only few conditions that returns NaN. |
Comment by Brokkonaut [ 02/Aug/19 ] |
Vector a = new Vector(-0.13154885489775203, 0.0, 0.12210868381700482); Vector b = new Vector(-0.7329152226448059, -0.0, 0.6803199648857117); System.out.println(a.angle(b)); System.out.println(a.dot(b) / (a.length() * b.length())); Output:
NaN 1.0000000000000002 Afaik his fix should work for not normalized vectors too, because its divided by a.length() * b.length() (this is the normalization) before the min/max is applied.
|
Comment by md_5 [ 02/Aug/19 ] |
Your fix looks only to be valid for normalized vectors also. |
Comment by md_5 [ 02/Aug/19 ] |
@Test public void testSmallAngle() { Vector a = new Vector(-0.13154885489775203, 0.0, 0.12210868381700482); Vector b = new Vector(-0.7329152226448059, -0.0, 0.6803199648857117); assertNotEquals(Double.NaN, a.angle(b)); } Seems to work ok for me. Assuming you are also using an x86 64 bit processor this should most definitely not be hardware dependent. |
Comment by Islandscout [ 02/Aug/19 ] |
Try with this:
Vector a = new Vector(-0.13154885489775203,0.0,0.12210868381700482); Vector b = new Vector(-0.7329152226448059,-0.0,0.6803199648857117); Bukkit.broadcastMessage(a.angle(b)); It could be hardware dependent, but regardless of what hardware the code runs on, it should not be returning NaN (at least for most hardware). The value of "dot" is 1.0000000000000002, which is clearly not within [-1, 1] |
Comment by md_5 [ 02/Aug/19 ] |
Can you provide a test case? |