-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathclient.py
475 lines (415 loc) · 12.8 KB
/
client.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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
import socket
import threading
import time
import cv2
import numpy as np
import array as ar
import matplotlib.pyplot as plt
def perspective(img):
src=np.float32(
[[2,349],
[161,233],
[478,233],
[631,349]])
dst=np.float32(
[[115,480],
[115,70],
[525,70],
[525,480]])
img_size=(img.shape[1],img.shape[0])
M=cv2.getPerspectiveTransform(src,dst)
warped=cv2.warpPerspective(img,M,img_size)
return warped
def threshold(img):
gray=cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)
blur=cv2.GaussianBlur(gray,(5,5),0)
a,thresh=cv2.threshold(blur,75,200,cv2.THRESH_BINARY)
return thresh
def plotpid():
plt.ion()
t=0
dss=25
mdss=[]
mt=[]
dis=[]
while True:
global distance
plt.clf()
mt.append(t)
mdss.append(dss)
dis.append(distance)
plt.plot(mt,dis,'-r')
plt.plot(mt, mdss, '-b')
plt.ylim(0,100)
plt.pause(1)
t=t+1
plt.ioff()
plt.show()
def PID(err,Ierr,derr):
KP = 0.5
KI = 0.12
KD = 15
out = KP*err + KI*(sum(Ierr)) + KD*derr
if out > 50:
out = 50
if out < 0:
out = 0
return out
def A_PID(err,Ierr,derr):
KP = 0.05
KI = 0
KD = 20
err = err
out = KP*err + KI*(sum(Ierr)) + KD*derr
if out > 15:
out = 15
if out < -15:
out = -15
return out
def cal_distance(pixel_width):
f = 510
phone_width = 9
distance = phone_width * f / pixel_width
return distance
def cal_box(box):
under_pix_x = box[0][0] - box[3][0]
under_pix_y = box[0][1] - box[3][1]
upper_pix_x = box[1][0] - box[2][0]
upper_pix_y = box[1][1] - box[2][1]
under_pix = np.sqrt(under_pix_x**2 + under_pix_y**2)
upper_pix = np.sqrt(upper_pix_x ** 2 + upper_pix_y ** 2)
return (under_pix+upper_pix)/2
def findline_edges(img, rangle):
global Istruning
Istruning = False
if np.where(img[240,:]!=0) is not None:
lowmaxv=max(max(np.where(img[240,:]!=0)),default=-1)
else:
lowmaxv=None
if np.where(img[240,:]!=0) is not None:
lowminv=min(max(np.where(img[240,:]!=0)),default=-1)
else:
lowminv=None
# if np.where(img[190,:]!=0) is not None:
# upmaxv=max(max(np.where(img[190,:]!=0)),default=-1)
# else:
# upmaxv=None
# if np.where(img[190,:]!=0) is not None:
# upminv=min(max(np.where(img[190,:]!=0)),default=-1)
# else:
# upminv=None
#中線偏移
if lowmaxv is None or lowminv is None:
averagev=320
else:
averagev=(lowminv+lowmaxv)/2
deltaLine = (lowmaxv-lowminv)
# cv2.circle(img, (int(averagev), 240), 5, (255, 255, 255), -1)
# if averagev < 320:
# theta = (-1) * (np.arctan2((320 - averagev), 480)) #240
# elif averagev >= 320:
theta = (np.arctan2((averagev - 320), 480))
theta = theta * 180 / np.pi
if theta > 40:
theta = 40
elif theta < -40:
theta = -40
else:
theta =theta
# if lowmaxv is None or upmaxv is None :
# rightangle=90
# else:
# x=upmaxv-lowmaxv
# if x == 0:
# rightangle=90
# else:
# rightangle=np.arctan2(80,x)*180/np.pi
#
# if lowminv is None or upminv is None:
# leftangle = 90
# else:
# x = upminv - lowminv
# if x == 0:
# leftangle =90
# else:
# leftangle = np.arctan2(80, x)*180/np.pi
# roadangle=(rightangle+leftangle)/2
# rangle=90-roadangle
# allangle=rangle*3/5+theta*2/5
allangle = theta
# print('總轉角=',allangle)
# print('theta=',theta)
# print('rightangle=',rightangle)
# print('leftangle=',leftangle)
if deltaLine<400:
if rangle >0 and averagev >350:
allangle=allangle
# print("1")
elif rangle <0 and averagev <290:
allangle=allangle
# print("2")
elif rangle >0 and averagev >290 and averagev<350:
allangle=20
# print("3")
Istruning = True
elif rangle <0 and averagev >290 and averagev<350:
allangle=-20
# print("4")
Istruning = True
else:
allangle = allangle*(-1)
# print("5")
# print("allangle ",allangle)
return allangle
def findMarker (image):
#轉正
angle = 0
persvideo = perspective(image)
# gray = cv2.cvtColor(persvideo, cv2.COLOR_BGR2GRAY)
persvideo[(persvideo < 5)] = 200
blur = cv2.GaussianBlur(persvideo,(5,5),0)
a, thresh=cv2.threshold(blur,100,250,cv2.THRESH_BINARY)
t_s = time.time()
# thpersvideo=threshold(persvideo)
th_Canny = cv2.Canny(thresh, 5, 90)
cv2.imshow('Canny', th_Canny)
p = np.where(th_Canny > 230)
x = p[1]
y = p[0]*-1
print(len(p[0]))
print(len(p[0])!=0)
if len(p[0])!=0:
zzz = np.zeros_like(th_Canny)
A = np.matrix.transpose(np.array([x, np.ones(x.size)]))
B = np.matrix.transpose(np.array([y]))
A_transpose = np.matrix.transpose(A)
A_square = np.dot(A_transpose, A)
A_square_inverse = np.linalg.inv(A_square)
X_matrix = np.dot(np.dot(A_square_inverse, A_transpose), B)
m = X_matrix[0]
t_e = time.time()
# print("time",t_e - t_s)
# print("m",m)
angle=findline_edges(th_Canny ,m)
# cv2.line(th_Canny, (320, 0), (320, 480), (255, 192, 203), 2)
#
# # cv2.line(img, (0, 190), (640, 190), (255, 255, 255), 2)
# cv2.line(th_Canny, (0, 240), (640, 240), (255, 192, 203), 2)
# cv2.imshow('persvideo',persvideo)
# cv2.imshow('thresh', thresh)
findcube = image[0:300,0:640]
#抓方框
cube, thresh_c = cv2.threshold(findcube, 60, 255, cv2.THRESH_BINARY_INV)
contours, _ = cv2.findContours(thresh_c, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
# cv2.imshow('cube',thresh_c)
if contours:
cnt_max = max ( contours, key = cv2.contourArea )
rect = cv2.minAreaRect(cnt_max)
box = np.int0(cv2.boxPoints(rect))
cv2.drawContours(findcube, [box], 0, (0, 255, 0), 2)
# findcube = cv2.cvtColor(findcube, cv2.COLOR_GRAY2RGB)
# cv2.imshow('pers', thpersvideo)
pixel_width = cal_box(box)
D = cal_distance(pixel_width)
if D is None:
D = 0
else:
D=0
cv2.imshow('drawcont', findcube)
return [D, angle]
def recv(sock, addr):
'''
一個UDP連線在接收訊息前必須要讓系統知道所佔埠
也就是需要send一次,否則win下會報錯
'''
sock.sendto(name.encode('utf-8'), addr)
# while True:
# data = sock.recv(1024)
def send(sock, addr):
'''
傳送資料的方法
引數:
sock:定義一個例項化socket物件
server:傳遞的伺服器IP和埠
'''
global backVoltage
global frontAngle
global Istruning
while True:
# string = input('')
# message = name + ' : ' + string
# messagelist = string.split(',')
try:
# # print(frontAngle-45)
# turn = abs(frontAngle-45)
#
# if turn >20:
# backVoltage = 45
# print("1111111")
#
# elif turn >10:
# backVoltage = 35
# print("2222222")
#
# else:
# backVoltage = 30
# print("3333333")
# backVoltage = 30 + abs(frontAngle-45)
# if backVoltage < 30:
# backVoltage = 30
# if backVoltage >40:
# backVoltage = 40
# if (frontAngle-45)>20:
# backVoltage = 50
# else:
# backVoltage = 30
# if abs(frontAngle-40) < 10:
# backVoltage = 30
# else:
# backVoltage = 30
A = ar.array('h',[int(backVoltage),int(frontAngle)])
# A = ar.array('h', [int(0), int(0)])
memA=memoryview(A)
# data = message.encode('utf-8')
sock.sendto(memA, addr)
# print('angle=', frontAngle)
time.sleep(0.1)
except:
continue
# sock.send(pickle.dumps(A))
# if string.lower() == 'EXIT'.lower():
# break
def main():
'''
主函式執行方法,通過多執行緒來實現多個客戶端之間的通訊
'''
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
server = ('192.168.50.5', 9999)
tr = threading.Thread(target=recv, args=(s, server), daemon=True)
ts = threading.Thread(target=send, args=(s, server))
tr.start()
ts.start()
tr.join()
ts.join()
s.close()
def ReceiveVideo():
global backVoltage
global frontAngle
global distance,plot_t
#distance pid 參數
err = 0
Ierr = [0,0,0]
preerr = 0
derr = 0
essential_distance = 25
#angle pid 參數
A_err = 0
A_Ierr = [0, 0, 0]
A_preerr = 0
A_derr = 0
print('recieve go')
# IP地址'0.0.0.0'為等待客戶端連線
address = ('192.168.50.5', 8002)
# 建立socket物件,引數意義見https://blog.csdn.net/rebelqsp/article/details/22109925
# socket.AF_INET:伺服器之間網路通訊
# socket.SOCK_STREAM:流式socket,for TCP
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 將套接字繫結到地址,在AF_INET下,以元組(host,port)的形式表示地址.
s.bind(address)
# 開始監聽TCP傳入連線。引數指定在拒絕連線之前,作業系統可以掛起的最大連線數量。該值至少為1,大部分應用程式設為5就可以了。
s.listen(1)
def recvall(sock, count):
buf = b'' # buf是一個byte型別
while count:
# 接受TCP套接字的資料。資料以字串形式返回,count指定要接收的最大資料量.
newbuf = sock.recv(count)
if not newbuf: return None
buf += newbuf
count -= len(newbuf)
return buf
# 接受TCP連線並返回(conn,address),其中conn是新的套接字物件,可以用來接收和傳送資料。addr是連線客戶端的地址。
# 沒有連線則等待有連線
conn, addr = s.accept()
print('connect from:' + str(addr))
# plt.ion()
# plt.figure(1)
t=[]
dis=[]
motorangle=45
angle = 45
while 1:
start = time.time() # 用於計算幀率資訊
length = recvall(conn, 16) # 獲得圖片檔案的長度,16代表獲取長度
# try:
stringData = recvall(conn, int(length)) # 根據獲得的檔案長度,獲取圖片檔案
data = np.frombuffer(stringData, np.uint8) # 將獲取到的字元流資料轉換成1維陣列
decimg = cv2.imdecode(data, cv2.IMREAD_UNCHANGED) # 將陣列解碼成影象
wData = findMarker(decimg)
distance = wData[0]
angle = wData[1]
# except Exception as e:
# print(e)
# pass
plot_t=start
err = distance - essential_distance
Ierr.pop(0)
Ierr.append(err)
derr = err - preerr
preerr=err
PID_out = PID(err,Ierr,derr)
A_err = angle
A_Ierr.pop(0)
A_Ierr.append(A_err)
A_derr = A_err - A_preerr
A_preerr=A_err
A_PID_out = A_PID(A_err,A_Ierr,A_derr)
# print("PID angle", A_PID_out)
frontAngle = angle+45 #motorangle+A_PID_out
turnEdge = 30
if frontAngle >=45+turnEdge:
frontAngle=45+turnEdge
elif frontAngle <=45-turnEdge:
frontAngle=45-turnEdge
else:
frontAngle=frontAngle
motorangle=frontAngle
# print("angle", (angle))
backVoltage = PID_out * (100/50)
# cv2.imshow('SERVER', decimg) # 顯示影象
# 進行下一步處理
# 。
# 。
# 。
# 將幀率資訊回傳,主要目的是測試可以雙向通訊
end = time.time()
seconds = end - start
if seconds == 0:
fps=0
# else:
# print("deltaTime = ",seconds)
# conn.send(bytes(str(int(fps)), encoding='utf-8'))
k = cv2.waitKey(10) & 0xff
if k == 27:
break
s.close()
cv2.destroyAllWindows()
if __name__ == '__main__':
distance = 0
backVoltage = 0
frontAngle = 0
Istruning = False
plot_t=0
print("-----歡迎來到聊天室,退出聊天室請輸入'EXIT(不分大小寫)'-----")
name = 'Kuo'
print('-----------------%s------------------' % name)
tvideo = threading.Thread(target=ReceiveVideo)
tvideo.start()
main()
# tplot = threading.Thread(target=plotpid)
# tplot.start()
# tmain = threading.Thread(target=main)
# tmain.start()
# main()
tvideo.join()
# tplot.join()
# tmain.join()