ezrgb24.cpp
Upload User: shjinghong
Upload Date: 2008-12-31
Package Size: 4820k
Code Size: 22k
Category:

DirextX

Development Platform:

Visual C++

  1. //------------------------------------------------------------------------------
  2. // File: EZRGB24.cpp
  3. //
  4. // Desc: DirectShow sample code - special effects image filter.
  5. //
  6. // Copyright (c) Microsoft Corporation.  All rights reserved.
  7. //------------------------------------------------------------------------------
  8. //
  9. //
  10. // What this sample illustrates
  11. //
  12. // A special effects filter that does image filtering and colour changes. We
  13. // work on decompressed true colour images and apply a range of effects. The
  14. // effect we do and when it is applied to the video stream can be configured
  15. // through a custom interface we implement and which our property page uses.
  16. // We can also save/load our property settings by supporting IPersistStream
  17. //
  18. //
  19. // Summary
  20. //
  21. // This is a sample special effects filter - it works only with RGB24 formats
  22. // We can be inserted between video decoders and renderers and apply a number
  23. // of different effects to the decompressed video stream. For example we can
  24. // apply red, green or blue filters to the images. We also have posterise and
  25. // greying effects amongst others. The effect can be applied to the stream at
  26. // a given start and end media time, this is defined by using the IIPEffect
  27. // interface and can be viewed by using the property page this filter supports
  28. //
  29. //
  30. // Demonstration instructions
  31. //
  32. // Start GraphEdit, which is available in the SDK DXUtils folder. Drag and drop
  33. // an MPEG, AVI or MOV file into the tool and it will be rendered. Then go to
  34. // the video renderer box and disconnect its input pin from the filter it is
  35. // connected to. Go to the Graph menu and select Insert Filters, from the
  36. // dialog select the Special Effects filter and dismiss the dialog. back in
  37. // the tool workspace connect the output of the decoder to the input of the
  38. // sample filter and then connect the output of the sample to the input of
  39. // the video renderer (GraphEdit may put a colour space convertor in between)
  40. // Then press play and then right click on the sample to select a transform
  41. // by default a transform will be applied from 2 to 7 seconds into the movie
  42. //
  43. //
  44. // Files
  45. //
  46. // ezprop.cpp           A property page to control the video effects
  47. // ezprop.h             Class definition for the property page object
  48. // ezprop.rc            Dialog box template for the property page
  49. // ezrgb24.cpp          Main filter code that does the special effects
  50. // ezrgb24.def          What APIs we import and export from this DLL
  51. // ezrgb24.h            Class definition for the special effects filter
  52. // ezuids.h             Header file containing the filter CLSIDs
  53. // iez.h                Defines the special effects custom interface
  54. // makefile             How we build it...
  55. // resource.h           Microsoft Visual C++ generated resource file
  56. //
  57. //
  58. // Base classes used
  59. //
  60. // CTransformFilter     A transform filter with one input and output pin
  61. // CPersistStream       Handles the grunge of supporting IPersistStream
  62. //
  63. //
  64. #include <windows.h>
  65. #include <streams.h>
  66. #include <initguid.h>
  67. #if (1100 > _MSC_VER)
  68. #include <olectlid.h>
  69. #else
  70. #include <olectl.h>
  71. #endif
  72. #include "EZuids.h"
  73. #include "iEZ.h"
  74. #include "EZprop.h"
  75. #include "EZrgb24.h"
  76. #include "resource.h"
  77. // Setup information
  78. const AMOVIESETUP_MEDIATYPE sudPinTypes =
  79. {
  80.     &MEDIATYPE_Video,       // Major type
  81.     &MEDIASUBTYPE_NULL      // Minor type
  82. };
  83. const AMOVIESETUP_PIN sudpPins[] =
  84. {
  85.     { L"Input",             // Pins string name
  86.       FALSE,                // Is it rendered
  87.       FALSE,                // Is it an output
  88.       FALSE,                // Are we allowed none
  89.       FALSE,                // And allowed many
  90.       &CLSID_NULL,          // Connects to filter
  91.       NULL,                 // Connects to pin
  92.       1,                    // Number of types
  93.       &sudPinTypes          // Pin information
  94.     },
  95.     { L"Output",            // Pins string name
  96.       FALSE,                // Is it rendered
  97.       TRUE,                 // Is it an output
  98.       FALSE,                // Are we allowed none
  99.       FALSE,                // And allowed many
  100.       &CLSID_NULL,          // Connects to filter
  101.       NULL,                 // Connects to pin
  102.       1,                    // Number of types
  103.       &sudPinTypes          // Pin information
  104.     }
  105. };
  106. const AMOVIESETUP_FILTER sudEZrgb24 =
  107. {
  108.     &CLSID_EZrgb24,         // Filter CLSID
  109.     L"Image Effects (EZRGB24)",       // String name
  110.     MERIT_DO_NOT_USE,       // Filter merit
  111.     2,                      // Number of pins
  112.     sudpPins                // Pin information
  113. };
  114. // List of class IDs and creator functions for the class factory. This
  115. // provides the link between the OLE entry point in the DLL and an object
  116. // being created. The class factory will call the static CreateInstance
  117. CFactoryTemplate g_Templates[] = {
  118.     { L"Image Effects"
  119.     , &CLSID_EZrgb24
  120.     , CEZrgb24::CreateInstance
  121.     , NULL
  122.     , &sudEZrgb24 }
  123.   ,
  124.     { L"Special Effects"
  125.     , &CLSID_EZrgb24PropertyPage
  126.     , CEZrgb24Properties::CreateInstance }
  127. };
  128. int g_cTemplates = sizeof(g_Templates) / sizeof(g_Templates[0]);
  129. ////////////////////////////////////////////////////////////////////////
  130. //
  131. // Exported entry points for registration and unregistration 
  132. // (in this case they only call through to default implementations).
  133. //
  134. ////////////////////////////////////////////////////////////////////////
  135. //
  136. // DllRegisterServer
  137. //
  138. // Handles sample registry and unregistry
  139. //
  140. STDAPI DllRegisterServer()
  141. {
  142.     return AMovieDllRegisterServer2( TRUE );
  143. } // DllRegisterServer
  144. //
  145. // DllUnregisterServer
  146. //
  147. STDAPI DllUnregisterServer()
  148. {
  149.     return AMovieDllRegisterServer2( FALSE );
  150. } // DllUnregisterServer
  151. //
  152. // DllEntryPoint
  153. //
  154. extern "C" BOOL WINAPI DllEntryPoint(HINSTANCE, ULONG, LPVOID);
  155. BOOL APIENTRY DllMain(HANDLE hModule, 
  156.                       DWORD  dwReason, 
  157.                       LPVOID lpReserved)
  158. {
  159. return DllEntryPoint((HINSTANCE)(hModule), dwReason, lpReserved);
  160. }
  161. //
  162. // Constructor
  163. //
  164. CEZrgb24::CEZrgb24(TCHAR *tszName,
  165.                    LPUNKNOWN punk,
  166.                    HRESULT *phr) :
  167.     CTransformFilter(tszName, punk, CLSID_EZrgb24),
  168.     m_effect(IDC_RED),
  169.     m_lBufferRequest(1),
  170.     CPersistStream(punk, phr)
  171. {
  172.     char sz[60];
  173.     GetProfileStringA("Quartz", "EffectStart", "2.0", sz, 60);
  174.     m_effectStartTime = COARefTime(atof(sz));
  175.     GetProfileStringA("Quartz", "EffectLength", "5.0", sz, 60);
  176.     m_effectTime = COARefTime(atof(sz));
  177. } // (Constructor)
  178. //
  179. // CreateInstance
  180. //
  181. // Provide the way for COM to create a EZrgb24 object
  182. //
  183. CUnknown *CEZrgb24::CreateInstance(LPUNKNOWN punk, HRESULT *phr)
  184. {
  185.     ASSERT(phr);
  186.     
  187.     CEZrgb24 *pNewObject = new CEZrgb24(NAME("Image Effects"), punk, phr);
  188.     if (pNewObject == NULL) {
  189.         if (phr)
  190.             *phr = E_OUTOFMEMORY;
  191.     }
  192.     return pNewObject;
  193. } // CreateInstance
  194. //
  195. // NonDelegatingQueryInterface
  196. //
  197. // Reveals IIPEffect and ISpecifyPropertyPages
  198. //
  199. STDMETHODIMP CEZrgb24::NonDelegatingQueryInterface(REFIID riid, void **ppv)
  200. {
  201.     CheckPointer(ppv,E_POINTER);
  202.     if (riid == IID_IIPEffect) {
  203.         return GetInterface((IIPEffect *) this, ppv);
  204.     } else if (riid == IID_ISpecifyPropertyPages) {
  205.         return GetInterface((ISpecifyPropertyPages *) this, ppv);
  206.     } else {
  207.         return CTransformFilter::NonDelegatingQueryInterface(riid, ppv);
  208.     }
  209. } // NonDelegatingQueryInterface
  210. //
  211. // Transform
  212. //
  213. // Copy the input sample into the output sample - then transform the output
  214. // sample 'in place'. If we have all keyframes, then we shouldn't do a copy
  215. // If we have cinepak or indeo and are decompressing frame N it needs frame
  216. // decompressed frame N-1 available to calculate it, unless we are at a
  217. // keyframe. So with keyframed codecs, you can't get away with applying the
  218. // transform to change the frames in place, because you'll mess up the next
  219. // frames decompression. The runtime MPEG decoder does not have keyframes in
  220. // the same way so it can be done in place. We know if a sample is key frame
  221. // as we transform because the sync point property will be set on the sample
  222. //
  223. HRESULT CEZrgb24::Transform(IMediaSample *pIn, IMediaSample *pOut)
  224. {
  225.     CheckPointer(pIn,E_POINTER);   
  226.     CheckPointer(pOut,E_POINTER);   
  227.     // Copy the properties across
  228.     HRESULT hr = Copy(pIn, pOut);
  229.     if (FAILED(hr)) {
  230.         return hr;
  231.     }
  232.     // Check to see if it is time to do the sample
  233.     CRefTime tStart, tStop ;
  234.     hr = pIn->GetTime((REFERENCE_TIME *) &tStart, (REFERENCE_TIME *) &tStop);
  235.     if (tStart >= m_effectStartTime) 
  236.     {
  237.         if (tStop <= (m_effectStartTime + m_effectTime)) 
  238.         {
  239.             return Transform(pOut);
  240.         }
  241.     }
  242.     return NOERROR;
  243. } // Transform
  244. //
  245. // Copy
  246. //
  247. // Make destination an identical copy of source
  248. //
  249. HRESULT CEZrgb24::Copy(IMediaSample *pSource, IMediaSample *pDest) const
  250. {
  251.     CheckPointer(pSource,E_POINTER);   
  252.     CheckPointer(pDest,E_POINTER);   
  253.     // Copy the sample data
  254.     BYTE *pSourceBuffer, *pDestBuffer;
  255.     long lSourceSize = pSource->GetActualDataLength();
  256. #ifdef DEBUG
  257.     long lDestSize = pDest->GetSize();
  258.     ASSERT(lDestSize >= lSourceSize);
  259. #endif
  260.     pSource->GetPointer(&pSourceBuffer);
  261.     pDest->GetPointer(&pDestBuffer);
  262.     CopyMemory( (PVOID) pDestBuffer,(PVOID) pSourceBuffer,lSourceSize);
  263.     // Copy the sample times
  264.     REFERENCE_TIME TimeStart, TimeEnd;
  265.     if (NOERROR == pSource->GetTime(&TimeStart, &TimeEnd)) {
  266.         pDest->SetTime(&TimeStart, &TimeEnd);
  267.     }
  268.     LONGLONG MediaStart, MediaEnd;
  269.     if (pSource->GetMediaTime(&MediaStart,&MediaEnd) == NOERROR) {
  270.         pDest->SetMediaTime(&MediaStart,&MediaEnd);
  271.     }
  272.     // Copy the Sync point property
  273.     HRESULT hr = pSource->IsSyncPoint();
  274.     if (hr == S_OK) {
  275.         pDest->SetSyncPoint(TRUE);
  276.     }
  277.     else if (hr == S_FALSE) {
  278.         pDest->SetSyncPoint(FALSE);
  279.     }
  280.     else {  // an unexpected error has occured...
  281.         return E_UNEXPECTED;
  282.     }
  283.     // Copy the media type
  284.     AM_MEDIA_TYPE *pMediaType;
  285.     pSource->GetMediaType(&pMediaType);
  286.     pDest->SetMediaType(pMediaType);
  287.     DeleteMediaType(pMediaType);
  288.     // Copy the preroll property
  289.     hr = pSource->IsPreroll();
  290.     if (hr == S_OK) {
  291.         pDest->SetPreroll(TRUE);
  292.     }
  293.     else if (hr == S_FALSE) {
  294.         pDest->SetPreroll(FALSE);
  295.     }
  296.     else {  // an unexpected error has occured...
  297.         return E_UNEXPECTED;
  298.     }
  299.     // Copy the discontinuity property
  300.     hr = pSource->IsDiscontinuity();
  301.     if (hr == S_OK) {
  302.     pDest->SetDiscontinuity(TRUE);
  303.     }
  304.     else if (hr == S_FALSE) {
  305.         pDest->SetDiscontinuity(FALSE);
  306.     }
  307.     else {  // an unexpected error has occured...
  308.         return E_UNEXPECTED;
  309.     }
  310.     // Copy the actual data length
  311.     long lDataLength = pSource->GetActualDataLength();
  312.     pDest->SetActualDataLength(lDataLength);
  313.     return NOERROR;
  314. } // Copy
  315. //
  316. // Transform (in place)
  317. //
  318. // 'In place' apply the image effect to this sample
  319. //
  320. HRESULT CEZrgb24::Transform(IMediaSample *pMediaSample)
  321. {
  322.     BYTE *pData;                // Pointer to the actual image buffer
  323.     long lDataLen;              // Holds length of any given sample
  324.     unsigned int grey,grey2;    // Used when applying greying effects
  325.     int iPixel;                 // Used to loop through the image pixels
  326.     int temp,x,y;               // General loop counters for transforms
  327.     RGBTRIPLE *prgb;            // Holds a pointer to the current pixel
  328.     AM_MEDIA_TYPE* pType = &m_pInput->CurrentMediaType();
  329.     VIDEOINFOHEADER *pvi = (VIDEOINFOHEADER *) pType->pbFormat;
  330.     ASSERT(pvi);
  331.     CheckPointer(pMediaSample,E_POINTER);
  332.     pMediaSample->GetPointer(&pData);
  333.     lDataLen = pMediaSample->GetSize();
  334.     // Get the image properties from the BITMAPINFOHEADER
  335.     int cxImage    = pvi->bmiHeader.biWidth;
  336.     int cyImage    = pvi->bmiHeader.biHeight;
  337.     int numPixels  = cxImage * cyImage;
  338.     // int iPixelSize = pvi->bmiHeader.biBitCount / 8;
  339.     // int cbImage    = cyImage * cxImage * iPixelSize;
  340.     switch (m_effect)
  341.     {
  342.         case IDC_NONE: break;
  343.         // Zero out the green and blue components to leave only the red
  344.         // so acting as a filter - for better visual results, compute a
  345.         // greyscale value for the pixel and make that the red component
  346.         case IDC_RED:
  347.                         
  348.             prgb = (RGBTRIPLE*) pData;
  349.             for (iPixel=0; iPixel < numPixels; iPixel++, prgb++) {
  350.                 prgb->rgbtGreen = 0;
  351.                 prgb->rgbtBlue = 0;
  352.             }
  353.             break;
  354.     
  355.         case IDC_GREEN:
  356.             prgb = (RGBTRIPLE*) pData;
  357.             for (iPixel=0; iPixel < numPixels; iPixel++, prgb++) {
  358.                 prgb->rgbtRed = 0;
  359.                 prgb->rgbtBlue = 0;
  360.             }
  361.             break;
  362.         case IDC_BLUE:
  363.             prgb = (RGBTRIPLE*) pData;
  364.             for (iPixel=0; iPixel < numPixels; iPixel++, prgb++) {
  365.                 prgb->rgbtRed = 0;
  366.                 prgb->rgbtGreen = 0;
  367.             }
  368.             break;
  369.         // Bitwise shift each component to the right by 1
  370.         // this results in the image getting much darker
  371.         case IDC_DARKEN:
  372.                         
  373.             prgb = (RGBTRIPLE*) pData;
  374.             for (iPixel=0; iPixel < numPixels; iPixel++, prgb++) {
  375.                 prgb->rgbtRed   = (BYTE) (prgb->rgbtRed >> 1);
  376.                 prgb->rgbtGreen = (BYTE) (prgb->rgbtGreen >> 1);
  377.                 prgb->rgbtBlue  = (BYTE) (prgb->rgbtBlue >> 1);
  378.             }
  379.             break;
  380.         // Toggle each bit - this gives a sort of X-ray effect
  381.         case IDC_XOR:   
  382.             prgb = (RGBTRIPLE*) pData;
  383.             for (iPixel=0; iPixel < numPixels; iPixel++, prgb++) {
  384.                 prgb->rgbtRed   = (BYTE) (prgb->rgbtRed ^ 0xff);
  385.                 prgb->rgbtGreen = (BYTE) (prgb->rgbtGreen ^ 0xff);
  386.                 prgb->rgbtBlue  = (BYTE) (prgb->rgbtBlue ^ 0xff);
  387.             }
  388.             break;
  389.         // Zero out the five LSB per each component
  390.         case IDC_POSTERIZE:
  391.             prgb = (RGBTRIPLE*) pData;
  392.             for (iPixel=0; iPixel < numPixels; iPixel++, prgb++) {
  393.                 prgb->rgbtRed   = (BYTE) (prgb->rgbtRed & 0xe0);
  394.                 prgb->rgbtGreen = (BYTE) (prgb->rgbtGreen & 0xe0);
  395.                 prgb->rgbtBlue  = (BYTE) (prgb->rgbtBlue & 0xe0);
  396.             }
  397.             break;
  398.         // Take pixel and its neighbor two pixels to the right and average
  399.         // then out - this blurs them and produces a subtle motion effect
  400.         case IDC_BLUR:
  401.             prgb = (RGBTRIPLE*) pData;
  402.             for (y = 0 ; y < pvi->bmiHeader.biHeight; y++) {
  403.                 for (x = 2 ; x < pvi->bmiHeader.biWidth; x++,prgb++) {
  404.                     prgb->rgbtRed   = (BYTE) ((prgb->rgbtRed + prgb[2].rgbtRed) >> 1);
  405.                     prgb->rgbtGreen = (BYTE) ((prgb->rgbtGreen + prgb[2].rgbtGreen) >> 1);
  406.                     prgb->rgbtBlue  = (BYTE) ((prgb->rgbtBlue + prgb[2].rgbtBlue) >> 1);
  407.                 }
  408.                 prgb +=2;
  409.             }
  410.             break;
  411.         // An excellent greyscale calculation is:
  412.         //      grey = (30 * red + 59 * green + 11 * blue) / 100
  413.         // This is a bit too slow so a faster calculation is:
  414.         //      grey = (red + green) / 2
  415.         case IDC_GREY:  
  416.             prgb = (RGBTRIPLE*) pData;
  417.             for (iPixel=0; iPixel < numPixels ; iPixel++, prgb++) {
  418.                 grey = (prgb->rgbtRed + prgb->rgbtGreen) >> 1;
  419.                 prgb->rgbtRed = prgb->rgbtGreen = prgb->rgbtBlue = (BYTE) grey;
  420.             }
  421.             break;
  422.         // Really sleazy emboss - rather than using a nice 3x3 convulution
  423.         // matrix, we compare the greyscale values of two neighbours. If
  424.         // they are not different, then a mid grey (128, 128, 128) is
  425.         // supplied.  Large differences get father away from the mid grey
  426.         case IDC_EMBOSS:
  427.             prgb = (RGBTRIPLE*) pData;
  428.             for (y = 0 ; y < pvi->bmiHeader.biHeight; y++) 
  429.             {
  430.                 grey2 = (prgb->rgbtRed + prgb->rgbtGreen) >> 1;
  431.                 prgb->rgbtRed = prgb->rgbtGreen = prgb->rgbtBlue = (BYTE) 128;
  432.                 prgb++;
  433.                 for (x = 1 ; x < pvi->bmiHeader.biWidth; x++) {
  434.                     grey = (prgb->rgbtRed + prgb->rgbtGreen) >> 1;
  435.                     temp = grey - grey2;
  436.                     if (temp > 127) temp = 127;
  437.                     if (temp < -127) temp = -127;
  438.                     temp += 128;
  439.                     prgb->rgbtRed = prgb->rgbtGreen = prgb->rgbtBlue = (BYTE) temp;
  440.                     grey2 = grey;
  441.                     prgb++;
  442.                 }
  443.             }   
  444.             break;
  445.     }
  446.     return NOERROR;
  447. } // Transform (in place)
  448. // Check the input type is OK - return an error otherwise
  449. HRESULT CEZrgb24::CheckInputType(const CMediaType *mtIn)
  450. {
  451.     CheckPointer(mtIn,E_POINTER);
  452.     // check this is a VIDEOINFOHEADER type
  453.     if (*mtIn->FormatType() != FORMAT_VideoInfo) {
  454.         return E_INVALIDARG;
  455.     }
  456.     // Can we transform this type
  457.     if (CanPerformEZrgb24(mtIn)) {
  458.         return NOERROR;
  459.     }
  460.     return E_FAIL;
  461. }
  462. //
  463. // Checktransform
  464. //
  465. // Check a transform can be done between these formats
  466. //
  467. HRESULT CEZrgb24::CheckTransform(const CMediaType *mtIn, const CMediaType *mtOut)
  468. {
  469.     CheckPointer(mtIn,E_POINTER);
  470.     CheckPointer(mtOut,E_POINTER);
  471.     if (CanPerformEZrgb24(mtIn)) 
  472.     {
  473.         if (*mtIn == *mtOut) 
  474.         {
  475.             return NOERROR;
  476.         }
  477.     }
  478.     return E_FAIL;
  479. } // CheckTransform
  480. //
  481. // DecideBufferSize
  482. //
  483. // Tell the output pin's allocator what size buffers we
  484. // require. Can only do this when the input is connected
  485. //
  486. HRESULT CEZrgb24::DecideBufferSize(IMemAllocator *pAlloc,ALLOCATOR_PROPERTIES *pProperties)
  487. {
  488.     // Is the input pin connected
  489.     if (m_pInput->IsConnected() == FALSE) {
  490.         return E_UNEXPECTED;
  491.     }
  492.     CheckPointer(pAlloc,E_POINTER);
  493.     CheckPointer(pProperties,E_POINTER);
  494.     HRESULT hr = NOERROR;
  495.     pProperties->cBuffers = 1;
  496.     pProperties->cbBuffer = m_pInput->CurrentMediaType().GetSampleSize();
  497.     ASSERT(pProperties->cbBuffer);
  498.     // Ask the allocator to reserve us some sample memory, NOTE the function
  499.     // can succeed (that is return NOERROR) but still not have allocated the
  500.     // memory that we requested, so we must check we got whatever we wanted
  501.     ALLOCATOR_PROPERTIES Actual;
  502.     hr = pAlloc->SetProperties(pProperties,&Actual);
  503.     if (FAILED(hr)) {
  504.         return hr;
  505.     }
  506.     ASSERT( Actual.cBuffers == 1 );
  507.     if (pProperties->cBuffers > Actual.cBuffers ||
  508.             pProperties->cbBuffer > Actual.cbBuffer) {
  509.                 return E_FAIL;
  510.     }
  511.     return NOERROR;
  512. } // DecideBufferSize
  513. //
  514. // GetMediaType
  515. //
  516. // I support one type, namely the type of the input pin
  517. // This type is only available if my input is connected
  518. //
  519. HRESULT CEZrgb24::GetMediaType(int iPosition, CMediaType *pMediaType)
  520. {
  521.     // Is the input pin connected
  522.     if (m_pInput->IsConnected() == FALSE) {
  523.         return E_UNEXPECTED;
  524.     }
  525.     // This should never happen
  526.     if (iPosition < 0) {
  527.         return E_INVALIDARG;
  528.     }
  529.     // Do we have more items to offer
  530.     if (iPosition > 0) {
  531.         return VFW_S_NO_MORE_ITEMS;
  532.     }
  533.     CheckPointer(pMediaType,E_POINTER);
  534.     *pMediaType = m_pInput->CurrentMediaType();
  535.     return NOERROR;
  536. } // GetMediaType
  537. //
  538. // CanPerformEZrgb24
  539. //
  540. // Check if this is a RGB24 true colour format
  541. //
  542. BOOL CEZrgb24::CanPerformEZrgb24(const CMediaType *pMediaType) const
  543. {
  544.     CheckPointer(pMediaType,FALSE);
  545.     if (IsEqualGUID(*pMediaType->Type(), MEDIATYPE_Video)) 
  546.     {
  547.         if (IsEqualGUID(*pMediaType->Subtype(), MEDIASUBTYPE_RGB24)) 
  548.         {
  549.             VIDEOINFOHEADER *pvi = (VIDEOINFOHEADER *) pMediaType->Format();
  550.             return (pvi->bmiHeader.biBitCount == 24);
  551.         }
  552.     }
  553.     return FALSE;
  554. } // CanPerformEZrgb24
  555. #define WRITEOUT(var)  hr = pStream->Write(&var, sizeof(var), NULL); 
  556.                if (FAILED(hr)) return hr;
  557. #define READIN(var)    hr = pStream->Read(&var, sizeof(var), NULL); 
  558.                if (FAILED(hr)) return hr;
  559. //
  560. // GetClassID
  561. //
  562. // This is the only method of IPersist
  563. //
  564. STDMETHODIMP CEZrgb24::GetClassID(CLSID *pClsid)
  565. {
  566.     return CBaseFilter::GetClassID(pClsid);
  567. } // GetClassID
  568. //
  569. // ScribbleToStream
  570. //
  571. // Overriden to write our state into a stream
  572. //
  573. HRESULT CEZrgb24::ScribbleToStream(IStream *pStream)
  574. {
  575.     HRESULT hr;
  576.     WRITEOUT(m_effect);
  577.     WRITEOUT(m_effectStartTime);
  578.     WRITEOUT(m_effectTime);
  579.     return NOERROR;
  580. } // ScribbleToStream
  581. //
  582. // ReadFromStream
  583. //
  584. // Likewise overriden to restore our state from a stream
  585. //
  586. HRESULT CEZrgb24::ReadFromStream(IStream *pStream)
  587. {
  588.     HRESULT hr;
  589.     READIN(m_effect);
  590.     READIN(m_effectStartTime);
  591.     READIN(m_effectTime);
  592.     return NOERROR;
  593. } // ReadFromStream
  594. //
  595. // GetPages
  596. //
  597. // Returns the clsid's of the property pages we support
  598. //
  599. STDMETHODIMP CEZrgb24::GetPages(CAUUID *pPages)
  600. {
  601.     CheckPointer(pPages,E_POINTER);
  602.     pPages->cElems = 1;
  603.     pPages->pElems = (GUID *) CoTaskMemAlloc(sizeof(GUID));
  604.     if (pPages->pElems == NULL) {
  605.         return E_OUTOFMEMORY;
  606.     }
  607.     *(pPages->pElems) = CLSID_EZrgb24PropertyPage;
  608.     return NOERROR;
  609. } // GetPages
  610. //
  611. // get_IPEffect
  612. //
  613. // Return the current effect selected
  614. //
  615. STDMETHODIMP CEZrgb24::get_IPEffect(int *IPEffect,REFTIME *pStart,REFTIME *pLength)
  616. {
  617.     CAutoLock cAutolock(&m_EZrgb24Lock);
  618.     CheckPointer(IPEffect,E_POINTER);
  619.     CheckPointer(pStart,E_POINTER);
  620.     CheckPointer(pLength,E_POINTER);
  621.     *IPEffect = m_effect;
  622.     *pStart = COARefTime(m_effectStartTime);
  623.     *pLength = COARefTime(m_effectTime);
  624.     return NOERROR;
  625. } // get_IPEffect
  626. //
  627. // put_IPEffect
  628. //
  629. // Set the required video effect
  630. //
  631. STDMETHODIMP CEZrgb24::put_IPEffect(int IPEffect,REFTIME start,REFTIME length)
  632. {
  633.     CAutoLock cAutolock(&m_EZrgb24Lock);
  634.     m_effect = IPEffect;
  635.     m_effectStartTime = COARefTime(start);
  636.     m_effectTime = COARefTime(length);
  637.     SetDirty(TRUE);
  638.     return NOERROR;
  639. } // put_IPEffect