Microsoft MVP성태의 닷넷 이야기
.NET Framework: 710. C# - OpenCvSharp을 이용한 Webcam 영상 처리 + Direct2D [링크 복사], [링크+제목 복사],
조회: 32970
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 2개 있습니다.)
(시리즈 글이 10개 있습니다.)
.NET Framework: 707. OpenCV 응용 프로그램을 C#으로 구현 - OpenCvSharp
; https://www.sysnet.pe.kr/2/0/11402

.NET Framework: 708. C# - OpenCvSharp을 이용한 동영상(avi, mp4, ...) 처리
; https://www.sysnet.pe.kr/2/0/11403

.NET Framework: 709. C# - OpenCvSharp을 이용한 동영상(avi, mp4, ...) 처리 + Direct2D
; https://www.sysnet.pe.kr/2/0/11404

.NET Framework: 710. C# - OpenCvSharp을 이용한 Webcam 영상 처리 + Direct2D
; https://www.sysnet.pe.kr/2/0/11405

.NET Framework: 711. C# - OpenCvSharp의 Mat 데이터 조작 방법
; https://www.sysnet.pe.kr/2/0/11406

.NET Framework: 723. C# - OpenCvSharp 사용 시 C/C++을 이용한 속도 향상 (for 루프 연산)
; https://www.sysnet.pe.kr/2/0/11422

VC++: 123. 내가 만든 코드보다 OpenCV의 속도가 월등히 빠른 이유
; https://www.sysnet.pe.kr/2/0/11423

.NET Framework: 781. C# - OpenCvSharp 사용 시 포인터를 이용한 속도 향상
; https://www.sysnet.pe.kr/2/0/11567

개발 환경 구성: 447. Visual Studio Code에서 OpenCvSharp 개발 환경 구성
; https://www.sysnet.pe.kr/2/0/11971

Graphics: 38. C# - OpenCvSharp.VideoWriter에 BMP 파일을 1초씩 출력하는 예제
; https://www.sysnet.pe.kr/2/0/12485




C# - OpenCvSharp을 이용한 Webcam 영상 처리 + Direct2D

OpenCV로 Webcam 영상 처리하는 것은 동영상 파일 처리 때와 거의 동일합니다.

C# - OpenCvSharp을 이용한 동영상(avi, mp4, ...) 처리 + Direct2D
; https://www.sysnet.pe.kr/2/0/11404

그 소스 코드에서 바뀌는 거라고는 VideoCapture 생성자에 전달하는 인자가 동영상 파일이 아닌, PC에 장착된 카메라를 식별하는 번호라는 점입니다.

VideoCapture capture = new VideoCapture("c:\\temp\\test.avi");

==>

VideoCapture capture = new VideoCapture(0);

단지, 0번으로 다소 애매하게 지정한다는 것이 좀 아쉽다면 UWP 라이브러리를 이용해 비디오 장치를 열람해 이름을 알아내는 것도 가능합니다.

데스크톱 윈도우 응용 프로그램에서 UWP 라이브러리를 이용한 비디오 장치 열람하는 방법
; https://www.sysnet.pe.kr/2/0/11284

그런데 문제가 하나 있다면, Webcam의 FPS 수치를 알 길이 없다는 것입니다. 동영상 파일과는 달리 capture.Fps 값이 0으로 나오기 때문인데, 검색해 보면 다음과 같이 직접 capture.Read를 일정 수만큼 반복해 대략 측정하는 방법이 나옵니다.

How to find frame rate or frames per second (fps) in OpenCV ( Python / C++ ) ?
; https://www.learnopencv.com/how-to-find-frame-rate-or-frames-per-second-fps-in-opencv-python-cpp/

참고로 위의 덧글에 보면 OpenCV 3.2부터 webcam에도 FPS가 동작한다고 하는데 OpenCvSharp 현재 버전이 OpenCV 3.2를 기반으로 하고 있는데도 동작하지 않는 거 보면 그다지 신뢰가 가지 않습니다.




아래의 코드는 지난번 "C# - OpenCvSharp을 이용한 동영상(avi, mp4, ...) 처리 + Direct2D" 글에서 VideoCapture(0)으로 바꾼 것과 함께 Direct2D Bitmap 하나에 2개의 Mat 영상을 함께 출력하는 변화를 포함합니다.

using OpenCvSharp;
using SharpDX;
using SharpDX.Direct2D1;
using System;
using System.Collections.Concurrent;
using System.Diagnostics;
using System.Threading;
using System.Windows.Forms;

