HSmartWindowControl 之 攝像頭實時顯示( 使用 WPF )

来源:https://www.cnblogs.com/oucsheep/archive/2018/09/11/9626923.html
-Advertisement-
Play Games

1、添加Halcon控制項,創建WPF項目在VS2013中創建一個WPF工程,然後添加halcon的控制項和工具包,參見:HSmartWindowControl之安裝篇 (Visual Studio 2013 & Halcon 18)在WPF工程中添加好HSmartWindowControlWPF控制項後... ...


1、添加Halcon控制項,創建WPF項目

在VS2013中創建一個WPF工程,然後添加halcon的控制項和工具包,參見:

HSmartWindowControl之安裝篇 (Visual Studio 2013 & Halcon 18)

在WPF工程中添加好HSmartWindowControlWPF控制項後,將其拖入主窗體即可。

2、生成攝像頭實時顯示的halcon代碼

使用Image Acquisition 連接筆記本自帶的攝像頭,然後生成實時顯示的代碼即可:

* Image Acquisition 01: Code generated by Image Acquisition 01
open_framegrabber ('DirectShow', 1, 1, 0, 0, 0, 0, 'default', 8, 'rgb', -1, 'false', 'default', '[0] Integrated Camera', 0, -1, AcqHandle)
grab_image_start (AcqHandle, -1)
while (true)
    grab_image_async (Image, AcqHandle, -1)
    * Image Acquisition 01: Do something
endwhile

3、導出C#代碼

這裡主要關註action函數:

// Main procedure 
  private void action()
  {
    // Local iconic variables 
    HObject ho_Image=null;
    // Local control variables 
    HTuple hv_AcqHandle = new HTuple();
    // Initialize local and output iconic variables 
    HOperatorSet.GenEmptyObj(out ho_Image);
    //Image Acquisition 01: Code generated by Image Acquisition 01
    //Image Acquisition 01: Attention: The initialization may fail in case parameters need to
    //Image Acquisition 01: be set in a specific order (e.g., image resolution vs. offset).
    hv_AcqHandle.Dispose();
    HOperatorSet.OpenFramegrabber("DirectShow", 1, 1, 0, 0, 0, 0, "default", 8, "rgb", 
        -1, "false", "default", "[0] Integrated Camera", 0, -1, out hv_AcqHandle);

    HOperatorSet.GrabImageStart(hv_AcqHandle, -1);
    while ((int)(1) != 0)
    {
      ho_Image.Dispose();
      HOperatorSet.GrabImageAsync(out ho_Image, hv_AcqHandle, -1);
      //Image Acquisition 01: Do something
    }
    ho_Image.Dispose();

    hv_AcqHandle.Dispose();

  }

這段代碼只是實現了採集圖像,沒有在界面上顯示。

4、向VS項目添加halcon導出的c#代碼

向action()的代碼中添加在視窗上顯示圖像的代碼,即使用DispObj函數。然後把action添加到按鈕響應中執行。

這樣點擊按鈕之後就會進入死迴圈不停的GrabImage和DispObj,但是實際操作時界面會停止響應,而且圖像並不會在視窗內刷新。

5、添加實時顯示子線程

主線程中運行迴圈無法實時顯示,必須創建子線程。

這裡使用了工作者線程,點擊按鈕後,啟動線程並執行一個迴圈顯示採集的圖像;再次點擊按鈕時,跳出迴圈結束worker線程。然後繼續點擊按鈕又可以進入子線程開啟實時顯示,實現隨意啟停。

private void Button1_Click(object sender, RoutedEventArgs e)
        {
            if (isCameraOpened == false)
            {
                Button1.Content = "關閉實時顯示";
                isCameraOpened = true;
                play_Thread = new Thread(action);
                play_Thread.Start();
            }
            else
            {
                Button1.Content = "開啟實時顯示";
                isCameraOpened = false;
            }            
        }

註意調用線程需添加引用:

using System.Threading;//多線程

需要註意的是,為了防止在實時顯示過程中突然關閉窗體導致程式崩潰,在窗體的Un_load事件中加入強行停止子線程的代碼。

private void Window_Unloaded(object sender, RoutedEventArgs e)
        {
            if(play_Thread.IsAlive)
            {
                play_Thread.Abort();
            }
        }

6、調整圖像尺寸以適應halconwindow視窗

