touchless3dtracking/Color/color.py

160 lines
5.1 KiB
Python
Executable File

#!/bin/env python3
# ============================================================================
# This script maps the 3D position of your hand, as detected by the electrodes
# in the RGB space, allowing you to pick a color for real.
# As all the other scripts in this repository, I release it under a very
# permissive license. To make a long story short : do whatever you want with
# this script (but try to have fun :), I don't mind. It would be cool to quote
# the origin of the script if you reuse it, but you don't have to. I'd like to
# be noticed of what you did cool with it (if you think it's worth). :)
# Ah, I almost forgot : If by chance we ever meet and you think this script is
# worth, you can buy me a soda :)
#
# Phyks
# =============================================================================
import serial
import pygame
import sys
import getopt
def compute_value(line, minimum, maximum):
if len(line) < 3:
return False
value = [0, 0, 0]
for i in range(3):
if maximum[i]-minimum[i] != 0:
value[i] = int(255*(line[i]-minimum[i])/(maximum[i]-minimum[i]))
else:
value[i] = 0
if value[i] > 255:
value[i] = 255
elif value[i] < 0:
value[i] = 0
return value
serial_port = "/dev/ttyACM0"
try:
opts, args = getopt.getopt(sys.argv[1:], "hs:", ["help", "serial="])
for opt, arg in opts:
if opt in ("-h", "--help"):
print("Touchless 3D tracking with color mapping")
print("\nUsage : "+sys.argv[0]+" [OPTIONS]")
print("\nTrack the position of your hand in 3D and map it " +
"in RGB space")
print("\nOptions :")
print("\t-h (--help) \t display this help message")
print("\t-s (--serial=) \t change serial port (default is " +
"/dev/tty/ACM0")
sys.exit(0)
elif opt in ("-s", "--serial"):
serial_port = arg
except getopt.GetoptError:
pass
ser = serial.Serial(serial_port, 115200)
pygame.init()
# Keep old value to determine a mean value
value = [[0, 0, 0], [0, 0, 0]]
font = pygame.font.Font(None, 36)
size = width, height = 640, 480
screen = pygame.display.set_mode(size)
pygame.display.set_caption("Touchless 3D tracking")
screen.fill((0, 0, 0))
try:
ser.open()
except Exception as e:
print("Error while opening serial port : "+str(e))
if ser.isOpen():
try:
ser.flushInput()
ser.flushOutput()
maximum = [-1, -1, -1]
minimum = [-1, -1, -1]
print("Calibration :")
print("Press any key to launch the program when calibration is " +
"finished.")
# Display info in window
label = font.render("Calibration...", 1, (255, 255, 255))
label_pos = label.get_rect()
label_pos.centerx = screen.get_rect().centerx
label_pos.centery = 20
screen.blit(label, label_pos)
pygame.display.flip()
running = True
while running:
for event in pygame.event.get():
if event.type == pygame.KEYDOWN:
running = False
continue
line = ser.readline()
line = line.decode().strip("\r\n")
line = line.split(" ")
line = [int(j or 0) for j in line]
for i in range(3):
if line[i] < minimum[i] or minimum[i] < 0:
minimum[i] = line[i]
if line[i] > maximum[i]:
maximum[i] = line[i]
print(line)
print("Running...")
running = True
while running:
# Quit if window is closed
for event in pygame.event.get():
if event.type == pygame.QUIT or event.type == pygame.KEYDOWN:
pygame.quit()
running = False
continue
# Read line from serial
line = ser.readline()
line = line.decode().strip("\r\n")
line = line.split(" ")
line = [int(j or 0) for j in line]
# Compute the value for red
value[1] = value[0]
value[0] = compute_value(line, minimum, maximum)
value_bg = [(value[0][i] + value[1][i])/2 for i in range(3)]
value_text = [255 - i for i in compute_value(line, minimum,
maximum)]
if value is not False:
screen.fill((value_bg[0], value_bg[1], value_bg[2]))
# Display info in window
label = font.render("Running...", 1, (value_text[0],
value_text[1], value_text[2]))
label_pos = label.get_rect()
label_pos.centerx = screen.get_rect().centerx
label_pos.centery = 20
screen.blit(label, label_pos)
pygame.display.flip()
ser.close()
except Exception as e:
print("Error while fetching data from serial : "+str(e))