ICCompressorXChoose使用

来源:互联网 发布:苹果mac镜像下载 编辑:程序博客网 时间:2024/04/28 16:11
数据压缩步骤:
1:定义COMPVARS对象
view plaincopy to clipboardprint?
  1. typedef struct {    
  2.     LONG         cbSize;    
  3.     DWORD        dwFlags;    
  4.     HIC          hic;    
  5.     DWORD        fccType;    
  6.     DWORD        fccHandler;    
  7.     LPBITMAPINFO lpbiIn;    
  8.     LPBITMAPINFO lpbiOut;    
  9.     LPVOID       lpBitsOut;    
  10.     LPVOID       lpBitsPrev;    
  11.     LONG         lFrame;    
  12.     LONG         lKey;    
  13.     LONG         lDataRate;    
  14.     LONG         lQ;    
  15.     LONG         lKeyCount;    
  16.     LPVOID       lpState;    
  17.     LONG         cbState;    
  18. } COMPVARS;    

cbSize:
必须设置该值为一个正确的值.
或者cbSize = sizeof(COMPVARS);
dwFlags:
ICMF_COMPVARS_VALID
如果你使用ICCompressorChoose 函数来初始化结构,请不要设置这个值.
hic:压缩的句柄,你可以使用ICOpen去获得一个句柄
fccType: ICTYPE_VIDEO
当然也可以设置为zero
fccHandler:四个字符的压缩引擎
lpbiIn:保留
lpbiOut:BITMAPINFO结构的指针,包含了输出图象格式,也可以通过使用函数ICCompressorChoose设置输出格式
lpBitsOut:保留
lpBitsPrev:保留
lFrame:保留
lKey:关键帧速率 ICSeqCompressFrameStart 函数使用这个值来创建关键帧
lDataRate:数据速率,可以通过ICCompressorChoose设置
lQ:质量设置.可以使用ICQUALITY_DEFAULT 默认. ICSeqCompressFrameStart 函数使用这个值来产生数据质量
lKeyCount:保留
lpState:保留
cbState:保留
引用

重要:
如果你要手动设置这个结构,你必须提供如下成员的值:
cbSize, hic, lpbiOut, lKey, and lQ.还有dwFlags为ICMF_COMPVARS_VALID.



对Video Compression Manager(VCM)要熟悉,你还需要了解如下几个结构:
1:BITMAPINFO
2:BITMAPINFOHEADER




VCM工作在应用程序以压缩解压缩驱动之间. 当一个程序调用VCM时候,VCM翻译成一个消息,消息通过ICSendMessage函数去选择调用相应的压缩解压缩器.
VCM服务,一般来说,一个应用程序使用VCM去处理如下的工作:
1:定位,打开,安装一个压缩解压缩器.
2:配置或者获取压缩解压缩器的配置信息.
3:使用一系列的函数去压缩,解压缩,显示数据.


一般显示数据是使用函数DrawDIB.
压缩解压缩基本信息:
你可以使用ICLocate和ICOpen函数定位打开一个压缩器.你可以使用ICLocate去找到一个特定类型的压缩器并获得她的句柄为下一步的VCM函数做准备.你还可以使用ICOpen去打开一个压缩器,你的程序使用ICOpen返回来的句柄来完成更多VCM的功能.

用户如何选择一个压缩器:
当压缩数据时候,你的应用程序可以使用ICCompressorChoose 函数去打开一个对话框去选择一个压缩器,然后返回一个句柄给COMPVARS 结构的hic成员.后面压缩的时候就可以使用这个句柄.

应用程序可以定位和打开一个已经安装的压缩解压缩器,通过使用函数ICLocate和ICOpen函数,当一个应用程序完成使用压缩解压缩器以后,要使用ICClose来关闭.

单图象压缩:
可以使用ICImageCompress 函数来完成单幅图片的压缩.


下面来讨论一下关键的问题:
流压缩的问题
你的程序可以使用 ICSeqCompressFrame, ICSeqCompressFrameStart, and ICSeqCompressFrameEnd函数去压缩一系列的帧.这些函数使用存储在COMPVARS结构中的数据,应用程序可以使用ICCompressorChoose去让用户选择一个压缩器.
在应用程序开始压缩一系列帧之前,必须使用ICSeqCompressFrameStart函数去分配必须的资源.资源分配以后,应用程序可以使用ICSeqCompressFrame去压缩.帧速率和关键帧参数以及其他的一系列参数都存储在COMPVARS结构中,最后程序要使用ICCompressorFree 释放资源.

下面的内容是图象压缩,暂时不翻译,感兴趣的可以去看MSDN.


如何将数据显示出来:
可以使用ICDraw.ICDrawStart.ICDrawBegin.