(1)使用HSmartWindowControlWPF控制項時,無需再添加滑鼠滾輪的響應函數便可使用滑鼠滾輪控製圖像縮放。

(2)顯示單張圖像時,可以直接向HSmartWindowControlWPF控制項的HDisplayCurrentObject屬性賦HObject類型的圖像即可自動調整圖像尺寸來適應視窗大小。

hSmartWindowControl1.HDisplayCurrentObject = new HImage("girl.png");

但是這個方式不能適用於動態捕獲的圖像,所以動態捕獲的圖像還需要手動判斷來進行縮放。調整的方式比較簡單,就是獲取視窗尺寸和圖像尺寸進行對比,根據圖像過寬還是過高來縮放。

7、源代碼

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
using HalconDotNet;
using System.Threading;//多線程

namespace HalconWindowWPFDemo
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        #region 全局變數
        private static MainWindow mainWindow = new MainWindow();
        private static HWindow hwindow; //全局視窗變數
        Thread play_Thread; //實時顯示線程
        private static bool isCameraOpened; //是否正在實時顯示
        private static double halconWindowHeight; //halcon視窗高度
        private static double halconWindowWidth; ////halcon視窗寬度
        #endregion

        #region 初始化
        public MainWindow()
        {
            InitializeComponent();           
        }

        private void Window_Loaded(object sender, RoutedEventArgs e)
        {
            hwindow = hSmartWindowControl1.HalconWindow;//初始化視窗變數          
            isCameraOpened = false;
            //獲取halcon視窗大小
            halconWindowHeight = mainWindow.hSmartWindowControl1.WindowSize.Width;
            halconWindowWidth = mainWindow.hSmartWindowControl1.WindowSize.Height;      
        }
        #endregion

        #region 界面響應
        /// <summary>
        /// 開啟和關閉實時顯示按鈕
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void Button1_Click(object sender, RoutedEventArgs e)
        {
            if (isCameraOpened == false)
            {
                Button1.Content = "關閉實時顯示";
                isCameraOpened = true;
                play_Thread = new Thread(action);
                play_Thread.Start();
            }
            else
            {
                Button1.Content = "開啟實時顯示";
                isCameraOpened = false;
            }
            
        }
        /// <summary>
        /// 顯示一張圖片
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void Button2_Click(object sender, RoutedEventArgs e)
        {
            hSmartWindowControl1.HDisplayCurrentObject = new HImage("girl.png");
        }
        #endregion

        #region 實時顯示用函數
        /// <summary>
        /// 攝像頭實時顯示
        /// </summary>
        private static void action()
        {
            // Local iconic variables 
            HObject ho_Image = null;
            // Local control variables 
            HTuple hv_AcqHandle = new HTuple();
            // Initialize local and output iconic variables 
            HOperatorSet.GenEmptyObj(out ho_Image);
            ho_Image.Dispose();
            hv_AcqHandle.Dispose();
            //Image Acquisition 01: Code generated by Image Acquisition 01          
            HOperatorSet.OpenFramegrabber("DirectShow", 1, 1, 0, 0, 0, 0, "default", 8, "rgb",
                -1, "false", "default", "[0] Integrated Camera", 0, -1, out hv_AcqHandle);
            HOperatorSet.GrabImageStart(hv_AcqHandle, -1);                 

            while (isCameraOpened)
            {            
                HOperatorSet.GrabImageAsync(out ho_Image, hv_AcqHandle, -1);
                if (!mainWindow.ShowHalconImage(ho_Image)) break;
            }
            ho_Image.Dispose();
            hv_AcqHandle.Dispose();
        }
        /// <summary>
        /// 顯示圖像
        /// </summary>
        /// <param name="image"></param>
        /// <returns></returns>
        private bool ShowHalconImage(HObject ho_Image)
        {           
            #region 縮放圖像,適應視窗           
            //獲取圖像大小及縱橫比
            HTuple hv_Width = new HTuple(), hv_Height = new HTuple();
            HOperatorSet.GetImageSize(ho_Image, out hv_Width, out hv_Height);
            int im_width = int.Parse(hv_Width.ToString());
            int im_height = int.Parse(hv_Height.ToString());
            double im_AspectRatio = (double)(im_width) / (double)(im_height);
            //獲取視窗大小及縱橫比        
            double w_AspectRatio = halconWindowWidth / halconWindowHeight;

            HOperatorSet.SetSystem("int_zooming", "false");//圖像縮放之前最好將此參數設置為false.
            HTuple para = new HTuple("constant");
            HObject ho_zoomImage;
            HOperatorSet.GenEmptyObj(out ho_zoomImage);

            ho_zoomImage.Dispose();
            if (halconWindowWidth < im_width && im_AspectRatio > w_AspectRatio)
            {
                //超寬圖像               
                HOperatorSet.ZoomImageSize(ho_Image, out ho_zoomImage, halconWindowWidth, halconWindowWidth / im_AspectRatio, "constant");             
            }
            else if (halconWindowHeight < im_height && im_AspectRatio < w_AspectRatio)
            {
                //超高圖像                
                HOperatorSet.ZoomImageSize(ho_Image, out ho_zoomImage, halconWindowHeight * im_AspectRatio, halconWindowHeight, para);
            }
            #endregion

            #region 顯示圖像                   
            try
            {
                hwindow.SetPart(0, 0, -2, -2); 
                hwindow.DispObj(ho_zoomImage);
            }
            catch (Exception exp)
            {
                //MessageBox.Show(exp.ToString());
                return false;
            }
            #endregion
            return true;            
        }
        #endregion

        #region 析構函數
        /// <summary>
        /// 關閉窗體時推出子線程,防止崩潰
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void Window_Unloaded(object sender, RoutedEventArgs e)
        {
            if(play_Thread.IsAlive)
            {
                play_Thread.Abort();
            }
        }
        #endregion
    }
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 本來這篇文章上個月就該發佈了,但是因為忙 QuarkDoc 一直沒有時間整理,所以耽擱到今天,現在回歸正軌。 C# 5.0 雖然只引入了2個新關鍵詞:async和await。然而它大大簡化了非同步方法的編程。 在 線程池(threadPool)大致介紹了微軟在不同時期使用的不同的非同步模式,有3種 ...
  • 獲取代碼 非實時:一開始的寬度是多少就多少,拉寬了 獲取的寬度還是剛開始的 ,同理高度 this.Width this.Height 獲取代碼 實時:調了窗體高度寬度,寬度 高度 也跟著變化,不再保持開始的寬 、高數據 int w =this.ClientSize.Width; int h = th ...
  • 結構體概念 在C#中,結構體是值類型,一般適用於表示類似Point、Rectangle、Color的對象 值類型能夠降低對堆的管理、使用。降低垃圾回收,表現出更好的性能。可是值類型也有不好的一面。會涉及到裝箱拆箱等操作 結構體聲明 結構體聲明定義了一種新的數據類型,這個數據類型可以為程式包含一個以上 ...
  • 在PHP開發項目中,後臺管理因為面向群體相對比較固定,大部分以實現業務邏輯和功能。使用Bootstrap後臺模板可以讓後端開發很輕鬆的就展現給客戶一個響應式的後臺,節約前端開發的時間。下麵PHP程式員雷雪松給大家分享一下最值得擁有的免費Bootstrap後臺管理模板。 AdminLTE AdminL ...
  • 一、先看看效果 二、原理 1、選項卡大小和位置 這次給大家介紹的控制項是比較常用的TabControl,網上常見的TabControl樣式有很多,其中一部分也支持拖動選項卡,但是帶動畫效果的很少見。這也是有原因的,因為想要做一個不失原有功能,還需要添加動畫效果的控制項可不是一行代碼的事。要做成上圖中的效 ...
  • 紙殼CMS是可視化建站系統,現已經從架構上支持多語言。但是多語言功能預設是沒有開啟的。您可以從設置中開啟多語言,或者隨時關閉它,您可以隨時進行切換。 ...
  • 業務描述 當用戶執行完業務操作,或者數據操作後,講業務記錄/數據追蹤插入到Redis中。ThreadPool.QueueUserWorkItem定時檢查隊列並將上述數據插入到資料庫中持久化。 實現流程 1、RedisHelp的實現 其中,Redsion是用StackExchange.Redis來實現 ...
  • 引言 上半年使用的thinkjs開發的node項目有一個優點就是所有請求都會有日誌記錄在控制台輸出,裡面包含了請求地址以及耗時。我就希望在.net中也可以實現這樣子的功能,正好想到了中間件,所以就用中間件來實現該功能。 實現 該功能主要分為兩部分:一個是IApplicationBuilder的拓展實 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...