'영상처리'에 해당되는 글 2건

내소스는 완벽하다 . 캠이 구린거다 ㅋㅋㅋ
그렇게 믿고 싶다 -_-

#include <windows.h>
#include "resource.h"
#include <Vfw.h>  

//#define PIXELHISTO 1
//#define LINEHISTO 1
#define ALLHISTO 1

LRESULT CALLBACK FramInfo(HWND , LPVIDEOHDR);
LRESULT CALLBACK WndProc(HWND,UINT,WPARAM,LPARAM);
BOOL CALLBACK DialogBoxProc(HWND,UINT,WPARAM,LPARAM);
void DrawBitmap(HDC, int, int, HBITMAP);
HINSTANCE g_hlnst;
LPCTSTR lpszClass=TEXT("WinCam");
HWND vfw;
HWND vfw2;
HWND Hwndmain;

int APIENTRY WinMain(HINSTANCE hlnstance,HINSTANCE hPrevlnstance,LPSTR lpszCmdParam,int nCmdShow)
{
 HWND hWnd;
 MSG Message;
 WNDCLASS WndClass;
 g_hlnst=hlnstance;

 WndClass.cbClsExtra=0;
 WndClass.cbWndExtra=0;
 WndClass.hbrBackground=(HBRUSH)GetStockObject(WHITE_BRUSH);

 WndClass.hCursor=LoadCursor(NULL,IDC_ARROW);
 WndClass.hIcon=LoadIcon(NULL,IDI_APPLICATION);
 WndClass.hInstance=hlnstance;
 WndClass.lpfnWndProc=WndProc;
 WndClass.lpszClassName=lpszClass;
 WndClass.lpszMenuName=NULL;
 WndClass.style=CS_HREDRAW|CS_VREDRAW;
 RegisterClass(&WndClass);

 hWnd=CreateWindow(lpszClass,
  lpszClass,
  WS_OVERLAPPEDWINDOW,
  CW_USEDEFAULT,
  CW_USEDEFAULT,
  CW_USEDEFAULT,
  CW_USEDEFAULT,
  NULL,
  (HMENU)NULL,
  hlnstance,NULL);
 ShowWindow(hWnd,nCmdShow);

 while(GetMessage(&Message,NULL,0,0))
 {
  TranslateMessage(&Message);
  DispatchMessage(&Message);
 }

 return (int)Message.wParam;
}

BITMAPINFO bm;
PAINTSTRUCT ps;
HBITMAP hbit;
DWORD dwsize;
HBITMAP OldBitmap;
HANDLE hFile;
u_char buf1[280000];
DWORD dwWrite;
DWORD dwRead;
int Temp;
LPBYTE lpData1;
u_int Histo[3][256];


LRESULT CALLBACK WndProc(HWND hWnd,UINT iMessage,WPARAM wParam,LPARAM lParam)
{

 HDC hdc;
 HDC hMemDC;
 
 switch(iMessage)
 {
  case WM_CREATE:
   hdc = GetDC(hWnd);
   Hwndmain = hWnd;
   
   vfw = capCreateCaptureWindow( TEXT("CAM")
    ,WS_CHILD | WS_VISIBLE
    ,0
    ,0
    ,500
    ,400
    ,hWnd
    ,0);
   capDriverConnect(vfw, 0);
   capPreviewRate(vfw, 1);  
   capPreview(vfw, TRUE);
   
   capGetVideoFormat(vfw, &bm , sizeof(bm));

   hbit = CreateCompatibleBitmap(hdc,  bm.bmiHeader.biWidth,   bm.bmiHeader.biHeight);
   capSetCallbackOnFrame(vfw, FramInfo);

   
   InvalidateRect(Hwndmain,NULL,TRUE);
   ReleaseDC(hWnd, hdc);
   return 0;

  case WM_LBUTTONDOWN:
   DialogBox(g_hlnst, MAKEINTRESOURCE(IDD_DIALOG1), hWnd, DialogBoxProc);
   return 0;

  case WM_PAINT:
   hdc = BeginPaint(Hwndmain, &ps);
   hMemDC= CreateCompatibleDC(hdc);
   DeleteDC(hMemDC);
   EndPaint(Hwndmain, &ps);
   return 0;

  case WM_DESTROY:
   PostQuitMessage(0);
   return 0;
 }
 return(DefWindowProc(hWnd,iMessage,wParam,lParam));
}


