LICCLIEN.CPP
资源名称:MSDN_VC98.zip [点击查看]
上传用户:bangxh
上传日期:2007-01-31
资源大小:42235k
文件大小:46k
源码类别:
Windows编程
开发平台:
Visual C++
- /*+==========================================================================
- File: LICCLIEN.CPP
- Summary: Based largely on the DLLCLIEN.EXE application code, this
- module is meant to use COM to load and access some COM
- components in two separate DLL COM Servers (DLLSERVE built in
- the sibling DLLSERVE directory and LICSERVE built in the
- sibline LICSERVE directory). Thus to run LICCLIEN you must
- build both DLLSERVE and LICSERVE first. The main idea in
- this code sample is to showcase licensing of components in
- servers. To this end the LICSERVE server provides the
- LicCruiseCar component as a licensed component and this
- LICCLIEN module must access it through the IClassFactory2
- licensing facilities.
- LICCLIEN composes its own native composite COM object,
- COUtilityCruiseCar, by aggregating an existing licensed
- COLicCruiseCar COM component (provided by the LICSERVE.DLL
- server) with a native implementation of the IUtility
- interface. Thus, this app showcases nested aggregation using
- COM's client/server protocol and involving a licensed
- component.
- LICCLIEN also instantiates the car-related components
- provided by the two servers and exercises them (ie, Car,
- UtilityCar, LicCruiseCar, and UtilityCruiseCar). LICCLIEN
- provides a set of menus for these Car related objects with
- selections for the respective methods of the Interfaces
- exposed by those COM Objects. LICCLIEN also exploits
- LICSERVE's LicCarSample component to setup operation of that
- server as a code sample that supports the Client's trace
- LOGging macros. After this is set up the server logs to the
- Client's logging facility. Similar logging is set up for the
- DLLSERVE server.
- For a comprehensive tutorial code tour of LICCLIEN's
- contents and offerings see the tutorial LICCLIEN.HTM file.
- For more specific technical details on the internal workings
- see the comments dispersed throughout the LICCLIEN source code.
- For more details on the LICSERVE.DLL that LICCLIEN works with
- see the LICSERVE.HTM file in the main tutorial directory.
- For more details on the DLLSERVE.DLL that LICCLIEN works with
- see the DLLSERVE.HTM file in the main tutorial directory.
- Classes: CMainWindow
- Functions: InitApplication, WinMain
- Origin: 10-5-95: atrent - Editor-inheritance from the DLLCLIEN source.
- ----------------------------------------------------------------------------
- This file is part of the Microsoft COM Tutorial Code Samples.
- Copyright (C) Microsoft Corporation, 1997. All rights reserved.
- This source code is intended only as a supplement to Microsoft
- Development Tools and/or on-line documentation. See these other
- materials for detailed information regarding Microsoft code samples.
- THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY
- KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
- IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A
- PARTICULAR PURPOSE.
- ==========================================================================+*/
- /*--------------------------------------------------------------------------
- We include WINDOWS.H for all Win32 applications.
- We include OLE2.H because we will be calling the COM/OLE Libraries.
- We include INITGUID.H only once (here) in the entire app because we
- will be defining GUIDs and want them as constants in the data segment.
- We include OLECTL.H because we will using IClassFactory2.
- We include COMMDLG.H because we will be using the Open File and
- potentially other Common dialogs.
- We include APPUTIL.H because we will be building this application using
- the convenient Virtual Window and Dialog classes and other
- utility functions in the APPUTIL Library (ie, APPUTIL.LIB).
- We include ICARS.H and CARGUIDS.H for the common car-related Interface
- class, GUID, and CLSID specifications.
- We include LICCLIEN.H because it has class and resource definitions
- specific to this LICCLIEN application.
- We include UTCRUCAR.H because it has COUtilityCar object class.
- ---------------------------------------------------------------------------*/
- #include <windows.h>
- #include <ole2.h>
- #include <initguid.h>
- #include <olectl.h>
- #include <commdlg.h>
- #include <apputil.h>
- #include <icars.h>
- #include <carguids.h>
- #include "licclien.h"
- #include "utcrucar.h"
- // Here is a pointer for use by the global Trace Message logging macros.
- CMsgLog* g_pMsgLog = NULL;
- /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
- Method: CMainWindow::CMainWindow
- Summary: CMainWindow Constructor.
- Args: .
- Modifies: .
- Returns: .
- M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
- CMainWindow::CMainWindow()
- {
- // Ensure these member variable strings are null strings.
- m_szFileName[0] = 0;
- m_szFileTitle[0] = 0;
- // Fill in the Open File Name Common Dialog's OPENFILENAME structure.
- m_ofnFile.lStructSize = sizeof(OPENFILENAME);
- m_ofnFile.hwndOwner = m_hWnd;
- m_ofnFile.hInstance = m_hInst;
- m_ofnFile.lpstrFilter = TEXT(OFN_DEFAULTFILES_STR);
- m_ofnFile.lpstrCustomFilter = NULL;
- m_ofnFile.nMaxCustFilter = 0;
- m_ofnFile.nFilterIndex = 1;
- m_ofnFile.lpstrFile = m_szFileName;
- m_ofnFile.nMaxFile = MAX_PATH;
- m_ofnFile.lpstrInitialDir = TEXT(".");
- m_ofnFile.lpstrFileTitle = m_szFileTitle;
- m_ofnFile.nMaxFileTitle = MAX_PATH;
- m_ofnFile.lpstrTitle = TEXT(OFN_DEFAULTTITLE_STR);
- m_ofnFile.lpstrDefExt = NULL;
- m_ofnFile.Flags = OFN_HIDEREADONLY;
- // NULL the license key string pointer.
- m_bstrLicKey = NULL;
- // NULL all the main interface pointers.
- m_pCar = NULL;
- m_pUtilityCar = NULL;
- m_pLicCruiseCar = NULL;
- m_pUtilityCruiseCar = NULL;
- m_pLicCarSample = NULL;
- m_pDllCarSample = NULL;
- // Null the Message object pointers.
- m_pMsgBox = NULL;
- m_pMsgLog = NULL;
- }
- /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
- Method: CMainWindow::~CMainWindow
- Summary: CMainWindow Destructor. Destruction of the main window
- indicates that the application should quit and thus the
- PostQuitMessage API is called.
- Args: .
- Modifies: .
- Returns: .
- M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
- CMainWindow::~CMainWindow()
- {
- // CMainWindow is derived from CVirWindow which traps the WM_DESTROY
- // message and causes a delete of CMainWindow which in turn causes this
- // destructor to run. The WM_DESTROY results when the window is destoyed
- // after a close of the window. Prior to exiting the main message loop:
- // We free any dynamically allocated data.
- SysFreeString(m_bstrLicKey);
- // We release any and all of the pointers to instantiated COM objects.
- RELEASE_INTERFACE(m_pCar);
- RELEASE_INTERFACE(m_pUtilityCar);
- RELEASE_INTERFACE(m_pLicCruiseCar);
- RELEASE_INTERFACE(m_pUtilityCruiseCar);
- RELEASE_INTERFACE(m_pLicCarSample);
- RELEASE_INTERFACE(m_pDllCarSample);
- // We delete the CMsgBox and CMsgLog objects that were made in
- // Initinstance.
- DELETE_POINTER(m_pMsgBox);
- DELETE_POINTER(m_pMsgLog);
- // We then post a WM_QUIT message to cause an exit of the main thread's
- // message loop and an exit of this instance of the application.
- PostQuitMessage(0);
- }
- /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
- Method: CMainWindow::InitInstance
- Summary: Instantiates an instance of the main application window.
- This method must be called only once, immediately after
- window class construction. We take care to delete 'this'
- CMainWindow if we must return the error condition FALSE.
- Args: HINSTANCE hInstance,
- Handle of the application instance.
- int nCmdShow)
- Command to pass to ShowWindow.
- Modifies: m_szHelpFile, m_pMsgBox, m_pMsgLog.
- Returns: BOOL.
- TRUE if succeeded.
- FALSE if failed.
- M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
- BOOL CMainWindow::InitInstance(
- HINSTANCE hInstance,
- int nCmdShow)
- {
- BOOL bOk = FALSE;
- HWND hWnd;
- // Create the Message Box and Message Log objects.
- m_pMsgBox = new CMsgBox;
- m_pMsgLog = new CMsgLog;
- if (NULL != m_pMsgBox && NULL != m_pMsgLog)
- {
- // Note, the Create method sets the m_hWnd member so we don't
- // need to set it explicitly here first.
- // Here is the create of this window. Size the window reasonably.
- // Create sets both m_hInst and m_hWnd.
- hWnd = Create(
- TEXT(MAIN_WINDOW_CLASS_NAME_STR),
- TEXT(MAIN_WINDOW_TITLE_STR),
- WS_OVERLAPPED | WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX
- | WS_MAXIMIZEBOX | WS_THICKFRAME,
- CW_USEDEFAULT,
- CW_USEDEFAULT,
- ::GetSystemMetrics(SM_CXSCREEN)*3/5,
- ::GetSystemMetrics(SM_CYSCREEN)*3/5,
- NULL,
- NULL,
- hInstance);
- if (hWnd)
- {
- // Ensure the new window is shown on screen and its content is
- // painted.
- ::ShowWindow(m_hWnd, nCmdShow);
- ::UpdateWindow(m_hWnd);
- // Build a path to where the help file should be (it should be in
- // the same directory as the .EXE but with the .HTM extension.
- MakeFamilyPath(hInstance, m_szHelpFile, TEXT(HELP_FILE_EXT));
- // Init the Message Box object.
- if (m_pMsgBox->Init(m_hInst, m_hWnd))
- {
- // Create the Trace Message Log ListBox as a child window that
- // fits the client area of the Main Window (the TRUE 3rd argument
- // specifies such an inside child).
- // If you want the Trace Message Log in a separate (but owned)
- // window, then pass a FALSE instead for the 3rd argument.
- if (m_pMsgLog->Create(m_hInst, m_hWnd, TRUE))
- {
- HRESULT hr;
- // Assign the global MsgLog pointer.
- g_pMsgLog = m_pMsgLog;
- // Use macro to log an initial start messsage.
- LOGID(IDS_START_MESSAGE_LOG);
- LOG("C: Start LICSERVE Server Trace Logging.");
- // Now ask COM for the ISample interface to the server's
- // LicCarSample component. This effectively loads the
- // LICSERVE DLL.
- hr = CoCreateInstance(
- CLSID_LicCarSample,
- NULL,
- CLSCTX_INPROC_SERVER,
- IID_ISample,
- (PPVOID)&m_pLicCarSample);
- if (SUCCEEDED(hr))
- {
- hr = m_pLicCarSample->Init(m_hWnd, g_pMsgLog);
- if (SUCCEEDED(hr))
- {
- HMENU hMenu = ::GetMenu(m_hWnd);
- ::CheckMenuItem(
- hMenu,
- IDM_LOG_LICSERVER,
- MF_BYCOMMAND | MF_CHECKED);
- LOG("C: Start DLLSERVE Server Trace Logging.");
- // Also set up logging from the subordinate DLLSERVE server
- // to the this client as well.
- hr = CoCreateInstance(
- CLSID_DllCarSample,
- NULL,
- CLSCTX_INPROC_SERVER,
- IID_ISample,
- (PPVOID)&m_pDllCarSample);
- if (SUCCEEDED(hr))
- {
- hr = m_pDllCarSample->Init(m_hWnd, g_pMsgLog);
- bOk = SUCCEEDED(hr);
- if (bOk)
- {
- ::CheckMenuItem(
- hMenu,
- IDM_LOG_DLLSERVER,
- MF_BYCOMMAND | MF_CHECKED);
- }
- else
- {
- RELEASE_INTERFACE(m_pDllCarSample);
- // We ask COM to unload any unused COM Servers.
- CoFreeUnusedLibraries();
- }
- }
- else
- m_pMsgBox->ErrorID(IDS_NODLLSERVER);
- }
- else
- {
- RELEASE_INTERFACE(m_pLicCarSample);
- // We ask COM to unload any unused COM Servers.
- CoFreeUnusedLibraries();
- }
- }
- else
- m_pMsgBox->ErrorID(IDS_NOLICSERVER);
- }
- }
- }
- }
- if (!bOk)
- {
- DELETE_POINTER(m_pMsgBox);
- DELETE_POINTER(m_pMsgLog);
- }
- return (bOk);
- }
- /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
- Method: CMainWindow::GetInterface
- Summary: Convenience method that wraps the QueryInterface call
- and accepts a main COM object IUnknown pointer as an argument.
- Args: IUnknown* pObj,
- Pointer to the COM Object we are getting an interface on.
- REFIID riid,
- The GUID for the interface that we are seeking.
- PPVOID ppv);
- Address of the caller's pointer variable that will
- receive the requested interface pointer.
- Modifies: .
- Returns: BOOL.
- TRUE if succeeded.
- FALSE if failed.
- M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
- BOOL CMainWindow::GetInterface(
- IUnknown* pObj,
- REFIID riid,
- PPVOID ppv)
- {
- BOOL bResult = FALSE;
- HRESULT hr;
- *ppv=NULL;
- if (NULL != pObj)
- {
- LOG("C: --Obtaining Interface Pointer.");
- hr=pObj->QueryInterface(riid, ppv);
- if (FAILED(hr))
- {
- LOGERROR("C: ???? QueryInterface", hr);
- }
- else
- {
- LOGF1("C: Interface obtained. *ppv=0x%X", *ppv);
- bResult = TRUE;
- }
- }
- else
- {
- LOG("C: ???? Create an object first.");
- }
- return (bResult);
- }
- /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
- Method: CMainWindow::DoMenu
- Summary: Dispatch and handle the main menu commands.
- Args: WPARAM wParam,
- First message parameter (word sized).
- LPARAM lParam)
- Second message parameter (long sized).
- Modifies: m_ofnFile, ...
- Returns: LRESULT
- Standard Windows WindowProc return value.
- M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
- LRESULT CMainWindow::DoMenu(
- WPARAM wParam,
- LPARAM lParam)
- {
- LRESULT lResult = FALSE;
- HRESULT hr;
- HMENU hMenu = ::GetMenu(m_hWnd);
- IClassFactory2* pICF2LicCruiseCar;
- // Here are some interface pointers used to call methods on our COCar,
- // COUtilityCar, COLicCruiseCar, and COUtilityCruiseCar COM objects.
- ICar* pICar;
- IUtility* pIUtility;
- ICruise* pICruise;
- switch (LOWORD(wParam))
- {
- //----------------------------------------------------------------------
- // Handle File Menu Commands.
- //----------------------------------------------------------------------
- case IDM_FILE_EXIT:
- // The user commands us to exit this application so we tell the
- // Main window to close itself.
- ::PostMessage(m_hWnd, WM_CLOSE, 0, 0);
- break;
- //----------------------------------------------------------------------
- // Handle Car Menu Commands.
- //----------------------------------------------------------------------
- case IDM_CAR_CREATE:
- LOG("C: === Car Menu: Create.");
- if (NULL == m_pCar)
- {
- // Call COM service to create an instance.
- hr = CoCreateInstance(
- CLSID_DllCar,
- NULL,
- CLSCTX_INPROC_SERVER,
- IID_IUnknown,
- (PPVOID)&m_pCar);
- if (SUCCEEDED(hr))
- {
- ::CheckMenuItem(
- hMenu,
- IDM_CAR_CREATE,
- MF_BYCOMMAND | MF_CHECKED);
- }
- else
- {
- LOGERROR("C: ???? CoCreateInstance",hr);
- }
- }
- else
- LOG("C: ???? Car already exists.");
- break;
- case IDM_CAR_RELEASE:
- LOG("C: === Car Menu: Release.");
- if (NULL != m_pCar)
- {
- RELEASE_INTERFACE(m_pCar);
- // We ask COM to unload any unused COM Servers.
- CoFreeUnusedLibraries();
- ::CheckMenuItem(
- hMenu,
- IDM_CAR_CREATE,
- MF_BYCOMMAND | MF_UNCHECKED);
- }
- else
- LOG("C: ???? No Car to Release.");
- break;
- case IDM_CAR_SHIFT:
- LOG("C: === Car Menu: ICar::Shift");
- if (GetInterface(m_pCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Shift");
- pICar->Shift(1);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_CAR_CLUTCH:
- LOG("C: === Car Menu: ICar::Clutch");
- if (GetInterface(m_pCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Clutch");
- pICar->Clutch(100);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_CAR_SPEED:
- LOG("C: === Car Menu: ICar::Speed");
- if (GetInterface(m_pCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Speed");
- pICar->Speed(20);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_CAR_STEER:
- LOG("C: === Car Menu: ICar::Steer");
- if (GetInterface(m_pCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Steer");
- pICar->Steer(0);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- //----------------------------------------------------------------------
- // Handle UtilityCar Menu Commands.
- //----------------------------------------------------------------------
- case IDM_UCAR_CREATE:
- LOG("C: === UtilityCar Menu: Create.");
- if (NULL == m_pUtilityCar)
- {
- // Call COM service to create an instance.
- hr = CoCreateInstance(
- CLSID_DllUtilityCar,
- NULL,
- CLSCTX_INPROC_SERVER,
- IID_IUnknown,
- (PPVOID)&m_pUtilityCar);
- if (SUCCEEDED(hr))
- {
- ::CheckMenuItem(
- hMenu,
- IDM_UCAR_CREATE,
- MF_BYCOMMAND | MF_CHECKED);
- }
- else
- {
- LOGERROR("C: ???? CoCreateInstance",hr);
- }
- }
- else
- LOG("C: ???? UtilityCar already exists.");
- break;
- case IDM_UCAR_RELEASE:
- LOG("C: === UtilityCar Menu: Release.");
- if (NULL != m_pUtilityCar)
- {
- RELEASE_INTERFACE(m_pUtilityCar);
- // We ask COM to unload any unused COM Servers.
- CoFreeUnusedLibraries();
- ::CheckMenuItem(
- hMenu,
- IDM_UCAR_CREATE,
- MF_BYCOMMAND | MF_UNCHECKED);
- }
- else
- LOG("C: ???? No UtilityCar to Release.");
- break;
- case IDM_UCAR_SHIFT:
- LOG("C: === UtilityCar Menu: ICar::Shift");
- if (GetInterface(m_pUtilityCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Shift");
- pICar->Shift(2);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_UCAR_CLUTCH:
- LOG("C: === UtilityCar Menu: ICar::Clutch");
- if (GetInterface(m_pUtilityCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Clutch");
- pICar->Clutch(100);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_UCAR_SPEED:
- LOG("C: === UtilityCar Menu: ICar::Speed");
- if (GetInterface(m_pUtilityCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Speed");
- pICar->Speed(30);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_UCAR_STEER:
- LOG("C: === UtilityCar Menu: ICar::Steer");
- if (GetInterface(m_pUtilityCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Steer");
- pICar->Steer(10);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_UCAR_OFFROAD:
- LOG("C: === UtilityCar Menu: IUtility::Offroad");
- if (GetInterface(m_pUtilityCar, IID_IUtility, (PPVOID)&pIUtility))
- {
- LOG("C: --Calling pIUtility->Offroad");
- pIUtility->Offroad(1);
- LOG("C: --Releasing pIUtility");
- pIUtility->Release();
- }
- break;
- case IDM_UCAR_WINCH:
- LOG("C: === UtilityCar Menu: IUtility::Winch");
- if (GetInterface(m_pUtilityCar, IID_IUtility, (PPVOID)&pIUtility))
- {
- LOG("C: --Calling pIUtility->Winch");
- pIUtility->Winch(0);
- LOG("C: --Releasing pIUtility");
- pIUtility->Release();
- }
- break;
- //----------------------------------------------------------------------
- // Handle LicCruiseCar Menu Commands.
- //----------------------------------------------------------------------
- case IDM_CCAR_CREATE:
- LOG("C: === LicCruiseCar Menu: Create.");
- if (NULL == m_pLicCruiseCar)
- {
- // Call COM service to create an instance.
- hr = CoCreateInstance(
- CLSID_LicCruiseCar,
- NULL,
- CLSCTX_INPROC_SERVER,
- IID_IUnknown,
- (PPVOID)&m_pLicCruiseCar);
- if (SUCCEEDED(hr))
- {
- ::CheckMenuItem(
- hMenu,
- IDM_CCAR_CREATE,
- MF_BYCOMMAND | MF_CHECKED);
- }
- else
- {
- LOGERROR("C: ???? CoCreateInstance",hr);
- if (CLASS_E_NOTLICENSED == hr)
- m_pMsgBox->ErrorID(IDS_NOLICENSE);
- }
- }
- else
- LOG("C: ???? LicCruiseCar already exists.");
- break;
- case IDM_CCAR_RELEASE:
- LOG("C: === LicCruiseCar Menu: Release.");
- if (NULL != m_pLicCruiseCar)
- {
- RELEASE_INTERFACE(m_pLicCruiseCar);
- // We ask COM to unload any unused COM Servers.
- CoFreeUnusedLibraries();
- ::CheckMenuItem(
- hMenu,
- IDM_CCAR_CREATE,
- MF_BYCOMMAND | MF_UNCHECKED);
- }
- else
- LOG("C: ???? No LicCruiseCar to Release.");
- break;
- case IDM_CCAR_SHIFT:
- LOG("C: === LicCruiseCar Menu: ICar::Shift");
- if (GetInterface(m_pLicCruiseCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Shift");
- pICar->Shift(4);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_CCAR_CLUTCH:
- LOG("C: === LicCruiseCar Menu: ICar::Clutch");
- if (GetInterface(m_pLicCruiseCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Clutch");
- pICar->Clutch(100);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_CCAR_SPEED:
- LOG("C: === LicCruiseCar Menu: ICar::Speed");
- if (GetInterface(m_pLicCruiseCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Speed");
- pICar->Speed(60);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_CCAR_STEER:
- LOG("C: === LicCruiseCar Menu: ICar::Steer");
- if (GetInterface(m_pLicCruiseCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Steer");
- pICar->Steer(0);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_CCAR_ENGAGE:
- LOG("C: === LicCruiseCar Menu: ICruise::Engage");
- if (GetInterface(m_pLicCruiseCar, IID_ICruise, (PPVOID)&pICruise))
- {
- LOG("C: --Calling pICruise->Engage");
- pICruise->Engage(TRUE);
- LOG("C: --Releasing pICruise");
- pICruise->Release();
- }
- break;
- case IDM_CCAR_ADJUST:
- LOG("C: === LicCruiseCar Menu: ICruise::Adjust");
- if (GetInterface(m_pLicCruiseCar, IID_ICruise, (PPVOID)&pICruise))
- {
- LOG("C: --Calling pICruise->Adjust");
- pICruise->Adjust(FALSE);
- LOG("C: --Releasing pICruise");
- pICruise->Release();
- }
- break;
- //----------------------------------------------------------------------
- // Handle UtilityCruiseCar Menu Commands.
- //----------------------------------------------------------------------
- case IDM_UCRU_GETLICKEY:
- LOG("C: === UtilityCruiseCar Menu: Get License Key.");
- // Get a class factory for LicCruiseCar and issue IClassFactory's
- // CreateInstance method to manufacture a COLicCruiseCar COM object.
- hr = CoGetClassObject(
- CLSID_LicCruiseCar,
- CLSCTX_INPROC_SERVER,
- NULL,
- IID_IClassFactory2,
- (PPVOID)&pICF2LicCruiseCar);
- if (SUCCEEDED(hr))
- {
- LOG("C: LicCruiseCar's IClassFactory2 obtained.");
- hr = pICF2LicCruiseCar->RequestLicKey(0, &m_bstrLicKey);
- pICF2LicCruiseCar->Release();
- }
- if (SUCCEEDED(hr))
- {
- LOG("C: Obtained LicCruiseCar's Runtime License Key.");
- ::CheckMenuItem(
- hMenu,
- IDM_UCRU_GETLICKEY,
- MF_BYCOMMAND | MF_CHECKED);
- }
- else
- {
- LOGERROR("C: ???? CoGetClassObject",hr);
- LOG("C: Failed to Obtain LicCruiseCar's Runtime License Key.");
- }
- break;
- case IDM_UCRU_CLEARLICKEY:
- LOG("C: === UtilityCruiseCar Menu: Clear License Key.");
- SysFreeString(m_bstrLicKey);
- m_bstrLicKey = NULL;
- ::CheckMenuItem(
- hMenu,
- IDM_UCRU_GETLICKEY,
- MF_BYCOMMAND | MF_UNCHECKED);
- break;
- case IDM_UCRU_CREATE:
- LOG("C: === UtilityCruiseCar Menu: Create with Key.");
- if (NULL == m_pUtilityCruiseCar)
- {
- // Call a create function to create an instance.
- hr = CreateUtilityCruiseCar(
- NULL,
- IID_IUnknown,
- m_bstrLicKey,
- (PPVOID)&m_pUtilityCruiseCar);
- if (SUCCEEDED(hr))
- {
- ::CheckMenuItem(
- hMenu,
- IDM_UCRU_CREATE,
- MF_BYCOMMAND | MF_CHECKED);
- }
- else
- {
- LOGERROR("C: ???? UtilityCruiseCar creation",hr);
- if (CLASS_E_NOTLICENSED == hr)
- m_pMsgBox->ErrorID(IDS_NORUNLICENSE);
- }
- }
- else
- LOG("C: ???? UtilityCruiseCar already exists.");
- break;
- case IDM_UCRU_RELEASE:
- LOG("C: === UtilityCruiseCar Menu: Release.");
- if (NULL != m_pUtilityCruiseCar)
- {
- RELEASE_INTERFACE(m_pUtilityCruiseCar);
- // We ask COM to unload any unused COM Servers.
- CoFreeUnusedLibraries();
- ::CheckMenuItem(
- hMenu,
- IDM_UCRU_CREATE,
- MF_BYCOMMAND | MF_UNCHECKED);
- }
- else
- LOG("C: ???? No UtilityCruiseCar to Release.");
- break;
- case IDM_UCRU_SHIFT:
- LOG("C: === UtilityCruiseCar Menu: ICar::Shift");
- if (GetInterface(m_pUtilityCruiseCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Shift");
- pICar->Shift(1);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_UCRU_CLUTCH:
- LOG("C: === UtilityCruiseCar Menu: ICar::Clutch");
- if (GetInterface(m_pUtilityCruiseCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Clutch");
- pICar->Clutch(80);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_UCRU_SPEED:
- LOG("C: === UtilityCruiseCar Menu: ICar::Speed");
- if (GetInterface(m_pUtilityCruiseCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Speed");
- pICar->Speed(10);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_UCRU_STEER:
- LOG("C: === UtilityCruiseCar Menu: ICar::Steer");
- if (GetInterface(m_pUtilityCruiseCar, IID_ICar, (PPVOID)&pICar))
- {
- LOG("C: --Calling pICar->Steer");
- pICar->Steer(10);
- LOG("C: --Releasing pICar");
- pICar->Release();
- }
- break;
- case IDM_UCRU_ENGAGE:
- LOG("C: === UtilityCruiseCar Menu: ICruise::Engage");
- if (GetInterface(m_pUtilityCruiseCar, IID_ICruise, (PPVOID)&pICruise))
- {
- LOG("C: --Calling pICruise->Engage");
- pICruise->Engage(FALSE);
- LOG("C: --Releasing pICruise");
- pICruise->Release();
- }
- break;
- case IDM_UCRU_ADJUST:
- LOG("C: === UtilityCruiseCar Menu: ICruise::Adjust");
- if (GetInterface(m_pUtilityCruiseCar, IID_ICruise, (PPVOID)&pICruise))
- {
- LOG("C: --Calling pICruise->Adjust");
- pICruise->Adjust(FALSE);
- LOG("C: --Releasing pICruise");
- pICruise->Release();
- }
- break;
- case IDM_UCRU_OFFROAD:
- LOG("C: === UtilityCruiseCar Menu: IUtility::Offroad");
- if (GetInterface(m_pUtilityCruiseCar, IID_IUtility, (PPVOID)&pIUtility))
- {
- LOG("C: --Calling pIUtility->Offroad");
- pIUtility->Offroad(3);
- LOG("C: --Releasing pIUtility");
- pIUtility->Release();
- }
- break;
- case IDM_UCRU_WINCH:
- LOG("C: === UtilityCruiseCar Menu: IUtility::Winch");
- if (GetInterface(m_pUtilityCruiseCar, IID_IUtility, (PPVOID)&pIUtility))
- {
- LOG("C: --Calling pIUtility->Winch");
- pIUtility->Winch(0);
- LOG("C: --Releasing pIUtility");
- pIUtility->Release();
- }
- break;
- //----------------------------------------------------------------------
- // Handle Log Menu Commands.
- //----------------------------------------------------------------------
- case IDM_LOG_LOGCLEAR:
- // Clear the message log.
- m_pMsgLog->Clear();
- // Use macro to log messages.
- LOGID(IDS_START_MESSAGE_LOG);
- break;
- case IDM_LOG_LICSERVER:
- // Toggle the state of the Message Logging from LICSERVE.
- // Toggle the checkmark indicator on the menu selection as well.
- {
- BOOL bLogServer = ::GetMenuState(
- hMenu,
- IDM_LOG_LICSERVER,
- MF_BYCOMMAND) & MF_CHECKED;
- if (bLogServer)
- {
- LOG("C: Stop LICSERVE Server Trace Logging.");
- RELEASE_INTERFACE(m_pLicCarSample);
- // We ask COM to unload any unused COM Servers.
- CoFreeUnusedLibraries();
- ::CheckMenuItem(
- hMenu,
- IDM_LOG_LICSERVER,
- MF_BYCOMMAND | MF_UNCHECKED);
- }
- else
- {
- LOG("C: Start LICSERVE Server Trace Logging.");
- hr = CoCreateInstance(
- CLSID_LicCarSample,
- NULL,
- CLSCTX_INPROC_SERVER,
- IID_ISample,
- (PPVOID)&m_pLicCarSample);
- if (SUCCEEDED(hr))
- hr = m_pLicCarSample->Init(m_hWnd, (PVOID)g_pMsgLog);
- if (SUCCEEDED(hr))
- {
- ::CheckMenuItem(
- hMenu,
- IDM_LOG_LICSERVER,
- MF_BYCOMMAND | MF_CHECKED);
- }
- else
- {
- RELEASE_INTERFACE(m_pLicCarSample);
- // We ask COM to unload any unused COM Servers.
- CoFreeUnusedLibraries();
- m_pMsgBox->ErrorID(IDS_NOLICSERVER);
- }
- }
- }
- break;
- case IDM_LOG_DLLSERVER:
- // Toggle the state of the Message Logging from DLLSERVE.
- // Toggle the checkmark indicator on the menu selection as well.
- {
- BOOL bLogServer = ::GetMenuState(
- hMenu,
- IDM_LOG_DLLSERVER,
- MF_BYCOMMAND) & MF_CHECKED;
- if (bLogServer)
- {
- LOG("C: Stop DLLSERVE Server Trace Logging.");
- RELEASE_INTERFACE(m_pDllCarSample);
- // We ask COM to unload any unused COM Servers.
- CoFreeUnusedLibraries();
- ::CheckMenuItem(
- hMenu,
- IDM_LOG_DLLSERVER,
- MF_BYCOMMAND | MF_UNCHECKED);
- }
- else
- {
- LOG("C: Start DLLSERVE Server Trace Logging.");
- hr = CoCreateInstance(
- CLSID_DllCarSample,
- NULL,
- CLSCTX_INPROC_SERVER,
- IID_ISample,
- (PPVOID)&m_pDllCarSample);
- if (SUCCEEDED(hr))
- hr = m_pDllCarSample->Init(m_hWnd, (PVOID)g_pMsgLog);
- if (SUCCEEDED(hr))
- {
- ::CheckMenuItem(
- hMenu,
- IDM_LOG_DLLSERVER,
- MF_BYCOMMAND | MF_CHECKED);
- }
- else
- {
- RELEASE_INTERFACE(m_pDllCarSample);
- // We ask COM to unload any unused COM Servers.
- CoFreeUnusedLibraries();
- m_pMsgBox->ErrorID(IDS_NODLLSERVER);
- }
- }
- }
- break;
- case IDM_LOG_LOGGING:
- // Toggle the state of all Message Logging.
- // Toggle the checkmark indicator on the menu selection as well.
- {
- BOOL bLogging = ::GetMenuState(
- hMenu,
- IDM_LOG_LOGGING,
- MF_BYCOMMAND) & MF_CHECKED;
- if (bLogging)
- {
- m_pMsgLog->Logging(FALSE);
- ::CheckMenuItem(
- hMenu,
- IDM_LOG_LOGGING,
- MF_BYCOMMAND | MF_UNCHECKED);
- }
- else
- {
- m_pMsgLog->Logging(TRUE);
- ::CheckMenuItem(
- hMenu,
- IDM_LOG_LOGGING,
- MF_BYCOMMAND | MF_CHECKED);
- }
- }
- break;
- case IDM_LOG_COPYCLIP:
- // Copy trace message log to clipboard.
- m_pMsgLog->Copy();
- break;
- //----------------------------------------------------------------------
- // Handle Help Menu Commands.
- //----------------------------------------------------------------------
- case IDM_HELP_CONTENTS:
- // We have some stubbed support here for bringing up the online
- // Help for this application.
- ReadHelp(m_hWnd, m_szHelpFile);
- break;
- case IDM_HELP_TUTORIAL:
- // Call the APPUTIL utility function, ReadTutorial, to Browse the HTML
- // tutorial narrative file associated with this tutorial code sample.
- ReadTutorial(m_hInst, m_hWnd, TEXT(HTML_FILE_EXT));
- break;
- case IDM_HELP_TUTLICSERVER:
- // Call the APPUTIL utility function, ReadTutorial, to Browse the HTML
- // tutorial narrative file associated with the LICSERVE server.
- ReadTutorial(m_hInst, m_hWnd, TEXT(LICSERVE_TUTFILE_STR));
- break;
- case IDM_HELP_TUTDLLSERVER:
- // Call the APPUTIL utility function, ReadTutorial, to Browse the HTML
- // tutorial narrative file associated with the DLLSERVE server.
- ReadTutorial(m_hInst, m_hWnd, TEXT(DLLSERVE_TUTFILE_STR));
- break;
- case IDM_HELP_READSOURCE:
- // Call the APPUTIL utility function ReadSource to allow the
- // user to open and read any of the source files of LICCLIEN.
- ReadSource(m_hWnd, &m_ofnFile);
- break;
- case IDM_HELP_ABOUT:
- {
- CAboutBox dlgAboutBox;
- LOG("C: === Help Menu: About LICCLIEN.");
- // Show the standard About Box dialog for this EXE by telling the
- // dialog C++ object to show itself by invoking its ShowDialog
- // method. Pass it this EXE instance and the parent window handle.
- // Use a dialog resource ID for the dialog template stored in
- // this EXE module's resources.
- dlgAboutBox.ShowDialog(
- m_hInst,
- MAKEINTRESOURCE(IDM_HELP_ABOUT),
- m_hWnd);
- }
- break;
- case IDM_HELP_ABOUTLICSERVER:
- // Call the LicCarSample in LICSERVE to show its AboutBox dialog.
- LOG("C: === Help Menu: About LICSERVE.");
- if (NULL != m_pLicCarSample)
- m_pLicCarSample->AboutBox(m_hWnd);
- break;
- case IDM_HELP_ABOUTDLLSERVER:
- // Call the DllCarSample in DLLSERVE to show its AboutBox dialog.
- LOG("C: === Help Menu: About DLLSERVE.");
- if (NULL != m_pDllCarSample)
- m_pDllCarSample->AboutBox(m_hWnd);
- break;
- default:
- // Defer all messages NOT handled here to the Default Window Proc.
- lResult = ::DefWindowProc(m_hWnd, WM_COMMAND, wParam, lParam);
- break;
- }
- return(lResult);
- }
- /*M+M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M+++M
- Method: CMainWindow::WindowProc
- Summary: Main window procedure for this window object. See CVirWindow
- in the APPUTIL library (APPUTIL.CPP) for details on how this
- method gets called by the global WindowProc.
- Args: UINT uMsg,
- Windows message that is "sent" to this window.
- WPARAM wParam,
- First message parameter (word sized).
- LPARAM lParam)
- Second message parameter (long sized).
- Modifies: ...
- Returns: LRESULT
- Standard Windows WindowProc return value.
- M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M---M-M*/
- LRESULT CMainWindow::WindowProc(
- UINT uMsg,
- WPARAM wParam,
- LPARAM lParam)
- {
- LRESULT lResult = FALSE;
- switch (uMsg)
- {
- case WM_CREATE:
- {
- // Setup for painting text in this window.
- HDC hdc = GetDC(m_hWnd);
- ::GetTextMetrics(hdc, &m_tm);
- ::ReleaseDC(m_hWnd, hdc);
- }
- break;
- case WM_MEASUREITEM:
- // Get setup for painting text in this window.
- {
- LPMEASUREITEMSTRUCT lpmis = (LPMEASUREITEMSTRUCT) lParam;
- lpmis->itemHeight = m_tm.tmHeight + m_tm.tmExternalLeading;
- lpmis->itemWidth = m_wWidth;
- lResult = TRUE;
- }
- case WM_SIZE:
- // Handle a resize of this window.
- m_wWidth = LOWORD(lParam);
- m_wHeight = HIWORD(lParam);
- // Resize the Message Log ListBox
- m_pMsgLog->Resize(m_wWidth, m_wHeight);
- break;
- case WM_COMMAND:
- // Dispatch and handle any Menu command messages received.
- lResult = DoMenu(wParam, lParam);
- break;
- case WM_CLOSE:
- // The user selected Close on the main window's System menu
- // or Exit on the File menu.
- case WM_QUIT:
- // If the app is being quit then close any associated help windows.
- default:
- // Defer all messages NOT handled here to the Default Window Proc.
- lResult = ::DefWindowProc(m_hWnd, uMsg, wParam, lParam);
- break;
- }
- return(lResult);
- }
- /*F+F++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
- Function: UnicodeOk
- Summary: Checks if the platform will handle unicode versions of
- Win32 string API calls.
- Args: void
- Returns: BOOL
- TRUE if unicode support; FALSE if not.
- ------------------------------------------------------------------------F-F*/
- BOOL UnicodeOk(void)
- {
- BOOL bOk = TRUE;
- TCHAR szUserName[MAX_STRING_LENGTH];
- DWORD dwSize = MAX_STRING_LENGTH;
- if (!GetUserName(szUserName, &dwSize))
- bOk = ERROR_CALL_NOT_IMPLEMENTED == GetLastError() ? FALSE : TRUE;
- return bOk;
- }
- /*F+F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F
- Function: InitApplication
- Summary: Initializes the application and registers its main window
- class. InitApplication is called only once (in WinMain).
- Args: HINSTANCE hInstance)
- Handle to the first instance of the application.
- Returns: BOOL.
- TRUE if success.
- FALSE if fail.
- F---F---F---F---F---F---F---F---F---F---F---F---F---F---F---F---F---F---F-F*/
- BOOL InitApplication(
- HINSTANCE hInstance)
- {
- BOOL bOk;
- // The window class for all instances of the main frame window.
- WNDCLASSEX wcf;
- // Assign the appropriate values for this main frame window class.
- wcf.cbSize = sizeof(WNDCLASSEX);
- wcf.cbClsExtra = 0; // No per-class extra data.
- wcf.cbWndExtra = 0; // No per-window extra data.
- wcf.hInstance = hInstance; // Application module instance.
- wcf.lpfnWndProc = &WindowProc; // Global Window Procedure (defined in
- // APPUTIL for all CVirWindows).
- wcf.hCursor = LoadCursor(NULL, IDC_ARROW); // Load app cursor.
- wcf.hIcon = (HICON) LoadIcon( // Load app icon.
- hInstance,
- TEXT("AppIcon"));
- wcf.hIconSm = (HICON) LoadImage( // Load small icon.
- hInstance,
- TEXT("AppIcon"),
- IMAGE_ICON,
- 16, 16,
- 0);
- wcf.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); // Default backgnd color.
- wcf.style = CS_HREDRAW | CS_VREDRAW; // Class style(s).
- wcf.lpszClassName = TEXT(MAIN_WINDOW_CLASS_NAME_STR); // Class name.
- wcf.lpszMenuName = TEXT(MAIN_WINDOW_CLASS_MENU_STR); // Menu name.
- // Register the window class and return FALSE if unsuccesful.
- bOk = RegisterClassEx(&wcf);
- return (bOk);
- }
- /*F+F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F+++F
- Function: WinMain
- Summary: The Windows main entry point function for this application.
- Initializes the application, the COM Libraries, and starts
- the main application message loop.
- Args: HINSTANCE hInstance,
- Instance handle; a new one for each invocation of this app.
- HINSTANCE hPrevInstance,
- Instance handle of the previous instance. NULL in Win32.
- LPSTR lpCmdLine,
- Windows passes a pointer to the application's
- invocation command line.
- int nCmdShow)
- Bits telling the show state of the application.
- Returns: int
- msg.wParam (upon exit of message loop).
- FALSE if this instance couldn't initialize and run.
- F---F---F---F---F---F---F---F---F---F---F---F---F---F---F---F---F---F---F-F*/
- extern "C" int PASCAL WinMain(
- HINSTANCE hInstance,
- HINSTANCE hPrevInstance,
- LPSTR lpCmdLine,
- int nCmdShow)
- {
- CMainWindow* pWin = NULL;
- MSG msg;
- HACCEL hAccel;
- int iRun = FALSE;
- // If we were compiled for UNICODE and the platform seems OK with this
- // then proceed. Else we error and exit the app.
- if (UnicodeOk())
- {
- // Call to initialize the COM Library. Use the SUCCEEDED macro
- // to detect success. If fail then exit app with error message.
- if (SUCCEEDED(CoInitialize(NULL)))
- {
- // If we succeeded in initializing the COM Library we proceed to
- // initialize the application. If we can't init the application
- // then we signal shut down with an error message exit.
- iRun = InitApplication(hInstance);
- if (iRun)
- {
- // Assume we'll set iRun to TRUE when initialization is done.
- iRun = FALSE;
- // We are still go for running so we try to create a nifty new
- // CMainWindow object for this app instance.
- pWin = new CMainWindow;
- if (NULL != pWin)
- {
- // Now we initialize an instance of the new CMainWindow.
- // This includes creating the main window.
- if (pWin->InitInstance(hInstance, nCmdShow))
- {
- // Load the keyboard accelerators from the resources.
- hAccel = LoadAccelerators(hInstance, TEXT("AppAccel"));
- if (NULL != hAccel)
- {
- // Signal App Initialization is successfully done.
- iRun = TRUE;
- }
- }
- }
- }
- if (iRun)
- {
- // If we initialized the app instance properly then we are still
- // go for running. We then start up the main message pump for
- // the application.
- while (GetMessage(&msg, NULL, 0, 0))
- {
- if (!TranslateAccelerator(
- pWin->GetHwnd(),
- hAccel,
- &msg))
- {
- TranslateMessage(&msg);
- DispatchMessage(&msg);
- }
- }
- // We ask COM to unload any unused COM Servers.
- CoFreeUnusedLibraries();
- // We'll pass to Windows the reason why we exited the message loop.
- iRun = msg.wParam;
- }
- else
- {
- // We failed to initialize the application. Put up error message
- // box saying that application couldn't be initialized. Parent
- // window is desktop (ie, NULL). Exit the failed application
- // (ie, by returning FALSE to WinMain).
- ErrorBox(hInstance, NULL, IDS_APPINITFAILED);
- // Delete the CMainWindow object.
- DELETE_POINTER(pWin);
- }
- // We're exiting this app (either normally or by init failure) so
- // shut down the COM Library.
- CoUninitialize();
- }
- else
- {
- // We failed to Initialize the COM Library. Put up error message box
- // saying that COM Library couldn't be initialized. Parent window
- // is desktop (ie, NULL). Exit the failed application (ie, by
- // returning FALSE to WinMain).
- ErrorBox(hInstance, NULL, IDS_COMINITFAILED);
- }
- }
- else
- {
- // If we were compiled for UNICODE but the platform has problems with
- // this then indicate an error and exit the app immediately.
- CHAR szMsg[MAX_STRING_LENGTH];
- if (LoadStringA(
- hInstance,
- IDS_NOUNICODE,
- szMsg,
- MAX_STRING_LENGTH))
- {
- MessageBoxA(
- NULL,
- szMsg,
- ERROR_TITLE_STR,
- MB_OK | MB_ICONEXCLAMATION);
- }
- }
- return iRun;
- }