下面的方法使用ICLocate找到一个压缩器能够压缩8bits的图象的:
view plaincopy to clipboardprint?
  1. BITMAPINFOHEADER bih;    
  2. HIC              hIC    
  3.     
  4. // Initialize the bitmap structure.    
  5. bih.biSize = sizeof(BITMAPINFOHEADER);    
  6. bih.biWidth = bih.biHeight = 0;    
  7. bih.biPlanes = 1;    
  8. bih.biCompression = BI_RGB;      // standard RGB bitmap    
  9. bih.biBitcount = 8;              // 8 bits-per-pixel format    
  10. bih.biSizeImage = 0;    
  11. bih.biXPelsPerMeter = bih.biYPelsPerMeter = 0;    
  12. bih.biClrUsed = bih.biClrImportant = 256;    
  13.     
  14. hIC = ICLocate (ICTYPE_VIDEO, 0L, (LPBITMAPINFOHEADER) &bih,    
  15.     NULL, ICMODE_COMPRESS);    
  16.   
  17. 下面的例子定位一个压缩器去压缩一个8bitRGB 为 8bit RLE格式   
  18. BITMAPINFOHEADER    bihIn, bihOut;    
  19. HIC                 hIC    
  20.     
  21. // Initialize the bitmap structure.    
  22.   
  23. biSize = bihOut.biSize = sizeof(BITMAPINFOHEADER);    
  24. bihIn.biWidth = bihIn.biHeight = bihOut.biWidth = bihOut.biHeight = 0;    
  25. bihIn.biPlanes = bihOut.biPlanes= 1;    
  26. bihIn.biCompression = BI_RGB;        // standard RGB bitmap for input    
  27. bihOut.biCompression = BI_RLE8;      // 8-bit RLE for output format    
  28. bihIn.biBitcount = bihOut.biBitCount = 8;  // 8 bits-per-pixel format    
  29. bihIn.biSizeImage = bihOut.biSizeImage = 0;    
  30. bihIn.biXPelsPerMeter = bih.biYPelsPerMeter =    
  31.     bihOut.biXPelsPerMeter = bihOut.biYPelsPerMeter = 0;    
  32. bihIn.biClrUsed = bih.biClrImportant =    
  33.     bihOut.biClrUsed = bihOut.biClrImportant = 256;    
  34.   
  35. hIC = ICLocate (ICTYPE_VIDEO, 0L,    
  36.     (LPBITMAPINFOHEADER)&bihIn,    
  37.     (LPBITMAPINFOHEADER)&bihOut, ICMODE_COMPRESS);    
  38.   
  39. 下面的函数完成一个安装压缩器:   
  40. // This function looks like a DriverProc entry point.    
  41.   
  42. LRESULT MyCodecFunction(DWORD dwID, HDRVR hDriver,    
  43.     UINT uiMessage, LPARAM lParam1, LPARAM lParam2);    
  44.     
  45. // This function installs the MyCodecFunction as a compressor.    
  46.   
  47. result = ICInstall ( ICTYPE_VIDEO, mmioFOURCC('s','a','m','p'),    
  48.     (LPARAM)(FARPROC)&MyCodecFunction, NULL, ICINSTALL_FUNCTION);    
  49.   

设置一个压缩器的输出格式:
view plaincopy to clipboardprint?
  1. LPBITMAPINFOHEADER   lpbiIn, lpbiOut;    
  2.     
  3. // *lpbiIn must be initialized to the input format.    
  4.     
  5. dwFormatSize = ICCompressGetFormatSize(hIC, lpbiIn);    
  6. h = GlobalAlloc(GHND, dwFormatSize);    
  7. lpbiOut = (LPBITMAPINFOHEADER)GlobalLock(h);    
  8. ICCompressGetFormat(hIC, lpbiIn, lpbiOut);    
  9.   

这里很重要
引用

方法是先使用ICCompressGetFormat 去获得压缩格式所需要的空间大小,然后使用GlobalAlloc 分配内存,最后使用ICCompressGetFormat去设置压缩信息.

下面是压缩数据的最最关键的代码:
view plaincopy to clipboardprint?
  1. DWORD dwCkID;    
  2. DWORD dwCompFlags;    
  3. DWORD dwQuality;    
  4. LONG  lNumFrames, lFrameNum;    
  5. // Assume dwNumFrames is initialized to the total number of frames.    
  6. // Assume dwQuality holds the proper quality value (0-10000).    
  7. // Assume lpbiOut, lpOut, lpbiIn, and lpIn are initialized properly.    
  8.     
  9. // If OK to start, compress each frame.    
  10. if (ICCompressBegin(hIC, lpbiIn, lpbiOut) == ICERR_OK)   
  11. {    
  12.     for ( lFrameNum = 0; lFrameNum < lNumFrames; lFrameNum++)   
  13.     {    
  14.         if (ICCompress(hIC, 0, lpbiOut, lpOut, lpbiIn, lpIn,    
  15.             &dwCkID, &dwCompFlags, lFrameNum,    
  16.             0, dwQuality, NULL, NULL) == ICERR_OK)   
  17.         {    
  18.             // Write compressed data to the AVI file.    
  19.                 
  20.             // Set lpIn to the next frame in the sequence.    
  21.                 
  22.         }    
  23.         else    
  24.         {    
  25.             // Handle compressor error.    
  26.         }    
  27.     }    
  28.     ICCompressEnd(hIC);    // terminate compression    
  29. }    
  30. else    
  31. {    
  32.     // Handle the error identifying the unsupported format.    
  33. }    
  34.   

