代码拉取完成,页面将自动刷新
import cv2
import numpy as np
import math
import json
import time
import copy
from utils import log
class GetArmor:
def __init__(self,debug,video_debug_set,color,mode):
#传入变量私有化
self.mode = mode
self.color = color
self.debug = debug
self.video_debug_set = video_debug_set
#记录opencv版本的标志位
if int(cv2.__version__[0]) == 4:
self.version = 1
else:
self.version = 0
self.read_aimbot()
#初始化debug参数
if self.debug:
self.frame_debug = np.zeros([480, 640], dtype=np.uint8)
self.mask = np.zeros([480,640],dtype=np.uint8)
self.lightBarList = []
self.realCenter_list = []
self.armorProp = -1
self.x = -1
self.y = -1
self.z = -1
self.t0 = time.time()
def reinit(self,color,mode):
#更改变量时重初始化,更新参数
self.mode = mode
self.color = color
self.read_aimbot()
def read_aimbot(self):
#该函数用于读取自瞄参数
if self.mode == 0:
log.print_info('aimbot mode')
self.path = './json/armor_find.json'
elif self.mode == 3:
log.print_info('sentry mode')
self.path = './json/sentry_find.json'
else:
self.path = './json/armor_find.json'
with open(self.path,'r',encoding = 'utf-8') as load_f:
load_dict = json.load(load_f,strict=False)
if self.color == 0:
hsvPara_high = load_dict["ImageProcess_red"]["hsvPara_high"]
hsvPara_low = load_dict["ImageProcess_red"]["hsvPara_low"]
self.hsvPara = np.array([hsvPara_low,hsvPara_high])
elif self.color == 1:
hsvPara_high = load_dict["ImageProcess_blue"]["hsvPara_high"]
hsvPara_low = load_dict["ImageProcess_blue"]["hsvPara_low"]
self.hsvPara = np.array([hsvPara_low,hsvPara_high])
else:
log.print_error('unknow color num')
hsvPara_high = [-1,-1,-1]
hsvPara_low = [-1,-1,-1]
self.hsvPara = np.array([hsvPara_low,hsvPara_high])
self.GBSet_size = [-1,-1]
self.lowHue = hsvPara_low[0]
self.lowSat = hsvPara_low[1]
self.lowVal = hsvPara_low[2]
self.highHue = hsvPara_high[0]
self.highSat = hsvPara_high[1]
self.highVal = hsvPara_high[2]
#灯条筛选
self.minlighterarea = load_dict["ArmorFind"]["minlighterarea"]
self.maxlighterarea = load_dict["ArmorFind"]["maxlighterarea"]
self.minlighterProp = load_dict["ArmorFind"]["minlighterProp"]
self.maxlighterProp = load_dict["ArmorFind"]["maxlighterProp"]
self.minAngleError = load_dict["ArmorFind"]["minAngleError"]
self.maxAngleError = load_dict["ArmorFind"]["maxAngleError"]
#装甲板筛选
self.minarealongRatio = load_dict["ArmorFind"]["minarealongRatio"]
self.maxarealongRatio = load_dict["ArmorFind"]["maxarealongRatio"]
self.lightBarAreaDiff = load_dict["ArmorFind"]["lightBarAreaDiff"]
self.armorAngleMin = load_dict["ArmorFind"]["armorAngleMin"]
self.minarmorArea = load_dict["ArmorFind"]["minarmorArea"]
self.maxarmorArea = load_dict["ArmorFind"]["maxarmorArea"]
self.minarmorProp = load_dict["ArmorFind"]["minarmorProp"]
self.maxarmorProp = load_dict["ArmorFind"]["maxarmorProp"]
self.minBigarmorProp = load_dict["ArmorFind"]["minBigarmorProp"]
self.maxBigarmorProp = load_dict["ArmorFind"]["maxBigarmorProp"]
self.angleDiff_near = load_dict["ArmorFind"]["angleDiff_near"]
self.angleDiff_far = load_dict["ArmorFind"]["angleDiff_far"]
self.minareawidthRatio = load_dict["ArmorFind"]["minareawidthRatio"]
self.maxareawidthRatio = load_dict["ArmorFind"]["maxareawidthRatio"]
self.minareaRatio = load_dict["ArmorFind"]["minareaRatio"]
self.maxareaRatio = load_dict["ArmorFind"]["maxareaRatio"]
self.area_limit = load_dict["ArmorFind"]["area_limit"]
self.xcenterdismax = load_dict["ArmorFind"]["xcenterdismax"]
self.ylengthmin = load_dict["ArmorFind"]["ylengthmin"]
self.ylengcenterRatio = load_dict["ArmorFind"]["ylengcenterRatio"]
self.yixaingangleDiff_near = load_dict["ArmorFind"]["yixaingangleDiff_near"]
self.yixaingangleDiff_far = load_dict["ArmorFind"]["yixaingangleDiff_far"]
#测距
self.kh = load_dict["ArmorFind"]["kh"]
with open('./json/common.json','r',encoding = 'utf-8') as load_f:
load_dict = json.load(load_f,strict=False)
self.xCenter = load_dict["Aimbot"]["width"]/2
self.yCenter = load_dict["Aimbot"]["height"]/2
def GetArmor(self,frame):
#识别装甲板主程序
x, y, z = -1, -1, -1
#根据debug参数更改图像来源格式
if self.video_debug_set == 0:
frame = cv2.cvtColor(frame, cv2.COLOR_BAYER_RG2RGB)
mask = self.HSV_Process(frame)# 进行HSV图像处理
lightBarList = self.GetLightBar(mask)# 进行获取灯条矩形列表
x,y,z = self.CombineLightBar_ground(lightBarList)#非卡尔曼滤波,将灯条拼凑成装甲板
#根据debug参数确定显示图像
if self.debug:
self.mask = mask
self.frame_debug = frame
#每秒更新1次json文件中的参数
if time.time()-self.t0 > 1:
self.update_json()
self.t0 = time.time()
return x,y,z
def HSV_Process(self,frame):
#二值化图像
frame = cv2.cvtColor(frame, cv2.COLOR_BGR2HSV)
mask = cv2.inRange(frame, self.hsvPara[0], self.hsvPara[1])
return mask
def GetLightBar(self,mask):
#筛选灯条
lightBarList = []
self.lightBarList = []
#对于opencv3,轮廓寻找函数有3个返回值,对于opencv4只有两个
if self.version:
contours, hierarchy = cv2.findContours(mask, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_TC89_L1)
else:
_, contours, hierarchy = cv2.findContours(mask, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_TC89_L1)
#轮廓数量超过400不认为正常识别
if(len(contours)>400):
lightBarList = []
else:
for contour in range(len(contours)):
#如果轮廓数小于5认为不是正常的灯条
if len(contours[contour]) >= 5:
#size[0]是灯条宽 size[1]是灯条长
center, size, angle = cv2.fitEllipse(contours[contour])#反馈的长宽其实就是长的轴和短的轴,所以一定长宽比大于1
#对轮廓拟合出现的值进行判断
if size[1] <= 0 or size[0] <= 0 or np.isnan(size[0]) or np.isnan(size[1]) :
continue
rectProp = size[1]/size[0]#灯条长比宽,值大于1
rectArea = size[0]*size[1]
angleHori = abs(int(angle)%180)
if angleHori > self.minAngleError and angleHori < self.maxAngleError:
# log.print_debug("angleHori = "+str(angleHori))
continue
elif rectProp < self.minlighterProp or rectProp > self.maxlighterProp:#得介于最小值最大值之间才要
# log.print_debug("rectProp = "+str(rectProp))
continue
elif rectArea > self.maxlighterarea or rectArea < self.minlighterarea:
# log.print_debug("rectArea = "+str(rectArea))
continue
lightBarList.append([center[0],center[1],size[0],size[1],angle])
if self.debug:
#如果开启了debug模式,向类变量更新值
self.lightBarList.append([center[0],center[1],size[0],size[1],angle,rectProp,angleHori,rectArea])
else :
# log.print_debug("<5")
pass
return lightBarList
def CombineLightBar_ground(self,lightBarList):
#筛选装甲板
realCenter_list = []
realCenter = [-1, -1]
x,y,z = -1,-1,-1
#灯条数量没有两个,构建不出装甲板
if len(lightBarList) < 2:
log.print_debug("NO Armor")
return x, y, z
for i in range(0,len(lightBarList)-1):
for j in range(i+1,len(lightBarList)):
x0, y0, l0, s0, a0 = lightBarList[i][0:5]#s0 s1是灯条长 l0 l1是灯条宽
x1, y1, l1, s1, a1 = lightBarList[j][0:5]
xcenterdis = abs(x0-x1) #灯条横向距离
ycenterdis = abs(y0-y1) #灯条纵向距离
angleDiff = abs(a0 - a1) #灯条角度差
areaDiff = abs(s0*l0 - s1*l1) #灯条面积差
areaRatio = (s0*l0)/(s1*l1) #灯条面积比
arealongRatio = s0/s1 #灯条长长比
areawidthRatio = l0/l1 #灯条宽宽比
xCenter = (x0+x1)/2 #装甲板中心x值
yCenter = (y0+y1)/2 #装甲板中心y值
ylength = (s0+s1)/2 #装甲板纵向长度
xlength = self.EuclideanDistance([x0,y0],[x1,y1]) #装甲板横向长度
armorProp = xlength/ylength #装甲板长宽比
armorArea = xlength*ylength #装甲板面积
angle = abs((y0-y1)/(x0-x1)) #装甲板角度
#根据面积选择筛选条件
if((s0*l0 + s1*l1) < self.area_limit):
self.angleDiff = self.angleDiff_near
yixaingangleDiff = self.yixaingangleDiff_near
else:
self.angleDiff = self.angleDiff_far
yixaingangleDiff = self.yixaingangleDiff_far
#区分左右装甲板,算出灯板角度
if x0 > x1:
angle = angle*180/math.pi
else:
angle = -angle*180/math.pi
#灯条长长过大过小不要
if arealongRatio < self.minarealongRatio or arealongRatio > self.maxarealongRatio:
# log.print_debug("arealongRatio = "+str(arealongRatio)+str(self.minarealongRatio)+str(self.maxarealongRatio))
continue
#灯条宽宽比过大过小不要
if areawidthRatio < self.minareawidthRatio or areawidthRatio > self.maxareawidthRatio:
# log.print_debug("areawidthRatio = "+str(areawidthRatio))
continue
#灯条角度差过大不要
if (angleDiff > self.angleDiff and angleDiff < 180-self.angleDiff) or (angleDiff > 90 and angleDiff < 180 - yixaingangleDiff):
# log.print_debug("angleDiff = "+str(angleDiff))
continue
#灯条面积比太大不要
if areaRatio > self.maxareaRatio or areaRatio < self.minareaRatio:
# log.print_debug("areaRatio = "+str(areaRatio))
continue
#灯条中心太近不要
if xcenterdis < self.xcenterdismax:
# log.print_debug( "abs(x0-x1) < 15")
continue
#灯条面积差太大不要
if areaDiff > self.lightBarAreaDiff:
# log.print_debug("areaDiff = "+str(areaDiff))
continue
#装甲板角度过偏不要
if abs(angle) > self.armorAngleMin:
# log.print_debug(" abs(angle) > = "+str(abs(angle)))
continue
#装甲板高度太小不要
if ylength < self.ylengthmin:
# log.print_debug(" ylength < 10")
continue
#装甲板面积太大或太小不要
if armorArea < self.minarmorArea or armorArea > self.maxarmorArea:
# log.print_debug("armorArea < self.minarmorArea"+str(armorArea))
continue
#两个灯条y轴高度差过大不要
if(ycenterdis>ylength*self.ylengcenterRatio):
# log.print_debug("两个灯条y轴高度差过大不要"+str(abs(y0-y1)/ylength))
continue
# #装甲板长宽比太大或太小不要
if (armorProp < self.minarmorProp or armorProp > self.maxarmorProp) and (armorProp < self.minBigarmorProp or armorProp > self.maxBigarmorProp):
continue
z = self.GetArmorDistance(s0,s1)
realCenter_list.append([xCenter,yCenter,xlength,ylength,angle,z])
#挑选距离屏幕中心最近的装甲板
temp_distence = -1
if len(realCenter_list) > 0:
for realCenter in realCenter_list:
armor_imgcenter_distence = self.EuclideanDistance([realCenter[0],realCenter[1]],[self.xCenter,self.yCenter])
if temp_distence > armor_imgcenter_distence or temp_distence == -1:
temp_distence = armor_imgcenter_distence
x = realCenter[0]
y = realCenter[1]
z = realCenter[5]
else :
# log.print_debug("no build Armor")
pass
if temp_distence == -1:
x,y,z = -1,-1,-1
if self.debug:
#如果开启了debug模式,向类变量更新值
self.realCenter_list = realCenter_list
self.armorProp = armorProp
self.x = x
self.y = y
self.z = z
return x, y, z
def GetArmorDistance(self,dShortside,dLongside):
#单目视觉测距,简单的根据装甲板最长边的对应像素去计算
#注意,调参时需要在hsv确定后再调整测距
height = dShortside + dLongside
distance = self.kh/height
return distance
def TrackerBar_value(self):
#返回滑动条的初值
if self.debug:
if self.getvar_label == False:
self.getvar_label = True
return \
self.lowHue,\
self.lowSat,\
self.lowVal,\
self.highHue,\
self.highSat,\
self.highVal,\
int(self.minlighterarea),\
int(self.maxlighterarea),\
int(self.minlighterProp*100),\
int(self.maxlighterProp*100),\
int(self.minAngleError*10),\
int(self.maxAngleError*10),\
int(self.maxarealongRatio*100),\
int(self.minarealongRatio*100),\
int(self.lightBarAreaDiff),\
int(self.armorAngleMin*10),\
int(self.minarmorArea),\
int(self.maxarmorArea),\
int(self.minarmorProp*100),\
int(self.maxarmorProp*100),\
int(self.minBigarmorProp*100),\
int(self.maxBigarmorProp*100),\
int(self.angleDiff_near*10),\
int(self.angleDiff_far*10),\
int(self.minareawidthRatio*100),\
int(self.maxareawidthRatio*100),\
int(self.minareaRatio*100),\
int(self.maxareaRatio*100),\
int(self.area_limit),\
int(self.xcenterdismax*10),\
int(self.ylengthmin),\
int(self.ylengcenterRatio),\
int(self.yixaingangleDiff_near),\
int(self.yixaingangleDiff_far),\
int(self.kh)
def updata_argument(self,temp):
#根据debug更新参数
if self.debug and self.getvar_label:
lowHue = float(temp[0])
lowSat = float(temp[1])
lowVal = float(temp[2])
highHue = float(temp[3])
highSat = float(temp[4])
highVal = float(temp[5])
hsvPara_high = [highHue,highSat,highVal]
hsvPara_low = [lowHue,lowSat,lowVal]
self.hsvPara = np.array([hsvPara_low,hsvPara_high])
self.minAngleError = float(temp[6])/10
self.maxAngleError = float(temp[7])/10
self.minlighterarea = float(temp[8])
self.maxlighterarea = float(temp[9])
self.minlighterProp = float(temp[10])/100
self.maxlighterProp = float(temp[11])/100
self.lightBarAreaDiff = float(temp[12])
self.maxarealongRatio = float(temp[13])/100
self.minarealongRatio = float(temp[14])/100
self.armorAngleMin = float(temp[15])/10
self.minarmorArea = float(temp[16])
self.maxarmorArea = float(temp[17])
self.minarmorProp = float(temp[18])/100
self.maxarmorProp = float(temp[19])/100
self.minBigarmorProp = float(temp[20])/100
self.maxBigarmorProp = float(temp[21])/100
self.angleDiff_near = float(temp[22])/10
self.angleDiff_far = float(temp[23])/10
self.minareawidthRatio = float(temp[24])/100
self.maxareawidthRatio = float(temp[25])/100
self.minareaRatio = float(temp[26])/100
self.maxareaRatio = float(temp[27])/100
self.area_limit = float(temp[28])
self.xcenterdismax = float(temp[29])/10
self.ylengthmin = float(temp[30])
self.ylengcenterRatio = float(temp[31])
self.yixaingangleDiff_near = float(temp[32])
self.yixaingangleDiff_far = float(temp[33])
self.kh = float(temp[34])
def update_json(self):
temp_list = [
self.minAngleError ,\
self.maxAngleError ,\
self.minlighterarea ,\
self.maxlighterarea ,\
self.minlighterProp ,\
self.maxlighterProp ,\
self.lightBarAreaDiff ,\
self.maxarealongRatio ,\
self.minarealongRatio ,\
self.armorAngleMin ,\
self.minarmorArea ,\
self.maxarmorArea ,\
self.minarmorProp ,\
self.maxarmorProp ,\
self.minBigarmorProp ,\
self.maxBigarmorProp ,\
self.angleDiff_near ,\
self.angleDiff_far ,\
self.minareawidthRatio ,\
self.maxareawidthRatio ,\
self.minareaRatio ,\
self.maxareaRatio ,\
self.area_limit ,\
self.xcenterdismax ,\
self.ylengthmin ,\
self.ylengcenterRatio ,\
self.kh
]
#这里用于校验,保证序列化写入json前的数字是正常的,防止程序在debug时出现bug,导致参数丢失
if temp_list.count(0.) > 15 or temp_list.count(-1) > 0 or \
self.hsvPara[1][1] < 230 or self.hsvPara[1][2] < 230:
log.print_error('json write failed, para is error')
else:
with open(self.path,'r',encoding = 'utf-8') as load_f:
load_dict = json.load(load_f,strict=False)
load_dict["ArmorFind"]["minAngleError"] = self.minAngleError
load_dict["ArmorFind"]["maxAngleError"] = self.maxAngleError
load_dict["ArmorFind"]["minlighterarea"] = self.minlighterarea
load_dict["ArmorFind"]["maxlighterarea"] = self.maxlighterarea
load_dict["ArmorFind"]["minlighterProp"] = self.minlighterProp
load_dict["ArmorFind"]["maxlighterProp"] = self.maxlighterProp
load_dict["ArmorFind"]["angleDiff_near"] = self.angleDiff_near
load_dict["ArmorFind"]["angleDiff_far"] = self.angleDiff_far
load_dict["ArmorFind"]["lightBarAreaDiff"] = self.lightBarAreaDiff
load_dict["ArmorFind"]["maxarealongRatio"] = self.maxarealongRatio
load_dict["ArmorFind"]["minarealongRatio"] = self.minarealongRatio
load_dict["ArmorFind"]["armorAngleMin"] = self.armorAngleMin
load_dict["ArmorFind"]["minarmorArea"] = self.minarmorArea
load_dict["ArmorFind"]["maxarmorArea"] = self.maxarmorArea
load_dict["ArmorFind"]["minarmorProp"] = self.minarmorProp
load_dict["ArmorFind"]["maxarmorProp"] = self.maxarmorProp
load_dict["ArmorFind"]["minBigarmorProp"] = self.minBigarmorProp
load_dict["ArmorFind"]["maxBigarmorProp"] = self.maxBigarmorProp
load_dict["ArmorFind"]["angleDiff_near"] = self.angleDiff_near
load_dict["ArmorFind"]["angleDiff_far"] = self.angleDiff_far
load_dict["ArmorFind"]["minareawidthRatio"] = self.minareawidthRatio
load_dict["ArmorFind"]["maxareawidthRatio"] = self.maxareawidthRatio
load_dict["ArmorFind"]["minareaRatio"] = self.minareaRatio
load_dict["ArmorFind"]["maxareaRatio"] = self.maxareaRatio
load_dict["ArmorFind"]["area_limit"] = self.area_limit
load_dict["ArmorFind"]["xcenterdismax"] = self.xcenterdismax
load_dict["ArmorFind"]["ylengthmin"] = self.ylengthmin
load_dict["ArmorFind"]["ylengcenterRatio"] = self.ylengcenterRatio
load_dict["ArmorFind"]["yixaingangleDiff_near"] = self.yixaingangleDiff_near
load_dict["ArmorFind"]["yixaingangleDiff_far"] = self.yixaingangleDiff_far
load_dict["ArmorFind"]["kh"] = self.kh
if self.color == 0:
load_dict["ImageProcess_red"]["hsvPara_high"] = [int(x) for x in list(self.hsvPara[1])]
load_dict["ImageProcess_red"]["hsvPara_low"] = [int(x) for x in list(self.hsvPara[0])]
elif self.color == 1:
load_dict["ImageProcess_blue"]["hsvPara_high"] = [int(x) for x in list(self.hsvPara[1])]
load_dict["ImageProcess_blue"]["hsvPara_low"] = [int(x) for x in list(self.hsvPara[0])]
dump_dict = load_dict
with open(self.path,'w',encoding = 'utf-8') as load_f:
json.dump(dump_dict,load_f,indent=4,ensure_ascii=False)
def EuclideanDistance(self,c,c0):
#欧式距离计算
return pow((c[0]-c0[0])**2+(c[1]-c0[1])**2, 0.5)
def nothing(self,*arg):
#空的回调函数
pass
def get_debug_frame(self):
#处理并返回显示所需要的debug图像
lightBarList = copy.deepcopy(self.lightBarList)
realCenter_list = copy.deepcopy(self.realCenter_list)
x,y,z = copy.deepcopy(self.x) ,copy.deepcopy(self.y), copy.deepcopy(self.z)
armorProp = copy.deepcopy(self.armorProp)
frame_debug = copy.deepcopy(self.frame_debug)
mask = copy.deepcopy(self.mask)
for i in range(len(lightBarList)):
log.print_debug(lightBarList[i])
cv2.ellipse(frame_debug,(int(lightBarList[i][0]),int(lightBarList[i][1])),(int(lightBarList[i][2]/2),int(lightBarList[i][3]/2)),lightBarList[i][4],0,360,(0,255,0),2,8)
for i in range(len(realCenter_list)):
cv2.ellipse(frame_debug,(int(realCenter_list[i][0]),int(realCenter_list[i][1])),(int(realCenter_list[i][2]/2),int(realCenter_list[i][3]/2)),realCenter_list[i][4],0,360,(255,0,255),3,2)
cv2.circle(frame_debug,(int(realCenter_list[i][0]),int(realCenter_list[i][1])),5,(0,255,0),-1)
cv2.circle(frame_debug,(int(x),int(y)),10,(255,255,255),-1)
cv2.putText(self.frame_debug,"distance = % d"%z,(50,50), cv2.FONT_HERSHEY_SIMPLEX, 0.75, (0,255,255),2)
cv2.putText(self.frame_debug,"armorProb = % d"%(armorProp*100),(50,100), cv2.FONT_HERSHEY_SIMPLEX, 0.75, (0,255,255),2)
return frame_debug,mask
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。