Files
kick-it/pykick/colorpicker.py

237 lines
7.2 KiB
Python

from __future__ import print_function
from __future__ import division
import json
import argparse
import cv2
from .imagereaders import VideoReader, NaoImageReader, PictureReader
from .finders import GoalFinder, BallFinder, FieldFinder
from .utils import read_config, imresize, hsv_mask
class Colorpicker(object):
WINDOW_CAPTURE_NAME = 'Object Detection (or not)'
WINDOW_DETECTION_NAME = 'Primary Mask'
def __init__(self, target=None):
parameters = ['low_h', 'low_s', 'low_v', 'high_h', 'high_s', 'high_v']
maxes = [180, 255, 255, 180, 255, 255]
checkers = [
lambda x: x, # LOW H
# lambda x: min(x, self.settings['high_h'] - 1), # LOW H
lambda x: min(x, self.settings['high_s'] - 1), # LOW S
lambda x: min(x, self.settings['high_v'] - 1), # LOW V
lambda x: x, # HIGH H
# lambda x: max(x, self.settings['low_h'] + 1), # HIGH H
lambda x: max(x, self.settings['low_s'] + 1), # HIGH S
lambda x: max(x, self.settings['low_v'] + 1), # HIGH V
]
self.settings = {
'low_h': 0,
'low_s': 0,
'low_v': 0,
'high_h': 180,
'high_s': 255,
'high_v': 255
}
if target == 'goal':
Marker = GoalFinder
elif target == 'ball':
Marker = BallFinder
elif target == 'field':
Marker = FieldFinder
else:
Marker = None
if Marker is not None:
self.marker = Marker(
tuple(map(self.settings.get, ('low_h', 'low_s', 'low_v'))),
tuple(map(self.settings.get, ('high_h', 'high_s', 'high_v')))
)
else:
self.marker = None
cv2.namedWindow(self.WINDOW_CAPTURE_NAME)
cv2.namedWindow(self.WINDOW_DETECTION_NAME)
self.trackers = [
cv2.createTrackbar(
name, self.WINDOW_DETECTION_NAME, self.settings[name], max_v,
lambda val, name=name, checker=checker: self._on_trackbar(
val, name, checker
)
)
for name, max_v, checker in zip(parameters, maxes, checkers)
]
def do_print(self):
print(tuple(map(self.settings.get, ('low_h', 'low_s', 'low_v'))),
tuple(map(self.settings.get, ('high_h', 'high_s', 'high_v'))))
def _on_trackbar(self, val, name, checker):
self.settings[name] = checker(val)
self._hsv_updated(name)
def _hsv_updated(self, param):
cv2.setTrackbarPos(param, self.WINDOW_DETECTION_NAME,
self.settings[param])
self.marker.hsv_lower = tuple(
map(self.settings.get, ('low_h', 'low_s', 'low_v'))
)
self.marker.hsv_upper = tuple(
map(self.settings.get, ('high_h', 'high_s', 'high_v'))
)
def show_frame(self, frame, width=None, manual=False):
frame = imresize(frame, width=width)
if self.marker is not None:
thr = self.marker.primary_mask(frame)
stuff = self.marker.find(frame)
frame = self.marker.draw(frame, stuff)
else:
hsv = cv2.cvtColor(frame, cv2.COLOR_BGR2HSV)
thr = hsv_mask(
hsv,
tuple(map(self.settings.get, ('low_h', 'low_s', 'low_v'))),
tuple(map(self.settings.get, ('high_h', 'high_s', 'high_v')))
)
cv2.imshow(self.WINDOW_CAPTURE_NAME, frame)
cv2.imshow(self.WINDOW_DETECTION_NAME, thr)
return cv2.waitKey(0 if manual else 1)
def save(self, filename, color):
try:
with open(filename) as f:
conf = json.load(f)
except IOError:
conf = {}
conf.update(
{color: [
list(map(self.settings.get, ['low_h', 'low_s', 'low_v'])),
list(map(self.settings.get, ['high_h', 'high_s', 'high_v']))
]}
)
with open(filename, 'w') as f:
json.dump(conf, f, indent=4)
def load(self, filename, color):
with open(filename) as f:
jdict = json.load(f)
self.settings = dict(
zip(['low_h', 'low_s', 'low_v', 'high_h', 'high_s', 'high_v'],
jdict[color][0] + jdict[color][1])
)
for name in self.settings:
self._hsv_updated(name)
if __name__ == '__main__':
defaults = read_config()
parser = argparse.ArgumentParser(
epilog='When called without arguments specifying the video source, ' +
'will try to use the webcam'
)
parser.add_argument(
'-o', '--output-config',
help='file, to which the settings will be saved (if given)'
)
parser.add_argument(
'-i', '--input-config',
help='file, from which to read the initial values'
)
imsource = parser.add_mutually_exclusive_group()
imsource.add_argument(
'--video-file',
help='video file to use'
)
imsource.add_argument(
'--image-file',
help='image to use'
)
imsource.add_argument(
'--nao-ip',
help='ip address of the nao robot, from which to capture',
default=False,
const=defaults['ip'],
nargs='?'
)
parser.add_argument(
'--still',
help='only take one image from video stream',
action='store_true'
)
parser.add_argument(
'--manual',
help='switch frames manually',
action='store_true',
default=False
)
parser.add_argument(
'--nao-cam',
choices=[0, 1],
type=int,
help='0 for top camera, 1 for bottom camera',
default=defaults['cam']
)
parser.add_argument(
'--nao-res',
choices=[1, 2, 3],
help='choose a nao resolution',
type=int,
default=defaults['res']
)
parser.add_argument(
'--width',
help='specify width of the image output',
type=int,
default=640
)
parser.add_argument(
'--target',
help='specify for what target is being calibrated',
default=None
)
args = parser.parse_args()
cp = Colorpicker(args.target)
if args.input_config:
cp.load(args.input_config, args.target)
if args.video_file:
rdr = VideoReader(args.video_file, loop=True)
elif args.image_file:
rdr = PictureReader(args.image_file)
elif args.nao_ip:
rdr = NaoImageReader(
args.nao_ip,
cam_id=args.nao_cam,
res=args.nao_res
)
else:
rdr = VideoReader(0)
try:
if args.still:
frame = rdr.get_frame()
rdr.close()
while True:
if not args.still:
try:
frame = rdr.get_frame()
except RuntimeError as e:
print(e)
continue
key = cp.show_frame(frame, width=args.width, manual=args.manual)
if key == ord('q') or key == 27:
break
finally:
cp.do_print()
if args.output_config:
cp.save(args.output_config, args.target)
if not args.still:
rdr.close()