LRESULT CALLBACK FramInfo(HWND vfw, LPVIDEOHDR VideoHdr)
{
 HDC hMemDC;
 HDC hdc;
 HBITMAP OldBitmap;
 int iCntX;
 int iCntY;
 int Jump=0;
 int R, G, B;
 WCHAR t[100];
 HBRUSH Myblush;
 HBRUSH Oldblush;
 HPEN MyPen;
 HPEN OldPen;
 
 
 hdc = GetDC(Hwndmain);
 hMemDC = CreateCompatibleDC(hdc);
 OldBitmap = (HBITMAP)SelectObject(hMemDC,hbit);
 
 Myblush = CreateSolidBrush(RGB(255,255,255));
 Oldblush = (HBRUSH)SelectObject(hMemDC, Myblush);
 
   
 memset(Histo, 0, sizeof(Histo));
 for(iCntY= 0; iCntY < bm.bmiHeader.biHeight ; ++iCntY)
 {    
  for(iCntX=0; iCntX  < bm.bmiHeader.biWidth  ; ++iCntX )
  {
       
    ++Histo[0][(VideoHdr->lpData[Jump + 2])];
    ++Histo[1][(VideoHdr->lpData[Jump + 1])];
    ++Histo[2][(VideoHdr->lpData[Jump + 0])];
    Jump += 3;

     
  }
 }
 wsprintf(t, TEXT("%d"), Histo[0][0]+Histo[1][0]+Histo[2][0]);
 SetWindowText(Hwndmain, t);
 


 MyPen = CreatePen(PS_SOLID, 1, RGB(0,0,0));
 OldPen = (HPEN)SelectObject(hMemDC, MyPen);

 // RGB 히스토그램
 Rectangle(hMemDC, 0, 0, 256 , bm.bmiHeader.biHeight);
 for(iCntY= 0; iCntY < 255 ; ++iCntY)
 {


#ifdef PIXELHISTO
  // 점버전
  SetPixel(hMemDC, iCntY+25, bm.bmiHeader.biHeight - ((((Histo[0][iCntY])+(Histo[1][iCntY])+(Histo[2][iCntY]))/15)/3), RGB(0,0,0));  
#endif
 
 
 
#ifdef LINEHISTO
  // 선버전
  MoveToEx(hMemDC, iCntY+25, bm.bmiHeader.biHeight - ((((Histo[0][iCntY])+(Histo[1][iCntY])+(Histo[2][iCntY]))/15)/3), NULL);
  LineTo(hMemDC, iCntY+25, bm.bmiHeader.biHeight - ((((Histo[0][iCntY+1])+(Histo[1][iCntY+1])+(Histo[2][iCntY+1]))/15)/3));
#endif  


#ifdef ALLHISTO
  // 면버전
  MoveToEx(hMemDC,iCntY+1, bm.bmiHeader.biHeight - ((((Histo[0][iCntY])+(Histo[1][iCntY])+(Histo[2][iCntY]))/15)/3), NULL);
  LineTo(hMemDC, iCntY+1, bm.bmiHeader.biHeight);
#endif
 
 }
 BitBlt(hdc, bm.bmiHeader.biWidth+30, 0, 256 , bm.bmiHeader.biHeight, hMemDC, 0, 0, SRCCOPY);


 

 //RED값 히스토그램
 MyPen = CreatePen(PS_SOLID, 1, RGB(0,0,0));
 (HPEN)SelectObject(hMemDC, MyPen);
 Rectangle(hMemDC, 0, 0, 256 , bm.bmiHeader.biHeight);
 MyPen = CreatePen(PS_SOLID, 1, RGB(255,0,0));
 (HPEN)SelectObject(hMemDC, MyPen);
 for(iCntY= 0; iCntY < 255 ; ++iCntY)
 {
 
#ifdef PIXELHISTO
  // 점버전
  SetPixel(hMemDC, iCntY+25, bm.bmiHeader.biHeight - ((Histo[0][iCntY])/15), RGB(0,0,0));
#endif
 

#ifdef LINEHISTO
  // 선버전
  MoveToEx(hMemDC, iCntY+25, bm.bmiHeader.biHeight - ((Histo[0][iCntY])/15), NULL);
  LineTo(hMemDC, iCntY+25, bm.bmiHeader.biHeight - ((Histo[0][iCntY+1])/15));
#endif


#ifdef ALLHISTO

  // 면버전
  MoveToEx(hMemDC,iCntY, bm.bmiHeader.biHeight - ((Histo[0][iCntY])/15), NULL);
  LineTo(hMemDC, iCntY, bm.bmiHeader.biHeight-1);
#endif
 }
 BitBlt(hdc, 2, bm.bmiHeader.biHeight+30, 256 , bm.bmiHeader.biHeight, hMemDC, 0, 0, SRCCOPY);



 //GREEN값 히스토그램
 MyPen = CreatePen(PS_SOLID, 1, RGB(0,0,0));
 (HPEN)SelectObject(hMemDC, MyPen);
// Rectangle(hMemDC, 0, 0, 256 , bm.bmiHeader.biHeight);
 MyPen = CreatePen(PS_SOLID, 1, RGB(0,255,0));
 (HPEN)SelectObject(hMemDC, MyPen);
 for(iCntY= 0; iCntY < 255 ; ++iCntY)
 {

#ifdef PIXELHISTO
  // 점버전
  //SetPixel(hMemDC, iCntY+25, bm.bmiHeader.biHeight - ((Histo[1][iCntY])/15), RGB(0,0,0));
#endif


#ifdef LINEHISTO
  // 선버전
  MoveToEx(hMemDC, iCntY+25, bm.bmiHeader.biHeight - ((Histo[1][iCntY])/15), NULL);
  LineTo(hMemDC, iCntY+25, bm.bmiHeader.biHeight - ((Histo[1][iCntY+1])/15));
#endif


#ifdef ALLHISTO

  // 면버전
  MoveToEx(hMemDC,iCntY, bm.bmiHeader.biHeight - ((Histo[1][iCntY])/15), NULL);
  LineTo(hMemDC, iCntY, bm.bmiHeader.biHeight-1);
#endif


 }
 BitBlt(hdc, 256+10, bm.bmiHeader.biHeight+30, 256 , bm.bmiHeader.biHeight, hMemDC, 0, 0, SRCCOPY);



 //BLUE값 히스토그램
 MyPen = CreatePen(PS_SOLID, 1, RGB(0,0,0));
 (HPEN)SelectObject(hMemDC, MyPen);
// Rectangle(hMemDC, 0, 0, 256 , bm.bmiHeader.biHeight);
 MyPen = CreatePen(PS_SOLID, 1, RGB(0,0,255));
 (HPEN)SelectObject(hMemDC, MyPen);
 for(iCntY= 0; iCntY < 255 ; ++iCntY)
 {

#ifdef PIXELHISTO
  // 점버전
  //SetPixel(hMemDC, iCntY+25, bm.bmiHeader.biHeight - ((Histo[2][iCntY])/15), RGB(0,0,0));
#endif


#ifdef LINEHISTO
  // 선버전
  MoveToEx(hMemDC, iCntY+25, bm.bmiHeader.biHeight - ((Histo[2][iCntY])/15), NULL);
  LineTo(hMemDC, iCntY+25, bm.bmiHeader.biHeight - ((Histo[2][iCntY+1])/15));
#endif


#ifdef ALLHISTO
  // 면버전
  MoveToEx(hMemDC,iCntY, bm.bmiHeader.biHeight - ((Histo[2][iCntY])/15), NULL);
  LineTo(hMemDC, iCntY, bm.bmiHeader.biHeight-1);
#endif

 }
 BitBlt(hdc, (256+10)<<1, bm.bmiHeader.biHeight+30, 256 , bm.bmiHeader.biHeight, hMemDC, 0, 0, SRCCOPY);


 SelectObject(hMemDC, Oldblush);
 DeleteObject(Myblush);
 SelectObject(hMemDC, OldPen);
 DeleteObject(MyPen);
 SelectObject(hMemDC,OldBitmap);
 DeleteDC(hMemDC);
 ReleaseDC(Hwndmain,hdc);
 return 0;

}

