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