namespace WindowsFormsApp1
{
    public partial class Form1 : Form
    {
        RenderTarget2D _renderTarget;
        Thread camera;
        BlockingCollection<SharpDX.Direct2D1.Bitmap> _q = new BlockingCollection<SharpDX.Direct2D1.Bitmap>();
        DataStream _bitmapData;

        public Form1()
        {
            InitializeComponent();
            _renderTarget = new RenderTarget2D();
        }

        protected override void OnFormClosing(FormClosingEventArgs e)
        {
            _renderTarget.Dispose();
            base.OnFormClosing(e);
        }

        protected override void OnPaintBackground(PaintEventArgs e)
        {
        }

        protected override void OnPaint(PaintEventArgs e)
        {
            base.OnPaint(e);

            while (true)
            {
                if (_q.TryTake(out SharpDX.Direct2D1.Bitmap item) == true)
                {
                    _renderTarget.Render(
                        (renderer) =>
                        {
                            this.Text = _q.Count.ToString();
                            renderer.DrawBitmap(item, 1.0f, BitmapInterpolationMode.Linear);
                        });
                    item.Dispose();
                }
                else
                {
                    break;
                }
            }
        }
        
        private void CaptureCameraCallback()
        {
            VideoCapture capture = new VideoCapture(0);

            if (capture.IsOpened() == false)
            {
                return;
            }

            this.Invoke((Action)(() =>
                {
                    this.Width = capture.FrameWidth;
                    this.Height = capture.FrameHeight * 2;
                    _renderTarget.Initialize(this.Handle, capture.FrameWidth, capture.FrameHeight * 2);
                }
            ));

            _bitmapData = new DataStream(capture.FrameWidth * capture.FrameHeight * 4 * 2, false, false);

            int expectedProcessTimePerFrame = 1000 / 30; // 30 fps
            Stopwatch st = new Stopwatch();
            st.Start();

            using (Mat image = new Mat())
            {
                while (true)
                {
                    long started = st.ElapsedMilliseconds;

                    capture.Read(image);
                    if (image.Empty() == true)
                    {
                        break;
                    }

                    using (Mat canny = image.Canny(50, 100))
                    {
                        SharpDX.Direct2D1.Bitmap bitmap = ToSharpDXBitmap(image, canny);
                        _q.Add(bitmap);
                    }

                    try
                    {
                        this.Invoke((Action)(() => this.Invalidate()));
                    }
                    catch (ObjectDisposedException) { }
                    catch (InvalidOperationException) { }

                    int elapsed = (int)(st.ElapsedMilliseconds - started);
                    int delay = expectedProcessTimePerFrame - elapsed;

                    if (delay > 0)
                    {
                        Thread.Sleep(delay);
                    }
                }
            }
        }

        private SharpDX.Direct2D1.Bitmap ToSharpDXBitmap(Mat org, Mat canny)
        {
            using (Mat orgData = org.CvtColor(ColorConversionCodes.BGR2BGRA))
            using (Mat cannyData = canny.CvtColor(ColorConversionCodes.GRAY2RGBA))
            {
                IntPtr dstPtr = _bitmapData.DataPointer;

                Utilities.CopyMemory(dstPtr, orgData.Data, (int)orgData.Total() * orgData.ElemSize());
                dstPtr = IntPtr.Add(dstPtr, (int)orgData.Total() * orgData.ElemSize());

                Utilities.CopyMemory(dstPtr, cannyData.Data, (int)cannyData.Total() * cannyData.ElemSize());

                return _renderTarget.CreateBitmap(_bitmapData);
            }
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            camera = new Thread(new ThreadStart(CaptureCameraCallback));
            camera.IsBackground = true;
            camera.Start();
        }
    }
}

따라서 실행해 보면 다음과 같이 화면의 높이를 이등분해서 위에는 원본 이미지, 아래에는 Canny 이미지가 나옵니다. (첨부 파일은 이 글의 예제 코드를 포함합니다.)

opencv_webcam_1.PNG




그러고 보니 webcam 관련해서 몇 번 다룬 적이 있군요. ^^

C# - AForge.NET을 이용한 웹캠 영상 출력
; https://www.sysnet.pe.kr/2/0/11085

Visual Studio 2013/2015 - Ceemple OpenCV 확장을 이용한 웹캠 영상 출력
; https://www.sysnet.pe.kr/2/0/11094




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 12/26/2021]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2022-02-17 04시19분
#OpenCV – Detect and blur faces 😁 using haar cascades in C#
; https://dev.to/azure/opencv-detect-and-blur-faces-using-haar-cascades-in-c-1ao5

