CAR.CPP
上传用户:bangxh
上传日期:2007-01-31
资源大小:42235k
文件大小:15k
源码类别:

Windows编程

开发平台:

Visual C++

  1. /*+==========================================================================
  2.   File:      CAR.CPP
  3.   Summary:   Implementation file for the COCar COM Object Class (for
  4.              aggregatable Car COM Objects).
  5.              Of particular interest for an COM application, CAR
  6.              showcases a COM Object Class offering a main IUnknown
  7.              interface and the ICar interface of Car-related features.
  8.              This multiple interface COM Object Class is achieved via
  9.              the technique of nested classes: the implementation of the
  10.              ICar interface is nested inside of the COCar COM
  11.              Object Class.
  12.              For a comprehensive tutorial code tour of this module's
  13.              contents and offerings see the tutorial COMOBJ.HTM
  14.              file.  For more specific technical details on the internal
  15.              workings see the comments dispersed throughout the
  16.              module's source code.
  17.   Classes:   COCar
  18.   Functions: none.
  19.   Origin:    8-19-95: atrent - Created.
  20. ----------------------------------------------------------------------------
  21.   This file is part of the Microsoft COM Tutorial Code Samples.
  22.   Copyright (C) Microsoft Corporation, 1997.  All rights reserved.
  23.   This source code is intended only as a supplement to Microsoft
  24.   Development Tools and/or on-line documentation.  See these other
  25.   materials for detailed information regarding Microsoft code samples.
  26.   THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY
  27.   KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
  28.   IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A
  29.   PARTICULAR PURPOSE.
  30. ==========================================================================+*/
  31. /*---------------------------------------------------------------------------
  32.   We include WINDOWS.H for all Win32 applications.
  33.   We include OLE2.H because we will make calls to the COM/OLE Libraries.
  34.   We include APPUTIL.H because we will be building this application using
  35.     the convenient Virtual Window and Dialog classes and other
  36.     utility functions in the APPUTIL Library (ie, APPUTIL.LIB).
  37.   We include ICARS.H and CARGUIDS.H for the common car-related Interface
  38.     class, GUID, and CLSID specifications.
  39.   We include COMOBJI.H because it has internal class declarations and
  40.     resource ID definitions specific for this DLL.
  41.   We include COMOBJ.H because it has the interface declarations.
  42.   We include CAR.H because it has the class COCar declarations.
  43. ---------------------------------------------------------------------------*/
  44. #include <windows.h>
  45. #include <ole2.h>
  46. #include <apputil.h>
  47. #include <icars.h>
  48. #include <carguids.h>
  49. #include "comobji.h"
  50. #include "comobj.h"
  51. #include "car.h"
  52. /*---------------------------------------------------------------------------
  53.   COCar's implementation of its main COM object class including
  54.   Constructor, Destructor, QueryInterface, AddRef, and Release.
  55. ---------------------------------------------------------------------------*/
  56. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  57.   Method:   COCar::COCar
  58.   Summary:  COCar Constructor. Note the member initializer:
  59.             "m_ImpICar(this, pUnkOuter)" which is used to pass the 'this'
  60.             and pUnkOuter pointers of this constructor function to the
  61.             constructor in the instantiation of the implementation of
  62.             the CImpICar interface (which is nested inside this present
  63.             COCar Object Class).
  64.   Args:     IUnknown* pUnkOuter)
  65.               Pointer to the the outer Unknown.  NULL means this COM Object
  66.               is not being Aggregated.  Non NULL means it is being created
  67.               on behalf of an outside COM object that is reusing it via
  68.               aggregation.
  69.   Modifies: m_cRefs, m_pUnkOuter.
  70.   Returns:  void
  71. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  72. COCar::COCar(
  73.   IUnknown* pUnkOuter) :
  74.   m_ImpICar(this, pUnkOuter)
  75. {
  76.   // Zero the COM object's reference count.
  77.   m_cRefs = 0;
  78.   // No AddRef necessary if non-NULL, as we're nested.
  79.   m_pUnkOuter = pUnkOuter;
  80.   LOGF1("D: COCar Constructor. m_pUnkOuter=0x%X.", m_pUnkOuter);
  81.   return;
  82. }
  83. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  84.   Method:   COCar::~COCar
  85.   Summary:  COCar Destructor.
  86.   Args:     void
  87.   Modifies: .
  88.   Returns:  void
  89. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  90. COCar::~COCar(void)
  91. {
  92.   LOG("D: COCar::Destructor.");
  93.   return;
  94. }
  95. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  96.   Method:   COCar::QueryInterface
  97.   Summary:  QueryInterface of the COCar non-delegating
  98.             IUnknown implementation.
  99.   Args:     REFIID riid,
  100.               [in] GUID of the Interface being requested.
  101.             PPVOID ppv)
  102.               [out] Address of the caller's pointer variable that will
  103.               receive the requested interface pointer.
  104.   Modifies: .
  105.   Returns:  HRESULT
  106. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  107. STDMETHODIMP COCar::QueryInterface(
  108.                REFIID riid,
  109.                PPVOID ppv)
  110. {
  111.   HRESULT hr = E_NOINTERFACE;
  112.   *ppv = NULL;
  113.   if (IID_IUnknown == riid)
  114.   {
  115.     *ppv = this;
  116.     LOG("D: COCar::QueryInterface. 'this' pIUnknown returned.");
  117.   }
  118.   else if (IID_ICar == riid)
  119.   {
  120.     *ppv = &m_ImpICar;
  121.     LOG("D: COCar::QueryInterface. pICar returned.");
  122.   }
  123.   if (NULL != *ppv)
  124.   {
  125.     // We've handed out a pointer to the interface so obey the COM rules
  126.     //   and AddRef the reference count.
  127.     ((LPUNKNOWN)*ppv)->AddRef();
  128.     hr = NOERROR;
  129.   }
  130.   return (hr);
  131. }
  132. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  133.   Method:   COCar::AddRef
  134.   Summary:  AddRef of the COCar non-delegating IUnknown implementation.
  135.   Args:     void
  136.   Modifies: m_cRefs.
  137.   Returns:  ULONG
  138.               New value of m_cRefs (COM object's reference count).
  139. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  140. STDMETHODIMP_(ULONG) COCar::AddRef(void)
  141. {
  142.   m_cRefs++;
  143.   LOGF1("D: COCar::AddRef. New cRefs=%i.", m_cRefs);
  144.   return m_cRefs;
  145. }
  146. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  147.   Method:   COCar::Release
  148.   Summary:  Release of the COCar non-delegating IUnknown implementation.
  149.   Args:     void
  150.   Modifies: m_cRefs.
  151.   Returns:  ULONG
  152.               New value of m_cRefs (COM object's reference count).
  153. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  154. STDMETHODIMP_(ULONG) COCar::Release(void)
  155. {
  156.   m_cRefs--;
  157.   LOGF1("D: COCar::Release. New cRefs=%i.", m_cRefs);
  158.   if (0 == m_cRefs)
  159.   {
  160.     // We artificially bump the main ref count to prevent reentrancy
  161.     // via the main object destructor.  Not really needed in this
  162.     // COCar but a good practice because we are aggregatable and
  163.     // may at some point in the future add something entertaining like
  164.     // some Releases to the COCar destructor.
  165.     m_cRefs++;
  166.     delete this;
  167.   }
  168.   return m_cRefs;
  169. }
  170. /*---------------------------------------------------------------------------
  171.   COCar's nested implementation of the ICar interface including
  172.   Constructor, Destructor, QueryInterface, AddRef, Release,
  173.   Shift, Clutch, Speed, and Steer.
  174. ---------------------------------------------------------------------------*/
  175. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  176.   Method:   COCar::CImpICar::CImpICar
  177.   Summary:  Constructor for the CImpICar interface instantiation.
  178.   Args:     COCar* pBackObj,
  179.               Back pointer to the parent outer object.
  180.             IUnknown* pUnkOuter)
  181.               Pointer to the outer Unknown.  For delegation.
  182.   Modifies: m_cRefI, m_pBackObj, m_pUnkOuter.
  183.   Returns:  void
  184. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  185. COCar::CImpICar::CImpICar(
  186.   COCar* pBackObj,
  187.   IUnknown* pUnkOuter)
  188. {
  189.   // Init the Interface Ref Count (used for debugging only).
  190.   m_cRefI = 0;
  191.   // Init the Back Object Pointer to point to the parent object.
  192.   m_pBackObj = pBackObj;
  193.   // Init the CImpICar interface's delegating Unknown pointer.  We use
  194.   // the Back Object pointer for IUnknown delegation here if we are not
  195.   // being aggregated.  If we are being aggregated we use the supplied
  196.   // pUnkOuter for IUnknown delegation.  In either case the pointer
  197.   // assignment requires no AddRef because the CImpICar lifetime is
  198.   // quaranteed by the lifetime of the parent object in which
  199.   // CImpICar is nested.
  200.   if (NULL == pUnkOuter)
  201.   {
  202.     m_pUnkOuter = pBackObj;
  203.     LOG("D: COCar::CImpICar Constructor. Non-Aggregating.");
  204.   }
  205.   else
  206.   {
  207.     m_pUnkOuter = pUnkOuter;
  208.     LOG("D: COCar::CImpICar Constructor. Aggregating.");
  209.   }
  210.   return;
  211. }
  212. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  213.   Method:   COCar::CImpICar::~CImpICar
  214.   Summary:  Destructor for the CImpICar interface instantiation.
  215.   Args:     void
  216.   Modifies: .
  217.   Returns:  void
  218. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  219. COCar::CImpICar::~CImpICar(void)
  220. {
  221.   LOG("D: COCar::CImpICar Destructor.");
  222.   return;
  223. }
  224. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  225.   Method:   COCar::CImpICar::QueryInterface
  226.   Summary:  The QueryInterface IUnknown member of this ICar interface
  227.             implementation that delegates to m_pUnkOuter, whatever it is.
  228.   Args:     REFIID riid,
  229.               [in] GUID of the Interface being requested.
  230.             PPVOID ppv)
  231.               [out] Address of the caller's pointer variable that will
  232.               receive the requested interface pointer.
  233.   Modifies: .
  234.   Returns:  HRESULT
  235.               Returned by the delegated outer QueryInterface call.
  236. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  237. STDMETHODIMP COCar::CImpICar::QueryInterface(
  238.                REFIID riid,
  239.                PPVOID ppv)
  240. {
  241.   LOG("D: COCar::CImpICar::QueryInterface. Delegating.");
  242.   // Delegate this call to the outer object's QueryInterface.
  243.   return m_pUnkOuter->QueryInterface(riid, ppv);
  244. }
  245. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  246.   Method:   COCar::CImpICar::AddRef
  247.   Summary:  The AddRef IUnknown member of this ICar interface
  248.             implementation that delegates to m_pUnkOuter, whatever it is.
  249.   Args:     void
  250.   Modifies: m_cRefI.
  251.   Returns:  ULONG
  252.               Returned by the delegated outer AddRef call.
  253. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  254. STDMETHODIMP_(ULONG) COCar::CImpICar::AddRef(void)
  255. {
  256.   // Increment the Interface Reference Count.
  257.   ++m_cRefI;
  258.   LOGF1("D: COCar::CImpICar::Addref. Delegating. New cI=%i.", m_cRefI);
  259.   // Delegate this call to the outer object's AddRef.
  260.   return m_pUnkOuter->AddRef();
  261. }
  262. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  263.   Method:   COCar::CImpICar::Release
  264.   Summary:  The Release IUnknown member of this ICar interface
  265.             implementation that delegates to m_pUnkOuter, whatever it is.
  266.   Args:     void
  267.   Modifies: .
  268.   Returns:  ULONG
  269.               Returned by the delegated outer Release call.
  270. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  271. STDMETHODIMP_(ULONG) COCar::CImpICar::Release(void)
  272. {
  273.   // Decrement the Interface Reference Count.
  274.   --m_cRefI;
  275.   LOGF1("D: COCar::CImpICar::Release. Delegating. New cI=%i.", m_cRefI);
  276.   // Delegate this call to the outer object's Release.
  277.   return m_pUnkOuter->Release();
  278. }
  279. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  280.   Method:   COCar::CImpICar::Shift
  281.   Summary:  The Shift member method of this ICar interface implementation.
  282.             A simple empty method on a COCar COM object for tutorial
  283.             purposes.  Presumably if this Car object were modeling
  284.             a real Car then the Shift method would shift to the specified
  285.             gear.
  286.   Args:     short nGear
  287.               0 - Neutral; 1 - 5 First 5 forward gears; 6 - Reverse.
  288.   Modifies: .
  289.   Returns:  HRESULT
  290.               NOERROR
  291. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  292. STDMETHODIMP COCar::CImpICar::Shift(
  293.                short nGear)
  294. {
  295.   LOGF1("D: COCar::CImpICar::Shift. Called. nGear=%i.",nGear);
  296.   return NOERROR;
  297. }
  298. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  299.   Method:   COCar::CImpICar::Clutch
  300.   Summary:  The Clutch member method of this ICar interface implementation.
  301.             A simple empty method on a COCar COM object for tutorial
  302.             purposes.  Presumably if this Car object were modeling
  303.             a real Car then the Clutch method would engage the clutch the
  304.             specified amount.
  305.   Args:     short nEngaged)
  306.               Percent clutch is engaged (0 to 100%).
  307.   Modifies: .
  308.   Returns:  HRESULT
  309.               NOERROR
  310. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  311. STDMETHODIMP COCar::CImpICar::Clutch(
  312.                short nEngaged)
  313. {
  314.   LOGF1("D: COCar::CImpICar::Clutch. Called. nEngaged=%i.",nEngaged);
  315.   return NOERROR;
  316. }
  317. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  318.   Method:   COCar::CImpICar::Speed
  319.   Summary:  The Speed member method of this ICar interface implementation.
  320.             A simple empty method on a COCar COM object for tutorial
  321.             purposes.  Presumably if this Car object were modeling
  322.             a real Car then this method would accelerate/brake to bring
  323.             the car to the specified speed.
  324.   Args:     short nMph
  325.               New speed in miles per hour.
  326.   Modifies: .
  327.   Returns:  HRESULT
  328.               NOERROR
  329. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  330. STDMETHODIMP COCar::CImpICar::Speed(
  331.                short nMph)
  332. {
  333.   LOGF1("D: COCar::CImpICar::Speed. Called. nMph=%i.",nMph);
  334.   return NOERROR;
  335. }
  336. /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
  337.   Method:   COCar::CImpICar::Steer
  338.   Summary:  The Steer member method of this ICar interface implementation.
  339.             A simple empty method on a COCar COM object for tutorial
  340.             purposes.  Presumably if this Car object were modeling
  341.             a real Car then the Steer method would set the steering
  342.             angle of the Car.
  343.   Args:     short nAngle)
  344.               0 degrees straight ahead, -45 Full left, +45 Full right.
  345.   Modifies: .
  346.   Returns:  HRESULT
  347.               NOERROR
  348. M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
  349. STDMETHODIMP COCar::CImpICar::Steer(
  350.                short nAngle)
  351. {
  352.   LOGF1("D: COCar::CImpICar::Steer. Called. nAngle=%i.",nAngle);
  353.   return NOERROR;
  354. }