BOOL CALLBACK DialogBoxProc(HWND hDlg,UINT iMessage,WPARAM wParam,LPARAM lParam)
{
 TCHAR str[20] = {};
 switch(iMessage)
 {
  case WM_COMMAND:
   switch (wParam)
   {
    case IDOK:
     GetDlgItemText(hDlg, IDC_EDIT1, str, sizeof(str));
     capFileSaveDIB(vfw,str);

    case IDCANCEL:
     EndDialog(hDlg,0);
     return TRUE;
   }
   break;
 }
 return FALSE;
}

사용자 삽입 이미지


Posted by 응이

VFW (Video For Window)

영상처리 2008. 10. 23. 10:26


VFW (Video For Window)

VFW 라이브러리를 사용한 비디오 캡쳐 방법을 살펴보자. 비디오 캡쳐를 위해서는 먼저 캡쳐
윈도우를 생성하고 이 윈도우와 캡쳐 드라이버를 연결해야 한다. 그리고 캡쳐된 비디오 프레임을 캡쳐 윈도우에 보여주기 위해서는 한 프레임이 캡쳐 될 때마다 특정 함수를 호출하도록
한 다음에 호출된 함수에서 그 프레임을 화면에 출력한다. 이러한 작업을 수행하기 위해서는
VFW 라이브러리 중에서 다음과 같은 함수들을 사용해야 한다.

