////////////////////////////////////////////////////////////////////////// // // container.cpp // // This file contains the complete implementation of an ActiveX // control container. This purpose of this container is to test // a single control being hosted. // // (C) Copyright 1997 by Microsoft Corporation. All rights reserved. // ////////////////////////////////////////////////////////////////////////// #include #include #include "container.h" /** * This method is the constructor for the Container object. */ Container::Container() { m_cRefs = 1; m_hwnd = NULL; m_punk = NULL; memset(&m_rect, 0, sizeof(m_rect)); } /** * This method is the destructor for the Container object. */ Container::~Container() { if (m_punk) { m_punk->Release(); m_punk=NULL; } } /** * This method is called when the caller wants an interface pointer. * * @param riid The interface being requested. * @param ppvObject The resultant object pointer. * * @return HRESULT S_OK, E_POINTER, E_NOINTERFACE */ STDMETHODIMP Container::QueryInterface(REFIID riid, PVOID *ppvObject) { if (!ppvObject) return E_POINTER; if (IsEqualIID(riid, IID_IOleClientSite)) *ppvObject = (IOleClientSite *)this; else if (IsEqualIID(riid, IID_IOleInPlaceSite)) *ppvObject = (IOleInPlaceSite *)this; else if (IsEqualIID(riid, IID_IOleInPlaceFrame)) *ppvObject = (IOleInPlaceFrame *)this; else if (IsEqualIID(riid, IID_IOleInPlaceUIWindow)) *ppvObject = (IOleInPlaceUIWindow *)this; else if (IsEqualIID(riid, IID_IOleControlSite)) *ppvObject = (IOleControlSite *)this; else if (IsEqualIID(riid, IID_IOleWindow)) *ppvObject = this; else if (IsEqualIID(riid, IID_IDispatch)) *ppvObject = (IDispatch *)this; else if (IsEqualIID(riid, IID_IUnknown)) *ppvObject = this; else { *ppvObject = NULL; return E_NOINTERFACE; } AddRef(); return S_OK; } /** * This method increments the current object count. * * @return ULONG The new reference count. */ ULONG Container::AddRef(void) { return ++m_cRefs; } /** * This method decrements the object count and deletes if necessary. * * @return ULONG Remaining ref count. */ ULONG Container::Release(void) { if (--m_cRefs) return m_cRefs; delete this; return 0; } // *********************************************************************** // IOleClientSite // *********************************************************************** HRESULT Container::SaveObject() { return E_NOTIMPL; } HRESULT Container::GetMoniker(DWORD dwAssign, DWORD dwWhichMoniker, LPMONIKER * ppMk) { return E_NOTIMPL; } HRESULT Container::GetContainer(LPOLECONTAINER * ppContainer) { return E_NOINTERFACE; } HRESULT Container::ShowObject() { return S_OK; } HRESULT Container::OnShowWindow(BOOL fShow) { return S_OK; } HRESULT Container::RequestNewObjectLayout() { return E_NOTIMPL; } // *********************************************************************** // IOleWindow // *********************************************************************** HRESULT Container::GetWindow(HWND * lphwnd) { if (!IsWindow(m_hwnd)) return S_FALSE; *lphwnd = m_hwnd; return S_OK; } HRESULT Container::ContextSensitiveHelp(BOOL fEnterMode) { return E_NOTIMPL; } // *********************************************************************** // IOleInPlaceSite // *********************************************************************** HRESULT Container::CanInPlaceActivate(void) { return S_OK; } HRESULT Container::OnInPlaceActivate(void) { return S_OK; } HRESULT Container::OnUIActivate(void) { return S_OK; } HRESULT Container::GetWindowContext (IOleInPlaceFrame ** ppFrame, IOleInPlaceUIWindow ** ppIIPUIWin, LPRECT lprcPosRect, LPRECT lprcClipRect, LPOLEINPLACEFRAMEINFO lpFrameInfo) { *ppFrame = (IOleInPlaceFrame *)this; *ppIIPUIWin = NULL; RECT rect; GetClientRect(m_hwnd, &rect); lprcPosRect->left = 0; lprcPosRect->top = 0; lprcPosRect->right = rect.right; lprcPosRect->bottom = rect.bottom; CopyRect(lprcClipRect, lprcPosRect); lpFrameInfo->cb = sizeof(OLEINPLACEFRAMEINFO); lpFrameInfo->fMDIApp = FALSE; lpFrameInfo->hwndFrame = m_hwnd; lpFrameInfo->haccel = 0; lpFrameInfo->cAccelEntries = 0; (*ppFrame)->AddRef(); return S_OK; } HRESULT Container::Scroll(SIZE scrollExtent) { return E_NOTIMPL; } HRESULT Container::OnUIDeactivate(BOOL fUndoable) { return E_NOTIMPL; } HRESULT Container::OnInPlaceDeactivate(void) { return S_OK; } HRESULT Container::DiscardUndoState(void) { return E_NOTIMPL; } HRESULT Container::DeactivateAndUndo(void) { return E_NOTIMPL; } HRESULT Container::OnPosRectChange(LPCRECT lprcPosRect) { return S_OK; } // *********************************************************************** // IOleInPlaceUIWindow // *********************************************************************** HRESULT Container::GetBorder(LPRECT lprectBorder) { return E_NOTIMPL; } HRESULT Container::RequestBorderSpace(LPCBORDERWIDTHS lpborderwidths) { return E_NOTIMPL; } HRESULT Container::SetBorderSpace(LPCBORDERWIDTHS lpborderwidths) { return E_NOTIMPL; } HRESULT Container::SetActiveObject(IOleInPlaceActiveObject * pActiveObject, LPCOLESTR lpszObjName) { return E_NOTIMPL; } // *********************************************************************** // IOleInPlaceFrame // *********************************************************************** HRESULT Container::InsertMenus(HMENU hmenuShared, LPOLEMENUGROUPWIDTHS lpMenuWidths) { return E_NOTIMPL; } HRESULT Container::SetMenu(HMENU hmenuShared, HOLEMENU holemenu, HWND hwndActiveObject) { return E_NOTIMPL; } HRESULT Container::RemoveMenus(HMENU hmenuShared) { return E_NOTIMPL; } HRESULT Container::SetStatusText(LPCOLESTR pszStatusText) { char status[MAX_PATH]; // ansi version of status text if (NULL == pszStatusText) return E_POINTER; WideCharToMultiByte(CP_ACP, 0, pszStatusText, -1, status, MAX_PATH, NULL, NULL); if (IsWindow(m_hwndStatus)) SendMessage(m_hwndStatus, SB_SETTEXT, (WPARAM)0, (LPARAM)status); return (S_OK); } HRESULT Container::EnableModeless(BOOL fEnable) { return E_NOTIMPL; } HRESULT Container::TranslateAccelerator(LPMSG lpmsg, WORD wID) { return S_OK; } // *********************************************************************** // IOleControlSite // *********************************************************************** HRESULT Container::OnControlInfoChanged() { return E_NOTIMPL; } HRESULT Container::LockInPlaceActive(BOOL fLock) { return E_NOTIMPL; } HRESULT Container::GetExtendedControl(IDispatch **ppDisp) { if (ppDisp == NULL) return E_INVALIDARG; *ppDisp = (IDispatch *)this; (*ppDisp)->AddRef(); return S_OK; } HRESULT Container::TransformCoords(POINTL *pptlHimetric, POINTF *pptfContainer, DWORD dwFlags) { return E_NOTIMPL; } HRESULT Container::TranslateAccelerator(LPMSG pMsg, DWORD grfModifiers) { return S_FALSE; } HRESULT Container::OnFocus(BOOL fGotFocus) { return E_NOTIMPL; } HRESULT Container::ShowPropertyFrame(void) { return E_NOTIMPL; } // *********************************************************************** // IDispatch // *********************************************************************** HRESULT Container::GetIDsOfNames(REFIID riid, OLECHAR FAR* FAR* rgszNames, unsigned int cNames, LCID lcid, DISPID FAR* rgdispid) { *rgdispid = DISPID_UNKNOWN; return DISP_E_UNKNOWNNAME; } HRESULT Container::GetTypeInfo(unsigned int itinfo, LCID lcid, ITypeInfo FAR* FAR* pptinfo) { return E_NOTIMPL; } HRESULT Container::GetTypeInfoCount(unsigned int FAR * pctinfo) { return E_NOTIMPL; } HRESULT Container::Invoke(DISPID dispid, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS FAR *pdispparams, VARIANT FAR *pvarResult, EXCEPINFO FAR * pexecinfo, unsigned int FAR *puArgErr) { return DISP_E_MEMBERNOTFOUND; } // *********************************************************************** // Public (non-interface) Methods // *********************************************************************** /** * This method will add an ActiveX control to the container. Note, for * now, this container can only have one control. * * @param bstrClsid The CLSID or PROGID of the control. * * @return No return value. */ void Container::add(BSTR bstrClsid) { CLSID clsid; // CLSID of the control object HRESULT hr; // return code CLSIDFromString(bstrClsid, &clsid); CoCreateInstance(clsid, NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER, IID_IUnknown, (PVOID *)&m_punk); if (!m_punk) return; IOleObject *pioo; hr = m_punk->QueryInterface(IID_IOleObject, (PVOID *)&pioo); if (FAILED(hr)) return; pioo->SetClientSite(this); pioo->Release(); IPersistStreamInit *ppsi; hr = m_punk->QueryInterface(IID_IPersistStreamInit, (PVOID *)&ppsi); if (SUCCEEDED(hr)) { ppsi->InitNew(); ppsi->Release(); } } /** * This method will remove the control from the container. * * @return No return value. */ void Container::remove() { if (!m_punk) return; HRESULT hr; IOleObject *pioo; IOleInPlaceObject *pipo; hr = m_punk->QueryInterface(IID_IOleObject, (PVOID *)&pioo); if (SUCCEEDED(hr)) { pioo->Close(OLECLOSE_NOSAVE); pioo->SetClientSite(NULL); pioo->Release(); } hr = m_punk->QueryInterface(IID_IOleInPlaceObject, (PVOID *)&pipo); if (SUCCEEDED(hr)) { pipo->UIDeactivate(); pipo->InPlaceDeactivate(); pipo->Release(); } m_punk->Release(); m_punk = NULL; } /** * This method sets the parent window. This is used by the container * so the control can parent itself. * * @param hwndParent The parent window handle. * * @return No return value. */ void Container::setParent(HWND hwndParent) { m_hwnd = hwndParent; } /** * This method will set the location of the control. * * @param x The top left. * @param y The top right. * @param width The width of the control. * @param height The height of the control. */ void Container::setLocation(int x, int y, int width, int height) { m_rect.left = x; m_rect.top = y; m_rect.right = width; m_rect.bottom = height; if (!m_punk) return; HRESULT hr; IOleInPlaceObject *pipo; hr = m_punk->QueryInterface(IID_IOleInPlaceObject, (PVOID *)&pipo); if (FAILED(hr)) return; pipo->SetObjectRects(&m_rect, &m_rect); pipo->Release(); } /** * Sets the visible state of the control. * * @param fVisible TRUE=visible, FALSE=hidden * @return No return value. */ void Container::setVisible(BOOL fVisible) { if (!m_punk) return; HRESULT hr; IOleObject *pioo; hr = m_punk->QueryInterface(IID_IOleObject, (PVOID *)&pioo); if (FAILED(hr)) return; if (fVisible) { pioo->DoVerb(OLEIVERB_INPLACEACTIVATE, NULL, this, 0, m_hwnd, &m_rect); pioo->DoVerb(OLEIVERB_SHOW, NULL, this, 0, m_hwnd, &m_rect); } else pioo->DoVerb(OLEIVERB_HIDE, NULL, this, 0, m_hwnd, NULL); pioo->Release(); } /** * This sets the focus to the control (a.k.a. UIActivate) * * @param fFocus TRUE=set, FALSE=remove * * @return No return value. */ void Container::setFocus(BOOL fFocus) { if (!m_punk) return; HRESULT hr; IOleObject *pioo; if (fFocus) { hr = m_punk->QueryInterface(IID_IOleObject, (PVOID *)&pioo); if (FAILED(hr)) return; pioo->DoVerb(OLEIVERB_UIACTIVATE, NULL, this, 0, m_hwnd, &m_rect); pioo->Release(); } } /** * If the container has an HWND for the status window (must be * common control), then this method is used to tell the container. * * @param hwndStatus Window handle of the status bar. * * @return No return value. */ void Container::setStatusWindow(HWND hwndStatus) { m_hwndStatus = hwndStatus; } /** * This method gives the control the opportunity to translate and use * key strokes. * * @param msg Key message. * * @return No return value. */ void Container::translateKey(MSG msg) { if (!m_punk) return; HRESULT hr; IOleInPlaceActiveObject *pao; hr = m_punk->QueryInterface(IID_IOleInPlaceActiveObject, (PVOID *)&pao); if (FAILED(hr)) return; pao->TranslateAccelerator(&msg); pao->Release(); } /** * Returns the IDispatch pointer of the contained control. Note, the * caller is responsible for calling IDispatch::Release(). * * @return Controls dispatch interface. */ IDispatch * Container::getDispatch() { if (!m_punk) return NULL; HRESULT hr; IDispatch *pdisp; hr = m_punk->QueryInterface(IID_IDispatch, (PVOID *)&pdisp); return pdisp; } /** * Returns the IUnknown interface pointer for the containd control. Note, * the caller is responsible for calling IUnknown::Release(). * * @return Controls unknown interface. */ IUnknown * Container::getUnknown() { if (!m_punk) return NULL; m_punk->AddRef(); return m_punk; }