5,276,406 members and growing! (17,511 online)
Email Password   helpLost your password?
Web Development » Web Services » ATL Web Services     Intermediate

Webcam Web Service

By Laurent Kempé

C++ ATL Web Service delivering a JPEG picture grabbed from a Webcam to an ActiveX client
VC6, C++Windows, Win2K, ATL, VS6, VS, Dev

Posted: 23 Apr 2002
Updated: 23 Apr 2002
Views: 233,527
Announcements
Want a new Job?



Search    
Advanced Search
Sitemap
32 votes for this Article.
Popularity: 6.20 Rating: 4.12 out of 5
1 vote, 5.0%
1
2 votes, 10.0%
2
2 votes, 10.0%
3
3 votes, 15.0%
4
12 votes, 60.0%
5

Contents

Introduction

Webcam is a Simple Object Access Protocol (SOAP) project that will permit client's software to get, from a Web Service, JPEG pictures captured by a Webcam. It is my first step experiment in the world of SOAP after several projects using COM/DCOM and ATL.

The project is split in three parts:

  • Server: Active Template Library Server capturing a picture from the Webcam.
  • Web Service: Generated with WSDL Generator from Microsoft®.
  • Client: ActiveX client connecting to Webcam SOAP Server getting pictures and displaying them.

Server

We use Visual C++ and the Active Template Library (ATL) object Wizard to create a Simple COM object, called Camera. It handles the grabbing of the picture and the JPEG compression.

The jpeg compression is achieved with Intel® JPEG Library v1.5. Get it here. You may use other libraries, like IJG JPEG LIBRARY, by using the abstract base class CPicturePacker.

Camera handling is done using Microsoft® Video for Windows® (VFW). Get more information on MSDN / Platform SDK Documentation / Graphics and Multimedia Services / Windows Multimedia / Video Capture.

