資料介紹
描述
介紹
增強(qiáng)現(xiàn)實(shí) (AR) 是真實(shí)世界環(huán)境的實(shí)時(shí)視圖,已通過計(jì)算機(jī)生成的圖形進(jìn)行了增強(qiáng)。AR有兩種實(shí)現(xiàn)方式:
- 基于標(biāo)記的增強(qiáng)現(xiàn)實(shí)
- 無標(biāo)記 AR
Marker-less AR 更加復(fù)雜,并且依賴于真實(shí)環(huán)境來確定參考點(diǎn)。這通常涉及識(shí)別物體和/或背景線索,例如地板和墻壁。
基于標(biāo)記的 AR 更易于實(shí)現(xiàn),并且依賴于場(chǎng)景中的方形標(biāo)記。
該項(xiàng)目將描述如何使用 OpenCV 在 Ultra96-V2 上實(shí)現(xiàn)基于標(biāo)記的 AR。
讓我們開始吧 !
靈感
在開始之前,我想分享一下這個(gè)項(xiàng)目的靈感和動(dòng)力。
我受到以下創(chuàng)新產(chǎn)品/教程的啟發(fā):
動(dòng)機(jī)
我實(shí)現(xiàn)類似功能的動(dòng)機(jī)是能夠使用標(biāo)記自動(dòng)觸發(fā)某種校準(zhǔn),例如:
- 白平衡,使用白色參考圖表
- 立體校準(zhǔn),使用棋盤參考圖
為了實(shí)現(xiàn)這一點(diǎn),我創(chuàng)建了以下三個(gè)圖表(使用 Microsoft Word)進(jìn)行實(shí)驗(yàn)。
在這項(xiàng)目中,我會(huì)檢測(cè)這些圖表的存在,并根據(jù)圖表進(jìn)行額外的處理:
- 圖表 1 - 在棋盤圖案周圍畫一個(gè)綠色框
- 圖表 2 - 測(cè)量標(biāo)記內(nèi)區(qū)域中 B、GR 像素的平均值,并在圖表上顯示帶有值的條形圖
- 圖表 2 - 測(cè)量標(biāo)記內(nèi)區(qū)域的顏色直方圖,并在圖表上顯示直方圖
有關(guān)生成這些標(biāo)記的更多信息,請(qǐng)參閱以下優(yōu)秀教程:
- Adrian Rosebrock,使用 OpenCV 和 Python 生成 ArUco 標(biāo)記,PyImageSearch,
檢測(cè)場(chǎng)景中的標(biāo)記
定義好目標(biāo)并創(chuàng)建帶有標(biāo)記的圖表后,我們就可以開始實(shí)施了。
我選擇用 C++ 實(shí)現(xiàn)這個(gè)項(xiàng)目,作為一個(gè) gstreamer 插件。我需要感謝Tom Simpson為創(chuàng)建 gstreamer 插件奠定了基礎(chǔ)。
第一步是檢測(cè)場(chǎng)景中的標(biāo)記。這是通過 OpenCV 完成的。
/* Aruco Markers */
#include
...
//
// Detect ARUCO markers
// ref : https://docs.opencv.org/master/d5/dae/tutorial_aruco_detection.html
//
std::vector markerIds;
std::vector<std::vector<cv::Point2f>> markerCorners, rejectedCandidates;
cv::Ptr<cv::aruco::DetectorParameters> parameters = cv::aruco::DetectorParameters::create();
cv::Ptr<cv::aruco::Dictionary> dictionary = cv::aruco::getPredefinedDictionary(cv::aruco::DICT_ARUCO_ORIGINAL);
cv::aruco::detectMarkers(img, dictionary, markerCorners, markerIds, parameters, rejectedCandidates);
if ( markerIds.size() > 0 )
{
cv::aruco::drawDetectedMarkers(img, markerCorners, markerIds);
}
此時(shí),您會(huì)注意到我正在使用 DICT_ARUCO_ORIGINAL 系列標(biāo)記。標(biāo)記包含一個(gè)由黑色邊框包圍的 5x5 方格。

