Skip to content

Instantly share code, notes, and snippets.

@ashish1405
Created August 22, 2017 10:02
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save ashish1405/8d0aa798da478a839ecdf23cee146838 to your computer and use it in GitHub Desktop.
Save ashish1405/8d0aa798da478a839ecdf23cee146838 to your computer and use it in GitHub Desktop.
#!/usr/bin/python
import os
import sys
import json
import math
import cv2
import numpy as np
import copy
import yaml
from argparse import ArgumentParser
parser = ArgumentParser(description='OpenALPR License Plate Cropper')
parser.add_argument( "--input_dir", dest="input_dir", action="store", type=str, required=True,
help="Directory containing plate images and yaml metadata" )
parser.add_argument( "--out_dir", dest="out_dir", action="store", type=str, required=True,
help="Directory to output cropped plates" )
parser.add_argument( "--zoom_out_percent", dest="zoom_out_percent", action="store", type=float, default=1.25,
help="Percent multiplier to zoom out before cropping" )
parser.add_argument( "--plate_width", dest="plate_width", action="store", type=float, required=True,
help="Desired aspect ratio width" )
parser.add_argument( "--plate_height", dest="plate_height", action="store", type=float, required=True,
help="Desired aspect ratio height" )
options = parser.parse_args()
if not os.path.isdir(options.input_dir):
print "input_dir (%s) doesn't exist"
sys.exit(1)
if not os.path.isdir(options.out_dir):
os.makedirs(options.out_dir)
def order_points(pts):
# initialzie a list of coordinates that will be ordered
# such that the first entry in the list is the top-left,
# the second entry is the top-right, the third is the
# bottom-right, and the fourth is the bottom-left
rect = np.zeros((4, 2), dtype = "float32")
# the top-left point will have the smallest sum, whereas
# the bottom-right point will have the largest sum
s = pts.sum(axis = 1)
rect[0] = pts[np.argmin(s)]
rect[2] = pts[np.argmax(s)]
# now, compute the difference between the points, the
# top-right point will have the smallest difference,
# whereas the bottom-left will have the largest difference
diff = np.diff(pts, axis = 1)
rect[1] = pts[np.argmin(diff)]
rect[3] = pts[np.argmax(diff)]
# return the ordered coordinates
return rect
def four_point_transform(image, pts):
# obtain a consistent order of the points and unpack them
# individually
rect = order_points(pts)
(tl, tr, br, bl) = rect
# compute the width of the new image, which will be the
# maximum distance between bottom-right and bottom-left
# x-coordiates or the top-right and top-left x-coordinates
widthA = np.sqrt(((br[0] - bl[0]) ** 2) + ((br[1] - bl[1]) ** 2))
widthB = np.sqrt(((tr[0] - tl[0]) ** 2) + ((tr[1] - tl[1]) ** 2))
maxWidth = max(int(widthA), int(widthB))
# compute the height of the new image, which will be the
# maximum distance between the top-right and bottom-right
# y-coordinates or the top-left and bottom-left y-coordinates
heightA = np.sqrt(((tr[0] - br[0]) ** 2) + ((tr[1] - br[1]) ** 2))
heightB = np.sqrt(((tl[0] - bl[0]) ** 2) + ((tl[1] - bl[1]) ** 2))
maxHeight = max(int(heightA), int(heightB))
# now that we have the dimensions of the new image, construct
# the set of destination points to obtain a "birds eye view",
# (i.e. top-down view) of the image, again specifying points
# in the top-left, top-right, bottom-right, and bottom-left
# order
dst = np.array([
[0, 0],
[maxWidth - 1, 0],
[maxWidth - 1, maxHeight - 1],
[0, maxHeight - 1]], dtype = "float32")
# compute the perspective transform matrix and then apply it
M = cv2.getPerspectiveTransform(rect, dst)
warped = cv2.warpPerspective(image, M, (maxWidth, maxHeight))
# return the warped image
return warped
def get_box(x1, y1, x2, y2, x3, y3, x4, y4):
height1 = int(round(math.sqrt((x1-x4)*(x1-x4) + (y1-y4)*(y1-y4))))
height2 = int(round(math.sqrt((x3-x2)*(x3-x2) + (y3-y2)*(y3-y2))))
height = height1
if height2 > height:
height = height2
# add 25% to the height
height *= options.zoom_out_percent
#height += (height * .05)
#print "Height: %d - %d" % (height1, height2)
points = [(x1,y1), (x2,y2), (x3,y3), (x4,y4)]
print points
moment = cv2.moments(np.int32(points))
print moment
centerx = int(round(moment['m10']/moment['m00']))
print centerx
centery = int(round(moment['m01']/moment['m00']))
print centery
training_aspect = options.plate_width / options.plate_height
width = int(round(training_aspect * height))
# top_left = ( int(centerx - (width / 2)), int(centery - (height / 2)))
# bottom_right = ( int(centerx + (width / 2)), int(centery + (height / 2)))
top_left_x = int(round(centerx - (width / 2)))
top_left_y = int(round(centery - (height / 2)))
return (top_left_x, top_left_y, width, int(round(height)))
def crop_rect(big_image, x,y,width,height):
# Crops the rectangle from the big image and returns a cropped image
# Special care is taken to avoid cropping beyond the edge of the image.
# It fills this area in with random pixels
(big_height, big_width, channels) = big_image.shape
if x >= 0 and y >= 0 and (y+height) < big_height and (x+width) < big_width:
crop_img = img[y:y+height, x:x+width]
else:
#print "Performing partial crop"
#print "x: %d y: %d width: %d height: %d" % (x,y,width,height)
#print "big_width: %d big_height: %d" % (big_width, big_height)
crop_img = np.zeros((height, width, 3), np.uint8)
cv2.randu(crop_img, (0,0,0), (255,255,255))
offset_x = 0
offset_y = 0
if x < 0:
offset_x = -1 * x
x = 0
width -= offset_x
if y < 0:
offset_y = -1 * y
y = 0
height -= offset_y
if (x+width) >= big_width:
offset_x = 0
width = big_width - x
if (y+height) >= big_height:
offset_y = 0
height = big_height - y
#print "offset_x: %d offset_y: %d, width: %d, height: %d" % (offset_x, offset_y, width, height)
original_crop = img[y:y+height-1, x:x+width-1]
(small_image_height, small_image_width, channels) = original_crop.shape
#print "Small shape: %dx%d" % (small_image_width, small_image_height)
# Draw the small image onto the large image
crop_img[offset_y:offset_y+small_image_height, offset_x:offset_x+small_image_width] = original_crop
#cv2.imshow("Test", crop_img)
return crop_img
count = 1
yaml_files = []
for in_file in os.listdir(options.input_dir):
if in_file.endswith('.yaml') or in_file.endswith('.yml'):
yaml_files.append(in_file)
yaml_files.sort()
for yaml_file in yaml_files:
print "Processing: " + yaml_file + " (" + str(count) + "/" + str(len(yaml_files)) + ")"
count += 1
yaml_path = os.path.join(options.input_dir, yaml_file)
yaml_without_ext = os.path.splitext(yaml_file)[0]
with open(yaml_path, 'r') as yf:
yaml_obj = yaml.load(yf)
image = yaml_obj['image_file']
# Skip missing images
full_image_path = os.path.join(options.input_dir, image)
if not os.path.isfile(full_image_path):
print "Could not find image file %s, skipping" % (full_image_path)
continue
img = cv2.imread(full_image_path)
plate_corners = yaml_obj['plate_corners_gt']
cc = plate_corners.strip().split()
for i in range(0, len(cc)):
cc[i] = int(cc[i])
print i,i%2
print "cc:",cc
points = [(cc[0], cc[1]), (cc[2], cc[3]), (cc[4], cc[5]), (cc[6], cc[7])]
pts=np.array(points)
# box = get_box(cc[0], cc[1], cc[2], cc[3], cc[4], cc[5], cc[6], cc[7])
cv2.imshow("original", img)
# print "box:",box
# crop = crop_rect(img, box[0], box[1], box[2], box[3])
warped = four_point_transform(img, pts)
# show the original and warped images
cv2.imshow("Warped", warped)
# cv2.imshow("test", crop)
cv2.waitKey(0)
out_crop_path = os.path.join(options.out_dir, yaml_without_ext + ".jpg")
print out_crop_path
cv2.imwrite(out_crop_path, warped )
print "%d Cropped images are located in %s" % (count-1, options.out_dir)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment