imager.py 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. from PIL import Image
  2. import os
  3. import subprocess
  4. ON = "X" # Dark pixel in an image
  5. OFF = "." # Light pixel in an image
  6. INTENSITY = 75
  7. # How bright does something have to be to trigger it being a dark or light pixel?
  8. # Looks like around 75 removes the extra stuff that s0urce.io does to prevent it from being just matching images.
  9. GREEN_DIFF = 10
  10. # How much brighter the green channel must be (compared to the others),
  11. # to be called green.
  12. def img_point(pix, x, y):
  13. """
  14. img_point, returns a pixel of an image,
  15. given the x and y on the image.
  16. """
  17. return pix[x, y]
  18. def img_avg(pix, x, y):
  19. """
  20. img_avg, returns the average brightness 0-255,
  21. given pixel, and the x and y on the image calls img_point,
  22. to get the individual rgb values to calculate,
  23. brightness. (Grey scale)
  24. """
  25. rgb = img_point(pix, x, y)
  26. # if(im.mode == 'P'):
  27. # rgb = pal[rgb*3:(rgb+1)*3]
  28. # if(im.mode == 'I'):
  29. # return rgb >> 8
  30. return int((rgb[0] + rgb[1] + rgb[2]) / 3)
  31. def is_set(pix, x, y):
  32. global INTENSITY
  33. """
  34. is_set, returns True or False of calculating,
  35. the brightness of the given point on a image,
  36. compared to given intensity.
  37. True means the brightness at the given x and y,
  38. is Less Than which means its dark.
  39. False means the brightness at the given x and y,
  40. is Greater Than which means its bright. (Grey Scale)
  41. """
  42. avg = img_avg(pix, x, y)
  43. return avg < INTENSITY
  44. def is_green(image, x, y):
  45. """
  46. Is this pixel Green?
  47. """
  48. (red, green, blue, _) = image.getpixel((x, y))
  49. # Find the difference between green and the other values.
  50. other = red
  51. if blue > other:
  52. other = blue
  53. diff = green - other
  54. return diff > GREEN_DIFF
  55. def scan_img(image):
  56. """
  57. Given an image, look at a image and find dark pixels,
  58. also find the bounding box size.
  59. (Where's the top,left,right,bottom-most pixels at?)
  60. returns start x, y and end x, y and total number of dark pixels.
  61. """
  62. total = 0
  63. size = image.size
  64. sx = size[0]
  65. ex = 0
  66. sy = size[1]
  67. ey = 0
  68. for y in range(0, size[1]):
  69. for x in range(0, size[0]):
  70. pnt_is = is_set(pix, x, y)
  71. if pnt_is:
  72. total += 1
  73. if x < sx:
  74. sx = x
  75. if x > ex:
  76. ex = x
  77. if y < sy:
  78. sy = y
  79. if y > ey:
  80. ey = y
  81. # print (sx,ex,sy,ey)
  82. # give us a little border to work with
  83. if sx > 0:
  84. sx -= 1
  85. if ex < size[0]:
  86. ex += 1
  87. if sy > 0:
  88. sy -= 1
  89. if ey < size[1]:
  90. ey += 1
  91. return (sx, sy, ex, ey, total)
  92. def output_image(image):
  93. """
  94. For the size of the area we have reduced down to where the majority of dark pixels,
  95. are located, store all that into a list and return the list.
  96. given pixel for function passing.
  97. returns multiple strings in a list that are edited to use characters to represent,
  98. the dark and light pixels of the image. (Grey Scale)
  99. """
  100. size = image.size
  101. for y in range(0, size[1]):
  102. s = ""
  103. for x in range(0, size[0]):
  104. # if is_set(pix, x, y):
  105. if not is_green(image, x, y):
  106. s += ON
  107. else:
  108. s += OFF
  109. print(s)
  110. def image_cleaner(source, destination, padfile=True):
  111. """ Given a source image, clean it and save to destination
  112. If padfile is set, we use imagemagik to add additional padding
  113. around the image, which helps with the conversion process.
  114. """
  115. image = Image.open(source)
  116. size = image.size
  117. for y in range(0, size[1]):
  118. for x in range(0, size[0]):
  119. (r, g, b, _) = image.getpixel( (x,y) )
  120. high = r
  121. if b > high:
  122. high = b
  123. diff = g - high
  124. is_green = diff > 0 # GREEN_DIFF
  125. if is_green:
  126. image.putpixel( (x,y), (255,255,255,255) )
  127. else:
  128. image.putpixel( (x,y), (0, 0, 0, 255) )
  129. if padfile:
  130. tmpfile = "temp.png"
  131. image.save(tmpfile)
  132. output = subprocess.run(
  133. ["convert", tmpfile, "-bordercolor", "White",
  134. "-border", "10x10", "-alpha", "off", destination],
  135. stderr=subprocess.DEVNULL,
  136. # capture_output=False,
  137. shell=False,
  138. )
  139. os.unlink(tmpfile)
  140. else:
  141. image.save(destination)