對(duì)于每個(gè)檢測(cè)到的標(biāo)記,OpenCV API 返回以下信息:
- markerIds : 每個(gè)標(biāo)記的標(biāo)識(shí)
- markerCorners : 每個(gè)標(biāo)記的坐標(biāo),格式如下: [0] top-left [1] top-right [2] bottom-right [3] bottom-left

下表和圖像說明了我創(chuàng)建的圖表中使用了哪些標(biāo)記。

我使用 switch case 狀態(tài)掃描檢測(cè)到的標(biāo)記,并識(shí)別感興趣的標(biāo)記。
...
if (markerIds.size() >= 4 )
{
int tl_id = 0;
int tr_id = 0;
int bl_id = 0;
int br_id = 0;
cv::Point2f tl_xy, tr_xy, bl_xy, br_xy;
for ( unsigned i = 0; i < markerIds.size(); i++ )
{
switch ( markerIds[i] )
{
case 923:
tl_id = markerIds[i];
tl_xy = markerCorners[i][3]; // bottom left corner of top left marker
break;
case 1001:
case 1002:
case 1003:
case 1004:
case 1005:
case 1006:
tr_id = markerIds[i];
tr_xy = markerCorners[i][2]; // bottom right corner of top right marker
break;
case 1007:
bl_id = markerIds[i];
bl_xy = markerCorners[i][0]; // top left corner of bottom left marker
break;
case 241:
br_id = markerIds[i];
br_xy = markerCorners[i][1]; // top right corner of bottom right marker
break;
default:
break;
}
}
...
}
...
下圖說明了我保留哪些 ID 和 X/Y 坐標(biāo)來定義感興趣區(qū)域 (ROI) 以進(jìn)行附加處理。

檢測(cè)特定圖表是通過以下簡(jiǎn)單的條件語句完成的。
// Chart 1 - Checkerboard (9x7)
if ( (tl_id==923) && (tr_id==1001) && (bl_id==1007) && (br_id==241) )
{
...
}
// Chart 2 - White Reference
if ( (tl_id==923) && (tr_id==1002) && (bl_id==1007) && (br_id==241) )
...
}
{
// Chart 3 - Histogram
if ( (tl_id==923) && (tr_id==1003) && (bl_id==1007) && (br_id==241) )
{
...
}
將計(jì)算機(jī)生成的圖形添加到場(chǎng)景中
對(duì)于“Chart 1 - CheckerBoard (9x7)”,繪制了一個(gè)綠色矩形來標(biāo)識(shí)感興趣的區(qū)域。
// Extract ROI (area, ideally within 4 markers)
std::vector<cv::Point> polygonPoints;
polygonPoints.push_back(cv::Point(tl_xy.x,tl_xy.y));
polygonPoints.push_back(cv::Point(tr_xy.x,tr_xy.y));
polygonPoints.push_back(cv::Point(br_xy.x,br_xy.y));
polygonPoints.push_back(cv::Point(bl_xy.x,bl_xy.y));
// Draw border around "checkerboard"
cv::polylines(img, polygonPoints, true, cv::Scalar (0, 255, 0), 2, 16);
對(duì)于“圖表 2 - 白色參考”,感興趣區(qū)域用于計(jì)算藍(lán)色 (B)、綠色 (G) 和紅色 (R) 分量中的每一個(gè)的平均值。下面的代碼使用了一個(gè)掩碼,它支持一個(gè)不是完美矩形的 ROI。
//
// Calculate color gains
// ref : https://stackoverflow.com/questions/32466616/finding-the-average-color-within-a-polygon-bound-in-opencv
//
cv::Point pts[1][4];
pts[0][0] = cv::Point(tl_xy.x,tl_xy.y);
pts[0][1] = cv::Point(tr_xy.x,tr_xy.y);
pts[0][2] = cv::Point(br_xy.x,br_xy.y);
pts[0][3] = cv::Point(bl_xy.x,bl_xy.y);
const cv::Point* points[1] = {pts[0]};
int npoints = 4;
// Create the mask with the polygon
cv::Mat1b mask(img.rows, img.cols, uchar(0));
cv::fillPoly(mask, points, &npoints, 1, cv::Scalar(255));
// Calculate mean in masked area
auto bgr_mean = cv::mean( img, mask );
double b_mean = bgr_mean(0);
double g_mean = bgr_mean(1);
double r_mean = bgr_mean(2);
計(jì)算出平均值后,將創(chuàng)建一個(gè) plotImage,其中包含一個(gè)帶有 B、G 和 R 像素平均值的條形圖圖像。
// Draw bars
int plot_w = 100, plot_h = 100;
cv::Mat plotImage( plot_h, plot_w, CV_8UC3, cv::Scalar(255,255,255) );
int b_bar = int((b_mean/256.0)*80.0);
int g_bar = int((g_mean/256.0)*80.0);
int r_bar = int((r_mean/256.0)*80.0);
// layout of bars : |<-10->|<---20-->|<-10->|<---20-->|<-10->|<---20-->|<-10->|
cv::rectangle(plotImage, cv::Rect(10,(80-b_bar),20,b_bar), cv::Scalar(255, 0, 0), cv::FILLED, cv::LINE_8);
cv::rectangle(plotImage, cv::Rect(40,(80-g_bar),20,g_bar), cv::Scalar(0, 255, 0), cv::FILLED, cv::LINE_8);
cv::rectangle(plotImage, cv::Rect(70,(80-r_bar),20,r_bar), cv::Scalar(0, 0, 255), cv::FILLED, cv::LINE_8);
std::stringstream b_str;
std::stringstream g_str;
std::stringstream r_str;
b_str << int(b_mean);
g_str << int(g_mean);
r_str << int(r_mean);
cv::putText(plotImage, b_str.str(), cv::Point(10,90), cv::FONT_HERSHEY_PLAIN, 0.75, cv::Scalar(255,0,0), 1, cv::LINE_AA);
cv::putText(plotImage, g_str.str(), cv::Point(40,90), cv::FONT_HERSHEY_PLAIN, 0.75, cv::Scalar(0,255,0), 1, cv::LINE_AA);
cv::putText(plotImage, r_str.str(), cv::Point(70,90), cv::FONT_HERSHEY_PLAIN, 0.75, cv::Scalar(0,0,255), 1, cv::LINE_AA);

最后,使用以下 OpenCV 函數(shù)將此繪圖圖像扭曲到感興趣區(qū)域:
- findHomography :計(jì)算源坐標(biāo)和目標(biāo)坐標(biāo)之間的變換矩陣
- warpPerspective :將源圖像扭曲到目標(biāo)坐標(biāo)
然后使用掩碼將扭曲的條形圖圖像與實(shí)時(shí)圖像組合。
// Calculate transformation matrix
std::vector<cv::Point2f> srcPoints;
std::vector<cv::Point2f> dstPoints;
srcPoints.push_back(cv::Point( 0, 0)); // top left
srcPoints.push_back(cv::Point(plot_w-1, 0)); // top right
srcPoints.push_back(cv::Point(plot_w-1,plot_h-1)); // bottom right
srcPoints.push_back(cv::Point( 0,plot_h-1)); // bottom left
dstPoints.push_back(tl_xy);
dstPoints.push_back(tr_xy);
dstPoints.push_back(br_xy);
dstPoints.push_back(bl_xy);
cv::Mat h = cv::findHomography(srcPoints,dstPoints);
// Warp plot image onto video frame
cv::Mat img_temp = img.clone();
cv::warpPerspective(plotImage, img_temp, h, img_temp.size());
cv::Point pts_dst[4];
for( int i = 0; i < 4; i++)
{
pts_dst[i] = dstPoints[i];
}
cv::fillConvexPoly(img, pts_dst, 4, cv::Scalar(0), cv::LINE_AA);
img = img + img_temp;
對(duì)于“圖表 3 - 直方圖”,使用了與“圖表 2”類似的技術(shù)。不是顯示顏色平均值的條形圖,而是顯示每個(gè)顏色分量的直方圖。

了解了所有理論之后,是時(shí)候在 Ultra96-V2 上進(jìn)行嵌入式實(shí)現(xiàn)了。
第 0 步 - 打印圖表
為了運(yùn)行此示例,您將需要三個(gè)圖表,這些圖表已在“原理圖”部分以 PDF 格式提供。
- 圖 1 - 棋盤 (9x7)
- 圖表 2 - 白色參考
- 圖表 2 - 直方圖
第 1 步 - 創(chuàng)建 SD 卡
為以下 Avnet 平臺(tái)提供了預(yù)構(gòu)建的 Vitis-AI 1.3 SD 卡映像:
- u96v2_sbc_base : Ultra96-V2 開發(fā)板
- uz7ev_evcc_base:UltraZed-EV SOM (7EV) + FMC 載卡
- uz3eg_iocc_base:UltraZed-EG SOM (3EG) + IO 載卡
可在此處找到預(yù)構(gòu)建 SD 卡映像的下載鏈接:
- 適用于 Avnet Vitis 平臺(tái)的 Vitis-AI 1.3 流程:https ://avnet.me/vitis-ai-1.3-project
下載并解壓后,.img 文件可以編程到 16GB 微型 SD 卡。
0.解壓壓縮包得到.img文件
1. 將開發(fā)板特定的 SD 卡映像編程到 16GB(或更大)的 micro SD 卡
一個(gè)。在 Windows 機(jī)器上,使用 Balena Etcher 或 Win32DiskImager(免費(fèi)開源軟件)
灣。在 linux 機(jī)器上,使用 Balena Etcher 或使用 dd 實(shí)用程序
$ sudo dd bs=4M if=Avnet-{platform}-Vitis-AI-1-3-{date}.img of=/dev/sd{X} status=progress conv=fsync
其中 {X} 是一個(gè)小寫字母,用于指定 SD 卡的設(shè)備。您可以使用“df -h”來確定您的 SD 卡對(duì)應(yīng)的設(shè)備。
第 2 步 - 克隆源代碼存儲(chǔ)庫(kù)
本項(xiàng)目中使用的源代碼可以從以下存儲(chǔ)庫(kù)中獲取:
如果您有活動(dòng)的互聯(lián)網(wǎng)連接,您可以簡(jiǎn)單地將存儲(chǔ)庫(kù)克隆到嵌入式平臺(tái)的根目錄:
$ cd ~
$ git clone
第 3 步 - 編譯和安裝 gstreamer 插件
gstreamer 插件可以在 Ultra96-V2 嵌入式平臺(tái)上使用 make 命令構(gòu)建:
$ cd vitis_ai_gstreamer_plugins
$ cd markerdetect
$ make
編譯完成后,gstreamer 插件可以按如下方式安裝:
$ cp libgstmarkerdetect.so /usr/lib/gstreamer-1.0/.
可以使用 gst-inspect-1.0 實(shí)用程序驗(yàn)證 gstreamer 插件的安裝:
$ gst-inspect-1.0 | grep markerdetect
markerdetect: markerdetect: Marker detection using the OpenCV Library
$ gst-inspect-1.0 markerdetect
Factory Details:
Rank none (0)
Long-name Marker detection using the OpenCV Library
Klass Video Filter
Description Marker Detection
Author FIXME
Plugin Details:
Name markerdetect
Description Marker detection using the OpenCV Library
Filename /usr/lib/gstreamer-1.0/libgstmarkerdetect.so
Version 0.0.0
License LGPL
Source module markerdetect
Binary package OpenCV Library
Origin URL http://avnet.com
GObject
+----GInitiallyUnowned
+----GstObject
+----GstElement
+----GstBaseTransform
+----GstVideoFilter
+----GstMarkerDetect
Pad Templates:
SRC template: 'src'
Availability: Always
Capabilities:
video/x-raw
format: { (string)BGR }
width: [ 1, 1920 ]
height: [ 1, 1080 ]
framerate: [ 0/1, 2147483647/1 ]
SINK template: 'sink'
Availability: Always
Capabilities:
video/x-raw
format: { (string)BGR }
width: [ 1, 1920 ]
height: [ 1, 1080 ]
framerate: [ 0/1, 2147483647/1 ]
Element has no clocking capabilities.
Element has no URI handling capabilities.
Pads:e--
SINK: 'sink'
Pad Template: 'sink'
SRC: 'src'
Pad Template: 'src'
Element Properties:
name : The name of the object
flags: readable, writable
String. Default: "markerdetect0"
parent : The parent of the object
flags: readable, writable
Object of type "GstObject"
qos : Handle Quality-of-Service events
flags: readable, writable
Boolean. Default: true
第 4 步 - 使用實(shí)時(shí)視頻源執(zhí)行示例
為了便于啟動(dòng)示例,請(qǐng)?jiān)谀那度胧狡脚_(tái)上創(chuàng)建以下啟動(dòng)腳本 (launch_usb_markerdetect.sh):
#!/bin/sh
gst-launch-1.0 \
v4l2src device=/dev/video0 io-mode=4 ! \
video/x-raw, width=640, height=480, format=YUY2, framerate=30/1 ! \
videoconvert ! \
video/x-raw, format=BGR ! \
queue ! markerdetect ! queue ! \
videoconvert ! \
fpsdisplaysink sync=false text-overlay=false fullscreen-overlay=true \
\
-v
在啟動(dòng)示例之前,我們要定義我們的 DISPLAY 環(huán)境變量,并配置我們的 DisplayPort 顯示器的分辨率。
$ export DISPLAY=:0.0
$ xrandr --output DP-1 --mode 800x600
該示例可以使用我們剛剛創(chuàng)建的腳本啟動(dòng):
$ ./launch_usb_markerdetect.sh
我第一次使用 USB 攝像頭執(zhí)行這個(gè)“markerdetect”gstreamer 插件讓我感到驚訝。我使用的 USB 相機(jī)是具有自動(dòng)白平衡功能的羅技 C720,所以我希望看到藍(lán)色、綠色和紅色的平均值大致相同。
事實(shí)證明,DisplayPort 監(jiān)視器正在生成一種藍(lán)色色調(diào),該色調(diào)被圖表拾取,并略微扭曲了結(jié)果。

