-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathPiece.py
89 lines (69 loc) · 3.07 KB
/
Piece.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
import numpy as np
# Piece class, holds data of each piece
class Piece:
def __init__(self, num, s_vert, s_horz, chn, start, data, total):
self.pieceNum = num
self.size_vertical = s_vert
self.size_horizontal = s_horz
self.pieceChn = chn
self.pieceStart = start
self.pieceData = np.ndarray((s_vert, s_horz, chn), buffer=data, dtype=np.uint8)
self.pieceTotal = total
# the 4 borders of the piece
self.sideUp = []
self.sideRight = []
self.sideDown = []
self.sideLeft = []
for i in range(self.size_horizontal):
self.sideUp.append(self.pieceData[0][i])
self.sideDown.append(self.pieceData[-1][i])
for i in range(self.size_vertical):
self.sideRight.append(self.pieceData[i][-1])
self.sideLeft.append(self.pieceData[i][0])
self.sides = [self.sideUp, self.sideRight, self.sideDown, self.sideLeft]
self.difference = [None for x in range(total)]
self.neighbors = [None for x in range(4)]
# determine difference of pixel's each channel value
def pixel_difference(px1, px2):
PIXEL_DIFFERENCE_THRESHOLD = 30
diff = 0
for i in range(len(px1)):
diff += abs(int(px1[i] - int(px2[i])))
return False if diff < PIXEL_DIFFERENCE_THRESHOLD else True
# calculate different pixels between two sides
def side_difference(side1, side2):
difference = 0
for i in range(len(side1)):
difference += 1 if pixel_difference(side1[i], side2[i]) else 0
return difference
# calculate difference between two pieces in all directions
def piece_difference(piece1: Piece, piece2: Piece):
vertical_12 = side_difference(piece1.sideDown, piece2.sideUp)
vertical_21 = side_difference(piece2.sideDown, piece1.sideUp)
horizontal_12 = side_difference(piece1.sideRight, piece2.sideLeft)
horizontal_21 = side_difference(piece2.sideRight, piece1.sideLeft)
# clockwise direction
temp1 = [vertical_21, horizontal_12, vertical_12, horizontal_21]
temp2 = [vertical_12, horizontal_21, vertical_21, horizontal_12]
for i in range(len(temp1)):
temp1[i] = (temp1[i], i)
temp2[i] = (temp2[i], i)
# non-decreasing sort of difference
piece1.difference[piece2.pieceNum] = sorted(temp1)
piece2.difference[piece1.pieceNum] = sorted(temp2)
# search for neighbors
def find_neighbors(piece: Piece):
DIFFERENCE_RATE_THRESHOLD = 0.6
candidates = [None for x in range(4)]
# find the best candidate for each direction
for i in range(len(piece.difference)):
if piece.difference[i] is None:
continue
temp = piece.difference[i][0]
if candidates[temp[1]] is None or candidates[temp[1]][1][0] > temp[0]:
candidates[temp[1]] = (i, temp)
# test if candidate is eligible as neighbor
for entry in candidates:
if entry is not None and entry[1][0] <= DIFFERENCE_RATE_THRESHOLD *\
(piece.size_vertical if (entry[1][1] == 1 or entry[1][1] == 3) else piece.size_horizontal):
piece.neighbors[entry[1][1]] = entry[0]