上面的压缩只是一定方式的,压缩视频帧的时候应该使用ICSeqCompressFrame
view plaincopy to clipboardprint?
  1. LPVOID ICSeqCompressFrame(   
  2.   PCOMPVARS pc,     
  3.   UINT uiFlags,     
  4.   LPVOID lpBits,    
  5.   BOOL * pfKey,     
  6.   LONG * plSize     
  7. );   

参数:
pc:一个COMPVARS结构的指针,必须先初始化一些压缩参数.
uiFlags:必须为0

lpBits:指向待压缩数据的指针,数据不包含header以及format格式
pfKey:返回是否该帧压缩为关键帧.

plSize:返回用户压缩数据的大小
该函数使用一个COMPVARS结构提供压缩器以及关键帧,速率,速率通过函数 ICSeqCompressorFrameStart 设置.

当压缩一系列视频流的时候,要使用这个函数去代替ICCompress函数.
你也可以通过使用函数ICCompressorChoose去让用户选择结构.你也可以自己手动设置这个参数.

使用ICSeqCompressFrameStart, ICSeqCompressFrame, and ICSeqCompressFrameEnd函数去压缩一系列帧,每出现一帧数据使用ICSeqCompressFrame 一次.
当完成了压缩,使用ICCompressorFree去释放COMPVARS的资源.



2:设置COMPVARS对象的参数:
  1:cbSize
  2:dwFlags
  3:cbState
  4:fccHandler
  5:fccType
  6:hic
    1:使用ICOpen
    2:使用ICLocate
3:使用ICCompressGetFormatSize获得数据大小
4:分配内存
5:ICCompressGetFormat 设置参数
6:ICCompressGetSize设置参数
7:ICSeqCompressFrameStart开始

8:断开连接的时候需要如下:
引用
  1:ICSeqCompressFrameEnd
  2:ICCompressorFree
  3:ICClose


9:然后在回调函数里面使用:
ICSeqCompressFrame来压缩.
这里要注意有一个参数是是否为关键帧.bKeyFrame!
一定要将这个参数传递给服务器端.
ICSeqCompressFrame函数返回的BUF也要发送到服务器端.这里就算是开发完毕了!
ICSeqCompressFrame的最后一个参数是压缩后数据的大小.


10:服务器端解压数据处理:
view plaincopy to clipboardprint?
  1.   //下面的代码是Delphi的处理方法:   
  2.   
  3.      //接着,一起来看看客户端的图像显示过程:      
  4.      //CapVar是COMPVARS对象 C++的定义为 COMPVARS CapVar;   
  5.   //先用取得的CapVar来连接视频编码器      
  6.   //Capvar.fccHandler是使用客户端一样的解码器   
  7.      CapVar.hic := ICOpen(CapVar.fccType,CapVar.fccHandler,ICMODE_DECOMPRESS);      
  8.          
  9.      //成功后,用服务器传来的BmpOutInfo当作客户端的BmpInInfo来取得解压输出的图像头BmpOutInfo      
  10.     //获得参数   
  11.      OutFormatSize:=ICDecompressGetFormatSize(CapVar.hic,@BmpInInfo.bmiHeader);      
  12.      //分配内存   
  13.   GetMem(BmpOutInfo,OutFormatSize);   
  14.   //初始化     
  15.      zeromemory(BmpOutInfo,OutFormatSize);      
  16.     //设置参数   
  17.      ICDecompressGetFormat(CapVar.hic, @BmpInInfo.bmiHeader, @BmpOutInfo^.bmiHeader);      
  18.     //设置参数   
  19.      OutBufferSize:=BmpOutInfo^.bmiHeader.biSizeImage;      
  20.      //分配内存   
  21.   getmem(OutBuffer,OutBufferSize);      
  22.     //初始化   
  23.      zeromemory(OutBuffer,OutBufferSize);      
  24.   //解压缩开始   
  25.      ICDecompressBegin(CapVar.hic,@BmpInInfo.bmiHeader, @BmpOutInfo^.bmiHeader);      
  26.      
  27.      
  28.      //最后,当然是视频数据的解压过程      
  29.     //如果是关键帧   
  30.      if VIDEO_DATA.bKeyFrame then     
  31.           Result:=ICDecompress(CapVar.hic,0,@BmpInInfo,@VIDEO_DATA.Buf,@BmpOutInfo.bmiHeader,OutBuffer)      
  32.      //如果是普通帧   
  33.   else     
  34.           Result:=ICDecompress(CapVar.hic,ICDECOMPRESS_NOTKEYFRAME,@BmpInInfo,@VIDEO_DATA.Buf,@BmpOutInfo.bmiHeader,OutBuffer);      
  35.      //如果解压成功   
  36.   if (Result=ICERR_OK) then     
  37.      begin     
  38.     //将图象画出来   
  39.           SetDIBitsToDevice(Canvas.Handle,0,0,bmptmp.Width,bmptmp.Height,0,0,0,BmpOutInfo^.bmiHeader.biHeight ,OutBuffer,BmpOutInfo^,DIB_RGB_COLORS);      
  40.    end;     

 
原创粉丝点击