nearest.py 2.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879
  1. #!/usr/bin/env python3
  2. import csv
  3. import math
  4. import random
  5. from argparse import ArgumentParser
  6. from convert import rgb_to_cieluv
  7. def norm(r, g, b):
  8. return math.sqrt(r * r + g * g + b * b)
  9. parser = ArgumentParser()
  10. parser.add_argument(
  11. "color", nargs="?", default=None, help="Target color, randomized if not provided"
  12. )
  13. parser.add_argument(
  14. "-n", "--number", default=1, type=int, help="Number of Pokemon to find"
  15. )
  16. parser.add_argument(
  17. "-c", "--closeness", default=2, type=int, help="Closeness coefficient"
  18. )
  19. parser.add_argument("-d", "--database", default="database.csv", help="Database file")
  20. parser.add_argument("-x", "--exclude-x", action="store_true", help="Exclude X")
  21. parser.add_argument("-z", "--normalize", action="store_true", help="Normalize q and Y")
  22. parser.add_argument(
  23. "-l",
  24. "--convert-luv",
  25. action="store_true",
  26. help="Convert input color to CIELUV before calculation",
  27. )
  28. parser.add_argument("-v", "--verbose", action="store_true", help="Print raw scores")
  29. args = parser.parse_args()
  30. if args.number <= 0:
  31. raise ValueError("Must request a number greater than 0")
  32. if args.color is not None:
  33. cleaned_color = args.color.strip("#")
  34. if len(cleaned_color) != 6:
  35. raise ValueError("Color must be a 6 digit hex")
  36. color = (
  37. int(cleaned_color[0:2], base=16),
  38. int(cleaned_color[2:4], base=16),
  39. int(cleaned_color[4:6], base=16),
  40. )
  41. else:
  42. color = tuple(int(random.random() * 255) for _ in range(3))
  43. print(f"Generated random color: #{''.join(hex(c)[2:] for c in color)} / {color}")
  44. if args.convert_luv:
  45. color = rgb_to_cieluv(*color)
  46. yfactor = args.closeness
  47. if args.normalize:
  48. yfactor /= norm(*color)
  49. results = []
  50. with open(args.database) as infile:
  51. for name, x, *y in csv.reader(infile, delimiter=",", quotechar="'"):
  52. xval = 0 if args.exclude_x else float(x)
  53. yvec = [float(y_c) for y_c in y]
  54. yval = sum(y_c * c for y_c, c in zip(yvec, color))
  55. if args.normalize:
  56. yval /= norm(*yvec)
  57. score = xval - yfactor * yval
  58. results.append((score, name))
  59. if args.number > 1:
  60. for i, (score, name) in enumerate(sorted(results)[: args.number]):
  61. print(f"{i+1}:\t{name}")
  62. if args.verbose:
  63. print(f"\t\t\t{score=}")
  64. else:
  65. best_score, best_name = min(results)
  66. print(f"Best match: {best_name}")
  67. if args.verbose:
  68. print(f"{score=}")