我跑了同樣的測(cè)試,這次遠(yuǎn)離顯示器,結(jié)果更符合我的預(yù)期。
這一次,藍(lán)色、綠色和紅色的平均值大致相同。

結(jié)論
我希望本教程能激發(fā)您在 Ultra96-V2 上嘗試增強(qiáng)現(xiàn)實(shí) (AR)。
你能想到其他應(yīng)用程序會(huì)使用這些類型的標(biāo)記嗎?
如果您還想看到任何其他相關(guān)內(nèi)容,請(qǐng)?jiān)谙旅娴脑u(píng)論中分享您的想法。
?
- Ultra96硬件用戶指南
- Ultra96 CSI-2視頻輸出到Raspberry Pi攝像頭輸入
- Ultra96上的實(shí)時(shí)攝像頭饋送網(wǎng)頁
- 使用PYNQ的Ultra96面部識(shí)別鎖栓
- 使用Tensil、TF-Lite和PYNQ在Ultra96板上運(yùn)行YOLO v4 Tiny
- Ultra96-V2上的頭部姿勢(shì)估計(jì)
- 在Ultra96 V2平臺(tái)上用Python實(shí)現(xiàn)人臉檢測(cè)和人臉跟蹤
- 用于Ultra96的夾層板96AnalogXperience
- Ultra96 FPGA上的Live NYC Subway Monitor應(yīng)用程序
- 關(guān)于Ultra96的Xilinx DDS編譯器IP教程
- 與Ultra96聯(lián)網(wǎng)端口轉(zhuǎn)發(fā)
- 使用Ultra96 PYNQ測(cè)定織物GSM
- Ultra96皮膚癌AI構(gòu)建
- 2018.2 Ultra96:從 Matchbox 桌面關(guān)斷 PetaLinux BSP,無法關(guān)斷電路板
- 一起玩Ultra96之GPIO操作
- 大模型系列:Flash Attention V2整體運(yùn)作流程 2112次閱讀
- PCB上那些米色標(biāo)記/圖標(biāo)/符號(hào)是什么?如何將其去除? 1131次閱讀
- PINE64便攜烙鐵Pinecil V2版本 3674次閱讀
- 虛擬現(xiàn)實(shí)和增強(qiáng)現(xiàn)實(shí)存在什么潛在的問題 3212次閱讀
- dfrobotGravity: 模擬pH計(jì)V2介紹 2195次閱讀
- dfrobotGravity:模擬電導(dǎo)率計(jì)V2 (K=1)介紹 1599次閱讀
- 微雪電子ST-LINK/V2 STM32仿真器簡(jiǎn)介 4548次閱讀
- 微雪電子迷你ST-LINK/V2 STM32仿真器 3191次閱讀
- 微雪電子ST-LINK/V2 STM32仿真器簡(jiǎn)介 6361次閱讀
- 亞太天能科技V2指紋鎖簡(jiǎn)介 1813次閱讀
- 微雪電子 樹莓派原裝攝像頭V2兼容模組介紹 2199次閱讀
- 微雪電子樹莓派原裝攝像頭V2簡(jiǎn)介 2753次閱讀
- 基于Arm技術(shù)的16nm MPSoC開發(fā)套件Ultra96 6201次閱讀
- 增強(qiáng)現(xiàn)實(shí)技術(shù)的應(yīng)用分析 8745次閱讀
- 增強(qiáng)現(xiàn)實(shí)技術(shù)和vr差異在哪 1850次閱讀
下載排行
本周
- 1晶體三極管的電流放大作用詳細(xì)說明
- 0.77 MB | 30次下載 | 2 積分
- 2雙極型三極管放大電路的三種基本組態(tài)的學(xué)習(xí)課件免費(fèi)下載
- 4.03 MB | 23次下載 | 1 積分
- 3多級(jí)放大電路的學(xué)習(xí)課件免費(fèi)下載
- 1.81 MB | 20次下載 | 2 積分
- 4九陽豆?jié){機(jī)高清原理圖
- 2.47 MB | 8次下載 | 1 積分
- 5數(shù)能NU501系列恒流芯片型號(hào)選型表
- 0.04 MB | 2次下載 | 免費(fèi)
- 6AIWA J470維修手冊(cè) (可直接打印)
- 3.16 MB | 2次下載 | 10 積分
- 7基于51/32單片機(jī)按鍵控制步進(jìn)電機(jī)正反轉(zhuǎn),A4988驅(qū)動(dòng)
- 2.05 MB | 2次下載 | 1 積分
- 8STM32英文參考手冊(cè)
- 13.17 MB | 1次下載 | 免費(fèi)
本月
- 1晶體三極管的電流放大作用詳細(xì)說明
- 0.77 MB | 30次下載 | 2 積分
- 2雙極型三極管放大電路的三種基本組態(tài)的學(xué)習(xí)課件免費(fèi)下載
- 4.03 MB | 23次下載 | 1 積分
- 3第十一屆 藍(lán)橋杯 單片機(jī)設(shè)計(jì)與開發(fā)項(xiàng)目 省賽 程序設(shè)計(jì)試題及源碼
- 90.88KB | 23次下載 | 3 積分
- 4Python從入門到精通背記手冊(cè)
- 18.77 MB | 22次下載 | 1 積分
- 5多級(jí)放大電路的學(xué)習(xí)課件免費(fèi)下載
- 1.81 MB | 20次下載 | 2 積分
- 6STM32F103C8T6開發(fā)板最小系統(tǒng)原理圖
- 0.78 MB | 18次下載 | 免費(fèi)
- 7SX1308應(yīng)用電路圖與SX1308升壓電路圖
- 0.18 MB | 8次下載 | 1 積分
- 8東芝彩色電視機(jī)29SF6C維修說明書
- 4.86 MB | 8次下載 | 1 積分
總榜
- 1matlab軟件下載入口
- 未知 | 935127次下載 | 10 積分
- 2開源硬件-PMP21529.1-4 開關(guān)降壓/升壓雙向直流/直流轉(zhuǎn)換器 PCB layout 設(shè)計(jì)
- 1.48MB | 420063次下載 | 10 積分
- 3Altium DXP2002下載入口
- 未知 | 233089次下載 | 10 積分
- 4電路仿真軟件multisim 10.0免費(fèi)下載
- 340992 | 191382次下載 | 10 積分
- 5十天學(xué)會(huì)AVR單片機(jī)與C語言視頻教程 下載
- 158M | 183337次下載 | 10 積分
- 6labview8.5下載
- 未知 | 81585次下載 | 10 積分
- 7Keil工具M(jìn)DK-Arm免費(fèi)下載
- 0.02 MB | 73814次下載 | 10 積分
- 8LabVIEW 8.6下載
- 未知 | 65988次下載 | 10 積分
評(píng)論