VPLC系列机器视觉运动控制一体机快速入门(七)
2024-10-09 来源:elecfans
此前,我们依次讲解了软硬件介绍及计数实例、相机的基本使用、基于形状匹配的视觉定位、BLOB有无检测、测量尺寸、机器视觉方案中使用到的标定功能以及使用ZDevelop软件实现坐标标定的方法。
本期课程我们继续和大家一起分享使用ZDevelop软件实现一维码和二维码的识别功能。
条形码是由不同的宽度、不同的反射率的条(黑色)和空(白色)组成的,根据特定的编码规则编制,用于表达一组数字、字母信息的图形标识符。
条形码可以标出商品的生产国、制造厂家、商品名称、生产日期、图书分类号、类别、日期等信息,因而在商品流通、图书管理、银行系统、生产制造等许多领域都得到了广泛的应用。
二维码是用某种特定的几何图形按一定规律在平面(二维方向上)分布的、黑白相间的、记录数据符号信息的图形。在代码编制上巧妙地利用构成计算机内部逻辑基础的“0”、“1”比特流的概念,使用若干个与二进制相对应的几何形体来表示文字数值信息,通过图像输入设备或光电扫描设备自动识读以实现信息自动处理。
二维码技术是在计算机技术与信息技术基础上发展起来的一门集编码、印刷、识别、数据采集和处理于一身的新兴技术,它解决了条形码表达信息有限的问题。
一维条形码只能在一个方向上(一般是水平方向)存储表达信息,只能存储数字和字母;二维码在水平和垂直两个方向上均能存储表达信息,它可以存储更多的信息包括数字、字母、汉字、图片、音频、视频等。
条形码需要按照一定规则的编码方式将条和空进行不同的排列用于表示不同的信息;二维码需要按照一定的编码规律使用黑白块在水平方向和垂直方向进行排列,用于表达不同的信息。它们都依赖于特定的编码规则---编码码制,才能准确实现信息的存储和表达。
正是由于条形码和二维码在日常生活和工业生产等多种领域中被广泛应用到,因此在自动化生产制造业中需要自动检测识别条形码和二维码的内容,对识别的内容进行判断,比如根据判断条形码和检测的字符是否一致来最终判断产品包装的信息的准确性;比如,通过识别二维码的内容,导入对应产品的信息。
机器视觉是常用于自动化生产制造行业的一门自动检测技术,它根据条形码和二维码的编码原理也相应生成了对应的识别算法,可应用于自动检测识别条形码和二维码。
识别流程图
演示实例说明:本课程实例将演示使用ZDevelop软件识别常用条形码和二维码的类型。
1.打开ZDevelop软件:新建项目→新建HMI文件→新建main.bas文件,用于编写界面响应函数→新建global_variable.bas文件用于存放全局变量并开启HMI自动运行任务→新建identify.bas文件用于初始化测量参数→新建camera.bas文件用于实现相机采集功能→新建draw.bas文件用于更新绘制检测区域ROI刷新界面→文件添加到项目。
2.设计HMI文件界面。
3.在global_variable.bas文件中定义全局变量。
'''''全局变量大部分使用数组结构'''''
''注:basic编程中很多函数会以TABLE(系统的数据结构)做为参数
''在这里table均是做为中间变量
''table 0-10 作为中间变量使用
''table 11-15,区域ROI参数,参数位置与dd_identfy_param对应,控件坐标系
''table 21-22,鼠标按键,控件坐标系
''table 31-35,控件坐标转换后对应的图像坐标,图像坐标系
'主任务状态
'0 - 未初始化
'1 - 停止
'2 - 运行中
'3 - 正在停止
GLOBAL DIM main_task_state
main_task_state = 1
'采集开关
'0 - 停止采集
'1 - 请求采集
GLOBAL DIM grab_switch
grab_switch = 0
'相机个数
GLOBAL cam_num
cam_num = 0
'相机种类,''
GLOBAL DIM CAMERA_TYPE(100)
'CAMERA_TYPE = 'zmotion;mindvision;basler;mvision;huaray'
CAMERA_TYPE = 'mvision'
' 任务号划分, 主任务id - 10
GLOBAL DIM main_task_id
main_task_id = 10
'连续采集线程id - 9
GLOBAL DIM grab_task_id
grab_task_id = 9
'目前不能作为函数参数,故使用全局变量表示
GLOBAL ZVOBJECT grabImg
'常用颜色变量
GLOBAL C_RED, C_GREEN, C_BLUE, C_YELLOW
C_RED = RGB(255,0,0)
C_GREEN = RGB(0,255,0)
C_BLUE = RGB(0,0,255)
C_YELLOW= RGB(255,255,0)
'数据码识别参数数组,依次为中心cx、cy、w、h、angle、data_code_type、step
GLOBAL DIM d_identfy_param(7) 'd开头表示数据结构
'识别消耗时间
GLOBAL DIM d_identfy_time
d_identfy_time = 0
'是否使用roi,0-不使用,不使用时就用全图进行识别数据码,1-使用时就用roi区域截取图像用来识别数据码
GLOBAL DIM d_useRoi
d_useRoi = 0
'条码类型
GLOBAL DIM code_type
code_type = 0
'识别结果,结果存储方式为:类型:结果,如EAN-13:123456789
GLOBAL DIM d_identfy_rst(256)
RUN 'Hmi.hmi',1
4.在HMI界面的元件中关联变量。
5.在identify.bas文件中初始化测量参数。
end
GLOBAL SUB init_param() '初始化测量参数
'初始化测量参数
d_identfy_param(0) = 320.0 'roi中心x
d_identfy_param(1) = 240.0 'roi中心y
d_identfy_param(2) = 160 'roi宽
d_identfy_param(3) = 120.0 'roi高
d_identfy_param(4) = 0.0 'roi角度
d_identfy_param(5) = 0 '条码类型为自动
d_identfy_param(6) = 4 '扫描步长
END SUB
6.在main.bas文件中添加初始化函数并在HMI编辑设置中关联函数名。
'HMI界面初始化函数
GLOBAL SUB hmi_init()
grab_switch = 0
main_task_state = 1
init_param() '初始化测量参数
ZV_RESETCLIPSIZE(1280, 1024) '初始化时依据图像分辨率设置区域的裁剪尺寸,此处图像分辨率为1280x1024
ZV_LATCHSETSIZE(0, HMI_CONTROLSIZEX(10, 5), HMI_CONTROLSIZEY(10, 5)) '设置锁存的大小
ZV_LATCHCLEAR(0)
'将检测测量器ROI的图像坐标数据转到控件坐标数据
TABLE(11, d_identfy_param(0), d_identfy_param(1))
ZV_POSFROMIMG(0, 1, 11, 11) '图像坐标转换到HMI控件坐标
TABLE(13) = ZV_LENFROMIMG(0, d_identfy_param(2))
TABLE(14) = ZV_LENFROMIMG(0, d_identfy_param(3))
TABLE(15) = d_identfy_param(4)
END SUB
7.在camera.bas文件中添加采集操作相关函数,并关联动作函数名。
end
'HMI界面按下扫描相机按钮时响应的函数
GLOBAL SUB cam_scan_all()
ZV_SETSYSINT('LogLevel', 7)
ZV_SETSYSSTR('DataDir','')
'扫描相机
CAM_SCAN(CAMERA_TYPE)
cam_num = CAM_COUNT()
?'cam_num = ' cam_num
if (0 = cam_num) then
?'未找到相机'
' ZV_READIMAGE(grabImg, 'QR.png', 1)
return
endif
'扫描到有相机就对一些相机参数进行设置
if cam_num > 0 then
CAM_SEL(0)
CAM_SETEXPOSURE(5000)
CAM_SETPARAM('GevSCPD', '3000')
CAM_SETPARAM('GevHeartbeatTimeout', '3000')
CAM_SETMODE(0)'设置触发模式为软触发
CAM_START(0)'开始采集
endif
END SUB
'HMI界面按下单次采集按钮时响应的函数
GLOBAL SUB btn_grab()
if cam_num=0 then
?'请先扫描相机!'
return
endif
CAM_SETPARAM('TriggerSoftware', 0)
CAM_GET(grabImg, 0)
ZV_LATCH(grabImg, 0) '将带显示的图像转换到锁存通道指定的锁存区域
END SUB
'HMI界面按下连续采集按钮时响应的函数
GLOBAL SUB btn_mea_cgrab()
if cam_num=0 then
?'请先扫描相机!'
return
endif
grab_switch = 1
if (1 = grab_switch) then
if (0 = PROC_STATUS(grab_task_id)) then
RUNTASK grab_task_id, grab_task
endif
endif
END SUB
'HMI界面按下停止采集按钮时响应的函数
GLOBAL SUB btn_mea_stopCgrab()
grab_switch = 0
END SUB
'连续采集任务
grab_task:
while(1)
if (0 = grab_switch) then
exit while
endif
btn_grab()
wend
END
8.在draw.bas文件中添加更新绘制Roi函数,并在自定义元件属性窗口关联刷新函数和绘图函数。
end
'和绘制(即选择ROI)有关的界面的刷新绘制函数放在这个bas文件里
DIM is_redraw
is_redraw = 0
DIM set_roi_open_init
set_roi_open_init = 0
DIM sr_mpos_x, sr_mpos_y, hit_pos
'根据鼠标操作更新检测区域ROI的坐标位置和形状大小
GLOBAL SUB update_identfy()
if mouse_scan(21) = 1 then '扫描按下操作
hit_pos = ZV_HMIADJRECT2(table(21), table(22), 11, -1) '只有按下时可以改变击中位置
is_redraw = 1
endif
if mouse_scan(21) = -1 then '扫描松开操作
ZV_HMIADJRECT2(table(21), table(22), 11, hit_pos)
is_redraw = 1
endif
if (MOUSE_state(21)) then
ZV_HMIADJRECT2(table(21), table(22), 11, hit_pos)
is_redraw = 1
endif
if (1 = is_redraw) then
'控件roi坐标转图像roi坐标
is_redraw = 0
ZV_POSTOIMG(0,2, 11, 0) 'TABLE(0)作为中间变量临时使用
d_identfy_param(0) = TABLE(0)
d_identfy_param(1) = TABLE(1)
d_identfy_param(2) = ZV_LENTOIMG(0, TABLE(2))
d_identfy_param(3) = ZV_LENTOIMG(0, TABLE(3))
d_identfy_param(4) = TABLE(4)
SET_REDRAW
endif
SET_REDRAW
END SUB
'更新ROI位置和大小后实时绘制ROI区域
GLOBAL SUB draw_identfy()
if d_useRoi =1 then
SET_COLOR(C_BLUE)
TABLE(16, 0, 0) '对子区域宽度和个数两个参数清零
ZV_HMIRECT2(11, 300)
DRAWLINE(TABLE(300), TABLE(301), TABLE(302), TABLE(303)) '外矩形
DRAWLINE(TABLE(302), TABLE(303), TABLE(304), TABLE(305))
DRAWLINE(TABLE(304), TABLE(305), TABLE(306), TABLE(307))
DRAWLINE(TABLE(306), TABLE(307), TABLE(300), TABLE(301))
DRAWLINE(TABLE(308), TABLE(309), TABLE(310), TABLE(311)) '方向箭头
DRAWLINE(TABLE(312), TABLE(313), TABLE(310), TABLE(311))
DRAWLINE(TABLE(314), TABLE(315), TABLE(310), TABLE(311))
endif
END SUB
9.添加在HMI界面按下【测试】按钮时响应的函数,并关联动作函数名。
'HMI界面按下测试按钮时响应的函数
GLOBAL SUB btn_identfy_test()
'开始识别
TICKS = 0
DIM tmp1(64),tmp2(64)
ZVOBJECT grayImg, codeList, codeRst
if ZV_IMGCNS(grabImg) > 1 then '获取图像通道数,单通道表示灰度图
ZV_RGBTOGRAY(grabImg,grayImg)
else
ZV_COPY(grabImg,grayImg) '复制grabImg图像到grayImg图像中
endif
code_type = d_identfy_param(5)
if code_type = 7 then '如果在界面中选择QR码类型
code_type = 20
elseif code_type = 8 then '如果在界面中选择DM码类型
code_type = 21
endif
ZV_CLEAR(codeList)
ZV_CODEREAD(grayImg,codeList,code_type,d_identfy_param(6))
if ZV_LISTCOUNT(codeList) > 0 then '获取列表中元素的数量
ZV_LISTGET(codeList,codeRst,0) '取出第一个条码结果作为显示
ZV_CODETYPESTR(codeRst,64,100) '获取数据码类型并将其存入起始索引为100的TABLE中
DMCPY tmp1(0),TABLE(100),64 '将TABLE中的数组拷贝至tmp1中
ZV_CODESTR(codeRst,64,100) '获取数据码结果并将其存入起始索引为100的TABLE中
DMCPY tmp2(0), TABLE(100), 64 '将TABLE中的数组拷贝至tmp2中
d_identfy_rst = tmp1 + ':'tmp2 '显示识别结果为 数据码类型:数据码结果
else
d_identfy_rst = 'identify fail!'
endif
d_identfy_time = abs(TICKS) '识别时间
END SUB
10.添加在HMI界面按下【运行】按钮时响应的函数,并关联动作函数名。
'HMI界面按下运行按钮时响应的函数
GLOBAL SUB btn_run()
if (1 = main_task_state) then
if (0 = PROC_STATUS(main_task_id)) then
main_task_state = 2
RUNTASK main_task_id, main_task
endif
endif
END SUB
'主任务执行的函数
main_task:
while(1)
if (3 = main_task_state) then
main_task_state = 1
exit while
endif
if cam_num = 0 then
btn_stop()
return
endif
'持续采集图像,对图像进行操作
btn_grab()
btn_identfy_test()
wend
END
11.添加在HMI界面按下【停止】按钮时响应的函数,并关联动作函数名。
'HMI界面按下停止按钮时响应的函数
GLOBAL SUB btn_stop()
if (2 = main_task_state) then
main_task_state = 3
endif
END SUB