CPOLY.CPP
资源名称:MSDN_VC98.zip [点击查看]
上传用户:bangxh
上传日期:2007-01-31
资源大小:42235k
文件大小:19k
源码类别:
Windows编程
开发平台:
Visual C++
- /***
- *cpoly.cpp
- *
- * This is a part of the Microsoft Source Code Samples.
- *
- * Copyright (C) 1992-1997 Microsoft Corporation. All rights reserved.
- *
- * This source code is only intended as a supplement to Microsoft Development
- * Tools and/or WinHelp documentation. See these sources for detailed
- * information regarding the Microsoft samples programs.
- *
- *Purpose:
- * This module implements the CPoly and CPolyCF classes.
- *
- * This module is intended as a sample implementation of the IDispatch
- * interface, and its purpose is to demonstrate how an object can
- * expose methods and properties for programatic and cross-process
- * access via the IDispatch interface.
- *
- *Implementation Notes:
- *
- *****************************************************************************/
- #include "spoly.h"
- #include "cpoint.h"
- #include "cpoly.h"
- #include "cenumpt.h"
- #ifndef _MAC
- extern CStatBar FAR* g_psb;
- #endif
- #ifdef _MAC
- extern "C" WindowPtr g_pwndClient;
- #endif
- extern unsigned int g_fQuit;
- // our global list of polygons.
- //
- POLYLINK FAR* g_ppolylink = (POLYLINK FAR*)NULL;
- // global count of polygons.
- //
- int g_cPoly = 0;
- CPoly::CPoly()
- {
- m_refs = 0;
- m_xorg = 0;
- m_yorg = 0;
- m_width = 0;
- m_cPoints = 0;
- m_red = 0;
- m_green = 0;
- m_blue = 0;
- m_ptinfo = NULL;
- m_ppointlink = NULL;
- m_ppointlinkLast = NULL;
- }
- /***
- *CPoly *CPoly::Create(void)
- *Purpose:
- * Create an instance of a CPoly object, and add it to the global
- * list of polygons.
- *
- *Entry:
- * None
- *
- *Exit:
- * returns a polygon object, NULL the allocation failed.
- *
- ***********************************************************************/
- CPoly FAR*
- CPoly::Create()
- {
- HRESULT hresult;
- CPoly FAR* ppoly;
- ITypeInfo FAR* ptinfo;
- POLYLINK FAR* ppolylink;
- extern INTERFACEDATA NEAR g_idataCPoly;
- if((ppolylink = new FAR POLYLINK) == (POLYLINK FAR*)NULL)
- return (CPoly FAR*)NULL;
- if((ppoly = new FAR CPoly()) == (CPoly FAR*)NULL)
- return (CPoly FAR*)NULL;
- ppoly->AddRef();
- // create and attach its TypeInfo
- //
- hresult = CreateDispTypeInfo(&g_idataCPoly, LOCALE_SYSTEM_DEFAULT, &ptinfo);
- if(hresult != NOERROR)
- goto LError0;
- ppoly->m_ptinfo = ptinfo;
- // push the new polygon onto the front of the polygon list.
- //
- ++g_cPoly;
- ppolylink->ppoly = ppoly;
- ppolylink->next = g_ppolylink;
- g_ppolylink = ppolylink;
- #ifndef _MAC
- SBprintf(g_psb, TSTR("#poly = %d"), g_cPoly);
- #endif
- IncObjectCount();
- return ppoly;
- LError0:;
- ppoly->Release();
- return NULL;
- }
- //---------------------------------------------------------------------
- // IUnknown Methods
- //---------------------------------------------------------------------
- STDMETHODIMP
- CPoly::QueryInterface(REFIID riid, void FAR* FAR* ppv)
- {
- if(!IsEqualIID(riid, IID_IUnknown))
- if(!IsEqualIID(riid, IID_IDispatch)) {
- *ppv = NULL;
- return E_NOINTERFACE;
- }
- *ppv = this;
- AddRef();
- return NOERROR;
- }
- STDMETHODIMP_(unsigned long)
- CPoly::AddRef()
- {
- return ++m_refs;
- }
- STDMETHODIMP_(unsigned long)
- CPoly::Release()
- {
- POLYLINK FAR* FAR* pppolylink, FAR* ppolylinkDead;
- if(--m_refs == 0){
- Reset(); // release all CPoints
- // remove ourselves from the global list of polygons
- //
- for( pppolylink = &g_ppolylink;
- *pppolylink != NULL;
- pppolylink = &(*pppolylink)->next)
- {
- if((*pppolylink)->ppoly == this){
- ppolylinkDead = *pppolylink;
- *pppolylink = (*pppolylink)->next;
- delete ppolylinkDead;
- break;
- }
- }
- if(m_ptinfo != NULL){
- m_ptinfo->Release();
- }
- --g_cPoly;
- #ifndef _MAC
- SBprintf(g_psb, TSTR("#poly = %d"), g_cPoly);
- #endif
- delete this;
- DecObjectCount();
- return 0;
- }
- return m_refs;
- }
- //---------------------------------------------------------------------
- // IDispatch Methods
- //---------------------------------------------------------------------
- STDMETHODIMP
- CPoly::GetTypeInfoCount(unsigned int FAR* pctinfo)
- {
- // This object has a single *introduced* interface
- //
- *pctinfo = 1;
- return NOERROR;
- }
- STDMETHODIMP
- CPoly::GetTypeInfo(unsigned int itinfo, LCID lcid, ITypeInfo FAR* FAR* pptinfo)
- {
- UNUSED(lcid);
- if(itinfo != 0)
- return DISP_E_BADINDEX;
- m_ptinfo->AddRef();
- *pptinfo = m_ptinfo;
- return NOERROR;
- }
- /***
- *HRESULT CPoly::GetIDsOfNames(char**, unsigned int, LCID, DISPID*)
- *Purpose:
- * This method translates the given array of names to a corresponding
- * array of DISPIDs.
- *
- * This method deferrs to a common implementation shared by
- * both the CPoly and CPoint objects. See the description of
- * 'SPolyGetIDsOfNames()' in dispimpl.cpp for more information.
- *
- *Entry:
- * rgszNames = pointer to an array of names
- * cNames = the number of names in the rgszNames array
- * lcid = the callers locale ID
- *
- *Exit:
- * return value = HRESULT
- * rgdispid = array of DISPIDs corresponding to the rgszNames array
- * this array will contain -1 for each entry that is not known.
- *
- ***********************************************************************/
- STDMETHODIMP
- CPoly::GetIDsOfNames(
- REFIID riid,
- OLECHAR FAR* FAR* rgszNames,
- unsigned int cNames,
- LCID lcid,
- DISPID FAR* rgdispid)
- {
- UNUSED(lcid);
- if(!IsEqualIID(riid, IID_NULL))
- return DISP_E_UNKNOWNINTERFACE;
- return DispGetIDsOfNames(m_ptinfo, rgszNames, cNames, rgdispid);
- }
- /***
- *HRESULT CPoly::Invoke(...)
- *Purpose:
- * Dispatch a method or property request for objects of type CPoly.
- *
- * see the IDispatch document for more information, and a general
- * description of this method.
- *
- *Entry:
- * dispidMember = the DISPID of the member being requested
- *
- * riid = reference to the interface ID of the interface on this object
- * that the requested member belongs to. IID_NULL means to interpret
- * the member as belonging to the implementation defined "default"
- * or "primary" interface.
- *
- * lcid = the caller's locale ID
- *
- * wFlags = flags indicating the type of access being requested
- *
- * pdispparams = pointer to the DISPPARAMS struct containing the
- * requested members arguments (if any) and its named parameter
- * DISPIDs (if any).
- *
- *Exit:
- * return value = HRESULT
- * see the IDispatch spec for a description of possible success codes.
- *
- * pvarResult = pointer to a caller allocated VARIANT containing
- * the members return value (if any).
- *
- * pexcepinfo = caller allocated exception info structure, this will
- * be filled in only if an exception was raised that must be passed
- * up through Invoke to an enclosing handler.
- *
- * puArgErr = pointer to a caller allocated UINT, that will contain the
- * index of the offending argument if a DISP_E_TYPEMISMATCH error
- * was returned indicating that one of the arguments was of an
- * incorrect type and/or could not be reasonably coerced to a proper
- * type.
- *
- ***********************************************************************/
- STDMETHODIMP
- CPoly::Invoke(
- DISPID dispidMember,
- REFIID riid,
- LCID lcid,
- unsigned short wFlags,
- DISPPARAMS FAR* pdispparams,
- VARIANT FAR* pvarResult,
- EXCEPINFO FAR* pexcepinfo,
- unsigned int FAR* puArgErr)
- {
- UNUSED(lcid);
- if(!IsEqualIID(riid, IID_NULL))
- return DISP_E_UNKNOWNINTERFACE;
- return DispInvoke(
- this, m_ptinfo,
- dispidMember, wFlags, pdispparams,
- pvarResult, pexcepinfo, puArgErr);
- }
- //---------------------------------------------------------------------
- // Introduced Methods
- //---------------------------------------------------------------------
- /***
- *void CPoly::Draw(void)
- *Purpose:
- * Draw the polygon, using the current x/y origin and line width
- * properties.
- *
- *Entry:
- * None
- *
- *Exit:
- * None
- *
- ***********************************************************************/
- void METHODCALLTYPE EXPORT
- CPoly::Draw()
- {
- short xorg, yorg;
- POINTLINK FAR* ppointlinkFirst, FAR* ppointlink;
- if((ppointlinkFirst = m_ppointlink) == (POINTLINK FAR*)NULL)
- return;
- #ifdef _MAC /* { */
- short x,y;
- RGBColor rgb;
- WindowPtr pwndSaved;
- GetPort(&pwndSaved);
- SetPort(g_pwndClient);
- PenNormal();
- PenSize(m_width, m_width);
- rgb.red = m_red;
- rgb.green = m_green;
- rgb.blue = m_blue;
- RGBForeColor(&rgb);
- xorg = m_xorg;
- yorg = m_yorg;
- MoveTo(
- xorg + ppointlinkFirst->ppoint->m_x,
- yorg + ppointlinkFirst->ppoint->m_y);
- for(ppointlink = ppointlinkFirst->next;
- ppointlink != (POINTLINK FAR*)NULL;
- ppointlink = ppointlink->next)
- {
- x = xorg + ppointlink->ppoint->m_x;
- y = yorg + ppointlink->ppoint->m_y;
- LineTo(x, y);
- }
- LineTo(
- xorg + ppointlinkFirst->ppoint->m_x,
- yorg + ppointlinkFirst->ppoint->m_y);
- SetPort(pwndSaved);
- #else /* }{ */
- HDC hdc;
- RECT rect;
- HPEN hpen, hpenOld;
- extern HWND g_hwndClient;
- GetClientRect(g_hwndClient, &rect);
- xorg = m_xorg + (short) rect.left;
- yorg = m_yorg + (short) rect.top;
- hdc = GetDC(g_hwndClient);
- hpen = CreatePen(PS_SOLID, m_width, RGB(m_red, m_green, m_blue));
- hpenOld = (HPEN)SelectObject(hdc, hpen);
- #ifdef WIN32
- MoveToEx(hdc,
- xorg + ppointlinkFirst->ppoint->m_x,
- yorg + ppointlinkFirst->ppoint->m_y, NULL);
- #else
- MoveTo(hdc,
- xorg + ppointlinkFirst->ppoint->m_x,
- yorg + ppointlinkFirst->ppoint->m_y);
- #endif
- for(ppointlink = ppointlinkFirst->next;
- ppointlink != (POINTLINK FAR*)NULL;
- ppointlink = ppointlink->next)
- {
- LineTo(hdc,
- xorg + ppointlink->ppoint->m_x,
- yorg + ppointlink->ppoint->m_y);
- }
- LineTo(hdc,
- xorg + ppointlinkFirst->ppoint->m_x,
- yorg + ppointlinkFirst->ppoint->m_y);
- SelectObject(hdc, hpenOld);
- DeleteObject(hpen);
- ReleaseDC(g_hwndClient, hdc);
- #endif /* } */
- }
- /***
- *void CPoly::Reset(void)
- *Purpose:
- * Release all points referenced by this poly.
- *
- *Entry:
- * None
- *
- *Exit:
- * None
- *
- ***********************************************************************/
- void METHODCALLTYPE EXPORT
- CPoly::Reset()
- {
- POINTLINK FAR* ppointlink, FAR* ppointlinkNext;
- for(ppointlink = m_ppointlink;
- ppointlink != (POINTLINK FAR*)NULL;
- ppointlink = ppointlinkNext)
- {
- ppointlinkNext = ppointlink->next;
- ppointlink->ppoint->Release();
- delete ppointlink;
- }
- m_cPoints = 0;
- m_ppointlink = NULL;
- m_ppointlinkLast = NULL;
- }
- /***
- *HRESULT CPoly::AddPoint(short, short)
- *Purpose:
- * Add a CPoint with the given coordinates to the end of our current
- * list of points.
- *
- *Entry:
- * x,y = the x and y coordinates of the new point.
- *
- *Exit:
- * return value = HRESULT
- *
- ***********************************************************************/
- HRESULT METHODCALLTYPE
- CPoly::AddPoint(short x, short y)
- {
- CPoint FAR* ppoint;
- POINTLINK FAR* ppointlink;
- ppoint = CPoint::Create();
- if(ppoint == (CPoint FAR*)NULL)
- return E_OUTOFMEMORY;
- ppoint->SetX(x);
- ppoint->SetY(y);
- ppointlink = new FAR POINTLINK;
- if(ppointlink == (POINTLINK FAR*)NULL){
- delete ppoint;
- return E_OUTOFMEMORY;
- }
- ppointlink->ppoint = ppoint;
- ppointlink->next = (POINTLINK FAR*)NULL;
- if(m_ppointlinkLast == (POINTLINK FAR*)NULL){
- m_ppointlink = m_ppointlinkLast = ppointlink;
- }else{
- m_ppointlinkLast->next = ppointlink;
- m_ppointlinkLast = ppointlink;
- }
- ++m_cPoints;
- return NOERROR;
- }
- /***
- *IUnknown FAR* CPoly::EnumPoints(void);
- *Purpose:
- * Return and enumerator for the points in this polygon.
- *
- *Entry:
- * None
- *
- *Exit:
- * return value = HRESULT
- *
- * *ppenum = pointer to an IEnumVARIANT for the points in this polygon
- *
- ***********************************************************************/
- IUnknown FAR* METHODCALLTYPE EXPORT
- CPoly::EnumPoints()
- {
- unsigned int i;
- HRESULT hresult;
- VARIANT var;
- SAFEARRAY FAR* psa;
- IUnknown FAR* punk;
- CEnumPoint FAR* penum;
- POINTLINK FAR* ppointlink;
- SAFEARRAYBOUND rgsabound[1];
- rgsabound[0].lLbound = 0;
- rgsabound[0].cElements = m_cPoints;
- psa = SafeArrayCreate(VT_VARIANT, 1, rgsabound);
- if(psa == NULL){
- hresult = E_OUTOFMEMORY;
- goto LError0;
- }
- ppointlink = m_ppointlink;
- for(i = 0; i < m_cPoints; ++i){
- long ix[1];
- if(ppointlink == NULL){
- // this indicates an internal consistency error.
- // (this test should probably be an assertion)
- hresult = E_FAIL;
- goto LError1;
- }
- V_VT(&var) = VT_DISPATCH;
- hresult = ppointlink->ppoint->QueryInterface(
- IID_IDispatch, (void FAR* FAR*)&V_DISPATCH(&var));
- if(hresult != NOERROR)
- goto LError1;
- ix[0] = i;
- SafeArrayPutElement(psa, ix, &var);
- // SafeArrayPutElement adds a reference to the contents of the
- // variant, so we must free the variable we have.
- //
- VariantClear(&var);
- ppointlink = ppointlink->next;
- }
- hresult = CEnumPoint::Create(psa, &penum);
- if(hresult != NOERROR)
- goto LError1;
- // ownership of the array has passed to the enumerator
- psa = NULL;
- hresult = penum->QueryInterface(IID_IUnknown, (void FAR* FAR*)&punk);
- if(hresult != NOERROR)
- goto LError2;
- penum->Release();
- return punk;
- LError2:;
- penum->Release();
- LError1:;
- // destroy the array if we were not successful creating the enumerator.
- if(psa != NULL)
- SafeArrayDestroy(psa);
- LError0:;
- return NULL;
- }
- short METHODCALLTYPE EXPORT
- CPoly::GetXOrigin()
- {
- return m_xorg;
- }
- void METHODCALLTYPE EXPORT
- CPoly::SetXOrigin(short x)
- {
- m_xorg = x;
- }
- short METHODCALLTYPE EXPORT
- CPoly::GetYOrigin()
- {
- return m_yorg;
- }
- void METHODCALLTYPE EXPORT
- CPoly::SetYOrigin(short y)
- {
- m_yorg = y;
- }
- short METHODCALLTYPE EXPORT
- CPoly::GetWidth()
- {
- return m_width;
- }
- void METHODCALLTYPE EXPORT
- CPoly::SetWidth(short width)
- {
- m_width = width;
- }
- short METHODCALLTYPE EXPORT
- CPoly::get_red()
- {
- return m_red;
- }
- void METHODCALLTYPE EXPORT
- CPoly::set_red(short red)
- {
- m_red = red;
- }
- short METHODCALLTYPE EXPORT
- CPoly::get_green()
- {
- return m_green;
- }
- void METHODCALLTYPE EXPORT
- CPoly::set_green(short green)
- {
- m_green = green;
- }
- short METHODCALLTYPE EXPORT
- CPoly::get_blue()
- {
- return m_blue;
- }
- void METHODCALLTYPE EXPORT
- CPoly::set_blue(short blue)
- {
- m_blue = blue;
- }
- /***
- *void CPoly::Dump(void)
- *Purpose:
- * Output a debug dump of this instance.
- *
- *Entry:
- * None
- *
- *Exit:
- * None
- *
- ***********************************************************************/
- void METHODCALLTYPE EXPORT
- CPoly::Dump()
- {
- #ifdef _MAC
- // REVIEW: implement for the mac
- #else
- TCHAR buffer[80];
- POINTLINK FAR* ppointlink;
- wsprintf(buffer, TSTR("CPoly(0x%x) =n"), (int)this);
- OutputDebugString(buffer);
- wsprintf(buffer,
- TSTR(" xorg = %d, yorg = %d, width = %d, rgb = {%d,%d,%d}n points = "),
- m_xorg, m_yorg, m_width,
- get_red(),
- get_green(),
- get_blue());
- OutputDebugString(buffer);
- for(ppointlink = m_ppointlink;
- ppointlink != (POINTLINK FAR*)NULL;
- ppointlink = ppointlink->next)
- {
- wsprintf(buffer, TSTR("{%d,%d}"),
- ppointlink->ppoint->GetX(),
- ppointlink->ppoint->GetY());
- OutputDebugString(buffer);
- wsprintf(buffer, TSTR(" "));
- OutputDebugString(buffer);
- }
- wsprintf(buffer, TSTR("n"));
- OutputDebugString(buffer);
- #endif
- }
- /***
- *void CPoly::Quit(void)
- *Purpose:
- * Mark this instance as quitting.
- *
- *Entry:
- * None
- *
- *Exit:
- * None
- *
- ***********************************************************************/
- void METHODCALLTYPE EXPORT
- CPoly::Quit()
- {
- g_fQuit = 1;
- }
- /***
- *void CPoly::PolyDraw(void)
- *Purpose:
- * Draw all polygons.
- *
- *Entry:
- * None
- *
- *Exit:
- * None
- *
- ***********************************************************************/
- void
- CPoly::PolyDraw()
- {
- POLYLINK FAR* polylink;
- for(polylink = g_ppolylink;
- polylink != (POLYLINK FAR*)NULL;
- polylink = polylink->next)
- {
- polylink->ppoly->Draw();
- }
- }
- /***
- *void PolyTerm(void)
- *Purpose:
- * Release all polygons.
- *
- *Entry:
- * None
- *
- *Exit:
- * None
- *
- ***********************************************************************/
- void
- CPoly::PolyTerm()
- {
- POLYLINK FAR* ppolylink;
- POLYLINK FAR* ppolylinkNext;
- for(ppolylink = g_ppolylink;
- ppolylink != (POLYLINK FAR*)NULL;
- ppolylink = ppolylinkNext)
- {
- ppolylinkNext = ppolylink->next;
- ppolylink->ppoly->Release();
- delete ppolylink;
- }
- g_ppolylink = NULL;
- }
- /***
- *void PolyDump(void)
- *Purpose:
- * Invoke the debug Dump() method on all polygons were currently
- * holding on to.
- *
- *Entry:
- * None
- *
- *Exit:
- * None
- *
- ***********************************************************************/
- void
- CPoly::PolyDump()
- {
- POLYLINK FAR* ppolylink;
- if(g_ppolylink == (POLYLINK FAR*)NULL){
- #ifndef _MAC
- OutputDebugString(TSTR("t(none)n"));
- #endif
- return;
- }
- for(ppolylink = g_ppolylink;
- ppolylink != (POLYLINK FAR*)NULL;
- ppolylink = ppolylink->next)
- {
- ppolylink->ppoly->Dump();
- }
- }
- //---------------------------------------------------------------------
- // Implementation of the CPoly Class Factory
- //---------------------------------------------------------------------
- CPolyCF::CPolyCF()
- {
- m_refs = 0;
- }
- IClassFactory FAR*
- CPolyCF::Create()
- {
- CPolyCF FAR* pCF;
- if((pCF = new FAR CPolyCF()) == NULL)
- return NULL;
- pCF->AddRef();
- return pCF;
- }
- STDMETHODIMP
- CPolyCF::QueryInterface(REFIID riid, void FAR* FAR* ppv)
- {
- if(IsEqualIID(riid, IID_IUnknown) || IsEqualIID(riid, IID_IClassFactory)){
- *ppv = this;
- ++m_refs;
- return NOERROR;
- }
- *ppv = NULL;
- return E_NOINTERFACE;
- }
- STDMETHODIMP_(unsigned long)
- CPolyCF::AddRef(void)
- {
- return ++m_refs;
- }
- STDMETHODIMP_(unsigned long)
- CPolyCF::Release(void)
- {
- if(--m_refs == 0){
- delete this;
- return 0;
- }
- return m_refs;
- }
- STDMETHODIMP
- CPolyCF::CreateInstance(
- IUnknown FAR* punkOuter,
- REFIID iid,
- void FAR* FAR* ppv)
- {
- HRESULT hresult;
- CPoly FAR *ppoly;
- UNUSED(punkOuter);
- if((ppoly = CPoly::Create()) == NULL){
- *ppv = NULL;
- return E_OUTOFMEMORY;
- }
- hresult = ppoly->QueryInterface(iid, ppv);
- ppoly->Release();
- return hresult;
- }
- STDMETHODIMP
- #ifdef _MAC
- CPolyCF::LockServer(unsigned long fLock)
- #else
- CPolyCF::LockServer(BOOL fLock)
- #endif
- {
- UNUSED(fLock);
- return NOERROR;
- }