Trick: VFW needs a handler on a window to attach the camera driver to. As we chose to create a Simple COM object we do not have access to any window handler. Calling GetDesktopWindow() got us one.

    // Don't have access to the container's window so just use the desktop.

    // RMK: If the desktop is not at least 24bits colors, then the grab will

    //      fail

    HWND hWnd = ::GetDesktopWindow();

    if ( hWnd )
    { ...
All functionalities needed to grab a picture on a Webcam are done in the class CWebCam. For example the method GrabFrame:
    bool CWebCam::GrabFrame( CPicturePacker * pPacker,
                             unsigned char  ** ppPackedPicture,
                             unsigned long  * pnPackedPictureBytes )
Use a reference to a CPicturePacker base class to pack the original picture's bits, permitting to extend the packing to whatever format you want, as mentioned before. In this sample I have written the class CIntelJpegPacker inheriting from CPicturePacker and using the Intel® JPEG Library v1.5 to pack the picture grabbed. You may consider using GDI+ for example to do the same.

The rest of the source is dealing with COM and is nothing special.

Server Interface

Our server needs to achieve two simple operations: grabbing a picture from a Webcam and compressing it as a JPEG picture according to a compression ratio defined by the caller. This is achieved in the GrabFrame method.

HRESULT GrabFrame( [in] short nQuality, 
                   [out, retval] SAFEARRAY(unsigned char) * ppGrabbedPicture );
The input parameter 'nQuality' represents the jpeg compression ratio, from 1 to 99. A value of one meaning lowest quality (highest compression) and a value of ninety-nine meaning highest quality (lowest compression).

As an output return value the client get the jpeg picture in a SAFEARRAY of unsigned char. We use this data type because it is fully supported by SOAP (See faced problem 1 and 2).

Our COM object supports the interface IErrorInfo as indicated by the ISupportErrorInfo interface. It permits sending back to the client information about possible issues encountered by the server. The good point is that it is fully automatic for us. Read more about that point in Microsoft SOAP User Guide: "Understanding the SOAP Fault <detail> Contents".

Before writing your interfaces check out the different types supported by SOAP in "Data Type Mappings" and there equivalence in programming languages. You find them in the Microsoft SOAP User Guide.

Web Service

We simply use Microsoft WSDL Generator to wrap our COM object, Camera, into a SOAP Web Service.

On the welcome page click Next.

On the dialog "Select the COM .dll to analyze":

Enter the name of your Web Service, i.e. webcam. 

Browse to select your dll.

Click Next.

On the dialog "Select the services you would like to expose":

Expand the list and check the GrabFrame method.

Click Next.

 

On the dialog "SOAP listener information":

In the URI text box, type the url of your webservice, i.e. http://localhost/webservices/webcam.

Choose ISAPI Listener type.

Then select 2001 as XSD Schema Namespace.

Click Next.

On the dialog "Specify the location for the new WSDL and WSML files":

Select UTF-8 as character set.

Then select the place you want to store the new files, i.e. c:\inetpub\WebServices\webcam.

Click Next.

Click Finish.

 


Now we have our Web Service! You must also define a Virtual Directory called WebServices in IIS.
In the case you want to change the location of the Web Service you need to change the WSDL file generated by Microsoft WSDL Generator.

Client

We use Visual C++ and Active Template Library (ATL) object Wizard to create an ActiveX, called Webcam. This ActiveX is the client part of the Web Service Webcam. It is connecting to the Webcam Web Service, receiving back a jpeg picture and displaying it.

To be able to compile the project you must have installed SOAP Toolkit on your computer. Find it here. For the client you must also have WTL installed. If It is not the case donwload it here.

Client Interface

To be able to grab a picture on the Webcam Web Service, we need to specify the location of the Web Service and a compression ratio. We define this interface:

    HRESULT GrabFrame([in] BSTR strWeb ServiceURL, [in] short nQuality);
  • Input parameter 'strWeb ServiceURL' represents Web Service's URL.
  • Input parameter 'nQuality' represents the jpeg compression ratio, from 1 to 99. A value of one meaning lowest quality (highest compression) and a value of ninety-nine meaning highest quality (lowest compression).

If we get an error, we display it in a tooltip. To create and display the tooltip we use Windows Template Library (WTL).

Test

You may test this Web Service after installing client ActiveX on this page.

Problems Faced

  • When adding the "GrabFrame" method to Webcam Server using the ATL "Add Method to Interface", the method was created correctly in .idl and .h files but the not in the .cpp. We can find a kind of explanation on MSDN: Q198017.
  • Before SOAP SDK SP2, SAFEARRAY(unsigned char) was not correctly handled. You need the SP2 to get a correct wrapping from SAFEARRAY(unisgned char) to base64Binary.

History

Version 1.00 December 4, 2001
First release.

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here

About the Author

Laurent Kempé




Laurent Kempé is the editor, founder, and primary contributor of Tech Head Brothers, a French portal about Microsoft .NET technologies.

Laurent is awarded by Microsoft since Avril 2002: Most Valuable Professional (MVP). You might contact him through his Blog.



Occupation: Web Developer
Location: France France

Other popular Web Services articles:

Article Top
Sign Up to vote for this article
You must Sign In to use this message board.
FAQ FAQ Noise ToleranceSearch Search Messages 
 Layout  Per page   
 Msgs 1 to 25 of 53 (Total in Forum: 53) (Refresh)FirstPrevNext
Subject  Author Date 
GeneralSOAP SDK v2.0 required?memberDan Thurman17:04 13 Jul '06  
GeneralRe: SOAP SDK v2.0 required?memberDan Thurman11:02 14 Jul '06  
Generalhow to caopture images in C++ using webcammemberrichytany4:10 12 Jul '06  
GeneralHow to get array of int, or byte without the safearraymemberguocang12:59 11 Jul '06  
Questionusing camera example for a printer filememberdavid barnwell11:13 16 May '06  
Questionvirtual WebCammemberolariuadrian2:00 17 Feb '06  
GeneralMicrosoft WSDL GeneratorsussAnonymous11:54 14 Jul '05  
GeneralYahoo ProtocolmemberManpreet Chadha20:01 25 Oct '04  
GeneralRe: Yahoo Protocolmemberpengdingfu18:01 24 Dec '04  
GeneralHelp -- Compile Error for ClientmemberZhijie Wang12:34 27 Aug '04  
GeneralWeb ServicesussAnonymous0:11 12 Aug '04  
Generalis .NET framework required ?memberhspc20:18 30 Jul '04  
GeneralWebcammemberhungrau18:33 28 Apr '04  
GeneralCamera pilotable via interface web et ActiveX de la carte de capture video ?memberALIBAS21:25 26 Apr '04  
GeneralCamera pilotable via interface web et pour l'ActiveX de la carte de capture video ?memberALIBAS5:46 26 Apr '04  
Generalweb page sample doesn't exist ?memberorenboskila14:37 19 Feb '04  
GeneralRe: web page sample doesn't exist ?memberLaurent Kempé22:20 19 Feb '04  
Generalhow to save a flash file created with a object as a gif/jpg/bmp file?memberzhongxunyang17:49 24 Sep '03  
GeneralRe: how to save a flash file created with a object as a gif/jpg/bmp file?editorNishant S18:26 24 Sep '03  
GeneralClient side questionmemberDee San10:29 26 Aug '03  
GeneralRe: Client side questionmemberLaurent Kempé19:52 26 Aug '03  
GeneralCool Example!memberNikolai Teofilov6:04 2 Jul '03  
GeneralRe: Cool Example!memberLaurent Kempé4:15 3 Aug '03  
GeneralNot a .NET code example actuallymemberJason Orphanidis4:36 10 Jun '03  
GeneralRe: Not a .NET code example actuallymemberLaurent Kempé22:17 14 Jun '03  

General General    News News    Question Question    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

PermaLink | Privacy | Terms of Use
Last Updated: 23 Apr 2002
Editor: Chris Maunder
Copyright 2002 by Laurent Kempé
Everything else Copyright © CodeProject, 1999-2008
Web10 | Advertise on the Code Project