Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

color.equals() should not rely on object equality #641

Open
LeaVerou opened this issue Mar 13, 2025 · 3 comments
Open

color.equals() should not rely on object equality #641

LeaVerou opened this issue Mar 13, 2025 · 3 comments
Labels
bug Something isn't working

Comments

@LeaVerou
Copy link
Member

I just spent a while debugging some pretty hairy issue where using equals() on two equal colors produced false. Turns out our equals() implementation uses object equality, which fails in a number of cases.

Some ideas for how to address this:

  1. Change how equals() works: convert both colors to the same color space if they are different. We can have a separate strictEquals() for what it does now.
  2. If checking color spaces fails, compare space ids. Or just compare space ids to begin with (will fail with anonymous spaces, though it's unclear whether we support this well right now)
  3. If the current check fails, serialize the colors and compare that (making sure to override any default formats) (issue: slower)
  4. Implement a ColorSpace.equals() and defer to that.
@LeaVerou LeaVerou added the bug Something isn't working label Mar 13, 2025
@kleinfreund
Copy link
Contributor

I wanted to mention a use case because I just came across this issue:

I didn't know about color.equals() and a piece of code that I'm currently working on uses colorA.distance(colorB) === 0 && colorA.alpha === colorB.alpha instead. Now aside from this potentially not being appropriate (I don't know, all my existing tests pass so introducing it seems to work on a surface level glance), dropping in colorA.equals(colorB) doesn't work on first try (my tests start to fail in a few places).

Some colors I want to consider as equal in my use case then turn up as different even though they have the same coords and alpha when compared in the same space. This could potentially be related/caused by this issue.

@LeaVerou
Copy link
Member Author

I wanted to mention a use case because I just came across this issue:

I didn't know about color.equals() and a piece of code that I'm currently working on uses colorA.distance(colorB) === 0 && colorA.alpha === colorB.alpha instead. Now aside from this potentially not being appropriate (I don't know, all my existing tests pass so introducing it seems to work on a surface level glance), dropping in colorA.equals(colorB) doesn't work on first try (my tests start to fail in a few places).

Some colors I want to consider as equal in my use case then turn up as different even though they have the same coords and alpha when compared in the same space. This could potentially be related/caused by this issue.

If space conversion is involved, I'd use Math.abs(distance) <= Number.EPSILON rather than comparing with 0 to account for precision errors.

@kleinfreund
Copy link
Contributor

If space conversion is involved, I'd use Math.abs(distance) <= Number.EPSILON rather than comparing with 0 to account for precision errors.

That's likely a good call, yes. Thanks!

My use case is a color picker and the application for the color comparison is to avoid firing a change event when the color hasn't changed. So if a precision error would get through the 0 comparison, it wouldn't be too bad and maybe some would expect the change event to fire in these cases. Still, I lean toward using a comparison that accounts for precision errors, too.

(Feel free to mark this as off-topic if you want to keep this issue more focused)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working
Projects
None yet
Development

No branches or pull requests

2 participants