capGetDriverDescription()
CapCreateCaptureWindow()
capDriverConnect()
capPreviewRate()
capSetVideoFormat()
capDriverDisconnect()
capSetCallbackOnFrame()


●capGetDriverDescription()함수

BOOLcapGetDriverDescription(index, name, name_size, version, version_size);

이 함수는 캡쳐 드라이버의 이름 및 버전 정보를 검색한다. 첫 번째 매개변수인
index는 검색하고자 하는 드라이버의 번호를 나타내는데, 0부터 9까지의 값을 가질 수 있다.
즉, 한 컴퓨터에서 9대의 캡쳐 장치가 사용될 수 있다고 가정하고 있다. 검색하고자 하는 번호의
드라이버가 존재하면 이 함수는 name에 드라이버의 이름을 저장하고 version에
드라이버 버전을 저장한 다음에 함수 결과값으로 TURE 값을 반환한다.


●capCreateCaptureWindow()함수

H중 capCreateCaptureWindow(name, style, x, y, width, height, hWnd, id);

이 함수는 캐쳐 윈도우를 생성한다. name 에는 윈도우위 이름을 지정한다.
style 윈도우위 스타일을 지정한다. (x,y)에는 캡쳐 윈도우의 좌측 상단의 좌표를 지정한다.
width height에는 캡쳐 윈도우의 크기를 지정한다. hWnd에는 부모윈도우의 핸들값을 입력한다.
id에는 윈도우의 식별 번호를 입력한다. 캡쳐 윈도우가 정상적으로 생성되면 캡쳐 윈도우의
핸들의 함수 결과값으로 반환되고 그렇지 않으면 NULL값이 반환된다.


●capDriverConnect()함수

capDriverConnect(hWnd, index);

이 함수는 캡쳐 윈도우를 캡쳐 드라이버에 연결한다. hWnd는 캡쳐 윈도우의 핸들을 나타내고
index는 캡쳐 드라이버의 번호를 나타낸다. 이 함수는 캡쳐 장치가 정상적으로 작동하여
연결이 성공되면  TURE 값을 반환하고 그렇지 않으면 FALSE 값을 반환한다.


●capPreviewRate()함수

capPreviewRate(hWnd, rate);

이 함수는 미리보기 (preview)모드에서의 프레임 재생 속도를 설정한다.
여기에서 미리보기란 카메라에서 입력된 비디오를 파일에 저장하는 것이 아니라 화면에
보여준다는 것을 의미한다. hWnd는 캡쳐 윈도우의 핸들 값으로 설정하고 rate는 밀리초(ms)
단 위의 시간으로 설정한다.
예를 들어, rate 값을 66으로 설정하면 0.066초마다 새로운 비디오 프레임을 캡쳐해서
디스플레이 하게 된다. 이와 같은 속도로 재생을 하면 1초에 15개의 비디오 프레임이
디스플레이된다.


●capSetVideFormat()함수

capSetVideoFormat(hWnd, videoFormat, videoFormat_size);