Cascade Classifier
; https://docs.opencv.org/3.4/db/d28/tutorial_cascade_classifier.html

잘 만들어진 모델만 있다면... 이렇게 간단하게 가져다 사용할 수 있으니, 세상 참 좋아졌습니다. ^^
정성태

... 61  62  63  64  65  [66]  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
11997정성태7/30/201913438.NET Framework: 850. C# - Excel(을 비롯해 Office 제품군) COM 객체를 제어 후 Excel.exe 프로세스가 남아 있는 문제 [2]파일 다운로드1
11996정성태7/25/201915861.NET Framework: 849. C# - Socket의 TIME_WAIT 상태를 없애는 방법파일 다운로드1
11995정성태7/23/201918943.NET Framework: 848. C# - smtp.daum.net 서비스(Implicit SSL)를 이용해 메일 보내는 방법 [2]
11994정성태7/22/201914439개발 환경 구성: 454. Azure 가상 머신(VM)에서 SMTP 메일 전송하는 방법파일 다운로드1
11993정성태7/22/20199882오류 유형: 561. Dism.exe 수행 시 "Error: 2 - The system cannot find the file specified." 오류 발생
11992정성태7/22/201911665오류 유형: 560. 서비스 관리자 실행 시 "Windows was unable to open service control manager database on [...]. Error 5: Access is denied." 오류 발생
11991정성태7/18/20199179디버깅 기술: 128. windbg - x64 환경에서 닷넷 예외가 발생한 경우 인자를 확인할 수 없었던 사례
11990정성태7/18/201911381오류 유형: 559. Settings / Update & Security 화면 진입 시 프로그램 종료
11989정성태7/18/201910292Windows: 162. Windows Server 2019 빌드 17763부터 Alt + F4 입력시 곧바로 로그아웃하는 현상
11988정성태7/18/201911739개발 환경 구성: 453. 마이크로소프트가 지정한 모든 Root 인증서를 설치하는 방법
11987정성태7/17/201916712오류 유형: 558. 윈도우 - KMODE_EXCEPTION_NOT_HANDLED 블루스크린(BSOD) 문제 [1]
11986정성태7/17/20199510오류 유형: 557. 드라이브 문자를 할당하지 않은 파티션을 탐색기에서 드라이브 문자와 함께 보여주는 문제
11985정성태7/17/20199633개발 환경 구성: 452. msbuild - csproj에 환경 변수 조건 사용 [1]
11984정성태7/9/201917836개발 환경 구성: 451. Microsoft Edge (Chromium)을 대상으로 한 Selenium WebDriver 사용법 [1]
11983정성태7/8/20198896오류 유형: 556. nodemon - 'mocha' is not recognized as an internal or external command, operable program or batch file.
11982정성태7/8/20198894오류 유형: 555. Visual Studio 빌드 오류 - result: unexpected exception occured (-1002 - 0xfffffc16)
11981정성태7/7/201911075Math: 64. C# - 3층 구조의 신경망(분류)파일 다운로드1
11980정성태7/7/201921515개발 환경 구성: 450. Visual Studio Code의 Java 확장을 이용한 간단한 프로젝트 구축파일 다운로드1
11979정성태7/7/201911047개발 환경 구성: 449. TFS에서 gitlab/github등의 git 서버로 마이그레이션하는 방법
11978정성태7/6/201910401Windows: 161. 계정 정보가 동일하지 않은 PC 간의 인증을 수행하는 방법 [1]
11977정성태7/6/201914959오류 유형: 554. git push - error: RPC failed; HTTP 413 curl 22 The requested URL returned error: 413 Request Entity Too Large
11976정성태7/4/20199324오류 유형: 553. (잘못 인증 한 후) 원격 git repo 재인증 시 "remote: HTTP Basic: Access denied" 오류 발생
11975정성태7/4/201917831개발 환경 구성: 448. Visual Studio Code에서 콘솔 응용 프로그램 개발 시 "입력"받는 방법
11974정성태7/4/201913189Linux: 22. "Visual Studio Code + Remote Development"로 윈도우 환경에서 리눅스(CentOS 7) C/C++ 개발
11973정성태7/4/201912401Linux: 21. 리눅스에서 공유 라이브러리가 로드되지 않는다면?
11972정성태7/3/201915275.NET Framework: 847. JAVA와 .NET 간의 AES 암호화 연동 [1]파일 다운로드1
... 61  62  63  64  65  [66]  67  68  69  70  71  72  73  74  75  ...