이 함수는 캡쳐된 비디오 데이터 형식을 설정한다. 사용자가 원하는 비디오 데이터형식이
캡쳐 장치에서 지원이 되면 이 함수는 TRUE 값을 반환하고 그렇지 않으면 FALSE 값을
반환하므로 반드시 이 함수의 결과값이 TRUE 인지 검사한 다음에 다음단계로 넘어가야 한다.

hWnd는 캡쳐 윈도우의 핸들 값으로 설정한다.
videoFormat은 설정하고자 하는 비디오 데이터 형식을 나타내는데,
비디오 데이터의 각 프레임에 대한 비트맵 형식을 BITMAPINFO 구조로 기술한다.
BITMAPINFO 구조는 다음과 같다.

typedef struct tagBITMAPINFO{

 BITMAPINFOHEADER bmiHeader;
 RGBQUAD bmiColrs[1];

 } BITMAPINFO;

BITMAPINFO는 BITMAPINFOHEADER와 RGBQUAD로 구성되는데, 여기에서는 다음과 같이
정의되는 BITMAPINFOHEADER 구조체의 값만 사용하면 된다.



pBmiInfo->bmiHeader.biSize          = 40;            //BITMAPINFOHEADER 구조체의 크기
pBmiInfo->bmiHeader.biHeight        = 480;         //영상의 가로크기
pBmiInfo->bmiHeader.biWidth         = 640;         //영상의 세로크기
    (양수:좌측 하단이 원점, 음수:좌측 상단이 원점)

pBmiInfo->bmiHeader.biPlanes       = 1;              //목표 장치의 플레인 수(1로 설정해야함)
pBmiInfo->bmiHeader.biBitCount     = (short) 24;  //각 픽셀의 비트수

pBmiInfo->bmiHeader.biCompression      = 0;    //압축 방법(bi_rgb 또는 0:무압축 비트맵)
pBmiInfo->bmiHeader.biSizeImage          = 0;    //비트맵 영상 크기(무압축인 경우 0으로 설정)
pBmiInfo->bmiHeader.biClrImportant        = 0;    // 비트맵 디스플레이에 사용되는 컬러수

pBmiInfo->bmiHeader.biClrUsed             = 0;         // 사용된 컬러의 수
pBmiInfo->bmiHeader.biXPelsPerMeter   = 10000;   //수평 해상도 (미터당 픽셀수)
pBmiInfo->bmiHeader.biYPelsPerMeter   = 10000;   //수직 해상도 (미터당 픽셀 수)

}BITMAPINFOHEADER;

비디오 캡쳐를 위해서는 다음과 같이 BITMAPINFOHEADER 구조체에서
biSize,biWidth, biHeight, biPlanes, biBitCount 값을 설정하고 나머지 값을은 0으로 설정하면된다.



BITMAPINFO bmi;

memset(&bmi.bmiHeader, 0, sizeof(bmiHeader));

pBmiInfo->bmiHeader.biSize = sizeof(bmi.bmiHeader);
pBmiInfo->bmiHeader.biWidth = 640;   //영상의 세로크기(양수:좌측 하단이 원점
pBmiInfo->bmiHeader.biHeight = 480;   //영상의 가로크기
pBmiInfo->bmiHeader.biPlanes = 1;   //목표 장치의 플레인 수(1로 설정해야함)
pBmiInfo->bmiHeader.biBitCount = 24;   //각 픽셀의 비트수


●capDriverDisconnect()함수;

capDriverDisconnect(hWnd);

이 함수는 carDriverConnect() 함수에 의하여 연결한 캡쳐 윈도우와 캡쳐 장치를 분리하는
함수이다. hWnd에는 분리하고자 하는 캡쳐 윈도우의 핸들 값을 설정한다.
 

●capSetCallbackOnFrame()함수

BOOL capSetCallbackOnFrame(hWnd, func);

VFW 라이브러리에서는 캡쳐된 비디오 프레임을 화면에 보여주는 작업을 callback함수를 사용해서
처리하도록 하고 있다. capSetCallbackOnFrame()함수는 캡쳐 장치로부터 비디오 프레임이 캡쳐되었을때에 를 화면에 보여주기 위해서 호출되는 callback 함수를 설정한다.
hWnd는 캡쳐 윈도우의 핸들 값으로 설정하고 func는 호출될 함수 이름으로 설정한다.


Posted by 응이
1

Dream come true.
응이

달력

태그목록