864 lines
27 KiB
C++
864 lines
27 KiB
C++
|
#include "stdafx.h"
|
||
|
#pragma hdrstop
|
||
|
|
||
|
#include "stdenum.h"
|
||
|
#include <mshtmdid.h>
|
||
|
|
||
|
#define SZ_ATL_SHEMBEDDING_WNDCLASS TEXT("ATL Shell Embedding")
|
||
|
|
||
|
HRESULT MakeSafeForScripting(IUnknown** ppDisp);
|
||
|
|
||
|
class ATL_NO_VTABLE CWebViewFolderContents
|
||
|
: public CComObjectRootEx<CComSingleThreadModel>
|
||
|
, public CComCoClass<CWebViewFolderContents, &CLSID_WebViewFolderContents>
|
||
|
, public CComControl<CWebViewFolderContents>
|
||
|
, public IDispatchImpl<IShellFolderViewDual, &IID_IShellFolderViewDual, &LIBID_Shell32, 1, 0, CComTypeInfoHolder>
|
||
|
, public IProvideClassInfo2Impl<&CLSID_WebViewFolderContents, NULL, &LIBID_Shell32, 1, 0, CComTypeInfoHolder>
|
||
|
, public IPersistImpl<CWebViewFolderContents>
|
||
|
, public IOleControlImpl<CWebViewFolderContents>
|
||
|
, public IOleObjectImpl<CWebViewFolderContents>
|
||
|
, public IViewObjectExImpl<CWebViewFolderContents>
|
||
|
, public IOleInPlaceActiveObjectImpl<CWebViewFolderContents>
|
||
|
, public IDataObjectImpl<CWebViewFolderContents>
|
||
|
, public IObjectSafetyImpl<CWebViewFolderContents>
|
||
|
, public IConnectionPointContainer
|
||
|
, public IOleInPlaceObject
|
||
|
, public IInternetSecurityMgrSite
|
||
|
, public IWebViewOCWinMan
|
||
|
{
|
||
|
public:
|
||
|
CWebViewFolderContents();
|
||
|
~CWebViewFolderContents();
|
||
|
|
||
|
DECLARE_POLY_AGGREGATABLE(CWebViewFolderContents);
|
||
|
DECLARE_NO_REGISTRY();
|
||
|
DECLARE_WND_CLASS(SZ_ATL_SHEMBEDDING_WNDCLASS)
|
||
|
|
||
|
BEGIN_COM_MAP(CWebViewFolderContents)
|
||
|
// ATL Uses these in IUnknown::QueryInterface()
|
||
|
COM_INTERFACE_ENTRY_IMPL_IID(IID_IViewObject, IViewObjectEx)
|
||
|
COM_INTERFACE_ENTRY_IMPL_IID(IID_IViewObject2, IViewObjectEx)
|
||
|
COM_INTERFACE_ENTRY_IMPL(IViewObjectEx)
|
||
|
COM_INTERFACE_ENTRY_IMPL_IID(IID_IOleWindow, IOleInPlaceActiveObject)
|
||
|
COM_INTERFACE_ENTRY_IMPL(IOleInPlaceActiveObject)
|
||
|
COM_INTERFACE_ENTRY(IOleInPlaceObject)
|
||
|
COM_INTERFACE_ENTRY(IDispatch)
|
||
|
COM_INTERFACE_ENTRY(IProvideClassInfo2)
|
||
|
COM_INTERFACE_ENTRY(IProvideClassInfo)
|
||
|
COM_INTERFACE_ENTRY_IMPL(IOleControl)
|
||
|
COM_INTERFACE_ENTRY_IMPL(IOleObject)
|
||
|
COM_INTERFACE_ENTRY_IMPL(IDataObject)
|
||
|
COM_INTERFACE_ENTRY_IMPL(IObjectSafety)
|
||
|
COM_INTERFACE_ENTRY(IConnectionPointContainer)
|
||
|
COM_INTERFACE_ENTRY(IShellFolderViewDual)
|
||
|
COM_INTERFACE_ENTRY_IMPL(IPersist)
|
||
|
COM_INTERFACE_ENTRY(IInternetSecurityMgrSite)
|
||
|
COM_INTERFACE_ENTRY(IWebViewOCWinMan)
|
||
|
END_COM_MAP()
|
||
|
|
||
|
|
||
|
// Declare the default message map
|
||
|
BEGIN_MSG_MAP(CWebViewFolderContents)
|
||
|
MESSAGE_HANDLER(WM_SIZE, _OnSizeMessage)
|
||
|
MESSAGE_HANDLER(WM_NOTIFY, _OnMessageForwarder)
|
||
|
MESSAGE_HANDLER(WM_CONTEXTMENU, _OnMessageForwarder)
|
||
|
MESSAGE_HANDLER(WM_SETCURSOR, _OnMessageForwarder)
|
||
|
MESSAGE_HANDLER(WM_ERASEBKGND, _OnEraseBkgndMessage)
|
||
|
END_MSG_MAP()
|
||
|
|
||
|
|
||
|
// *** IDispatch ***
|
||
|
virtual STDMETHODIMP Invoke(DISPID dispidMember, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS * pdispparams, VARIANT * pvarResult, EXCEPINFO * pexcepinfo, UINT * puArgErr);
|
||
|
virtual STDMETHODIMP GetTypeInfo(UINT itinfo, LCID lcid, ITypeInfo** pptinfo);
|
||
|
virtual STDMETHODIMP GetIDsOfNames(REFIID riid, OLECHAR **rgszNames, UINT cNames, LCID lcid, DISPID * rgdispid);
|
||
|
|
||
|
// *** IProvideClassInfo ***
|
||
|
virtual STDMETHODIMP GetClassInfo(ITypeInfo** pptinfo);
|
||
|
|
||
|
// *** IInternetSecurityMgrSite ***
|
||
|
// virtual STDMETHODIMP GetWindow(HWND * lphwnd); // Also in IOleWindow
|
||
|
virtual STDMETHODIMP EnableModeless(BOOL fEnable) { return IOleInPlaceActiveObjectImpl<CWebViewFolderContents>::EnableModeless(fEnable); }; // Also in IOleInPlaceActiveObject
|
||
|
|
||
|
// *** ShellFolderView ***
|
||
|
virtual STDMETHODIMP get_Application(IDispatch **ppid);
|
||
|
virtual STDMETHODIMP get_Parent (IDispatch **ppid);
|
||
|
virtual STDMETHODIMP get_Folder(Folder **ppid);
|
||
|
virtual STDMETHODIMP SelectedItems(FolderItems **ppid);
|
||
|
virtual STDMETHODIMP get_FocusedItem(FolderItem **ppid);
|
||
|
virtual STDMETHODIMP SelectItem(VARIANT *pvfi, int dwFlags);
|
||
|
virtual STDMETHODIMP PopupItemMenu(FolderItem * pfi, VARIANT vx, VARIANT vy, BSTR * pbs);
|
||
|
virtual STDMETHODIMP get_Script(IDispatch **ppid);
|
||
|
virtual STDMETHODIMP get_ViewOptions(long *plSetting);
|
||
|
|
||
|
// *** IOleWindow ***
|
||
|
virtual STDMETHODIMP GetWindow(HWND * lphwnd) { return IOleInPlaceActiveObjectImpl<CWebViewFolderContents>::GetWindow(lphwnd); };
|
||
|
virtual STDMETHODIMP ContextSensitiveHelp(BOOL fEnterMode) { return IOleInPlaceActiveObjectImpl<CWebViewFolderContents>::ContextSensitiveHelp(fEnterMode); };
|
||
|
|
||
|
// *** IOleObject ***
|
||
|
virtual STDMETHODIMP GetMiscStatus(DWORD dwAspect, DWORD *pdwStatus);
|
||
|
|
||
|
// *** IOleInPlaceObject ***
|
||
|
virtual STDMETHODIMP InPlaceDeactivate(void);
|
||
|
virtual STDMETHODIMP UIDeactivate(void) { return IOleInPlaceObject_UIDeactivate(); };
|
||
|
virtual STDMETHODIMP SetObjectRects(LPCRECT lprcPosRect, LPCRECT lprcClipRect);
|
||
|
virtual STDMETHODIMP ReactivateAndUndo(void) { return E_NOTIMPL; };
|
||
|
|
||
|
// *** IOleInPlaceActiveObject ***
|
||
|
// our frame was activated, better do the activation thing.
|
||
|
STDMETHODIMP OnFrameWindowActivate(BOOL fActivate)
|
||
|
{
|
||
|
if (_hwndLV && fActivate)
|
||
|
::SetFocus(_hwndLV);
|
||
|
return S_OK;
|
||
|
};
|
||
|
virtual STDMETHODIMP TranslateAccelerator(LPMSG pMsg);
|
||
|
|
||
|
// *** IWebViewOCWinMan ***
|
||
|
STDMETHODIMP SwapWindow(HWND hwndLV, IWebViewOCWinMan **pocWinMan);
|
||
|
|
||
|
// *** IConnectionPointContainer ***
|
||
|
STDMETHODIMP EnumConnectionPoints(IEnumConnectionPoints **ppEnum);
|
||
|
STDMETHODIMP FindConnectionPoint(REFIID riid, IConnectionPoint **ppCP);
|
||
|
|
||
|
// Over ride ATL functions.
|
||
|
LRESULT _OnMessageForwarder(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL & bHandled);
|
||
|
LRESULT _OnEraseBkgndMessage(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL & bHandled);
|
||
|
LRESULT _OnSizeMessage(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL & bHandled);
|
||
|
HRESULT DoVerbUIActivate(LPCRECT prcPosRect, HWND hwndParent);
|
||
|
HRESULT DoVerbInPlaceActivate(LPCRECT prcPosRect, HWND hwndParent);
|
||
|
virtual STDMETHODIMP Close(DWORD dwSaveOption);
|
||
|
|
||
|
protected:
|
||
|
// Helper functions;
|
||
|
HRESULT _SetupAutomationForwarders(void);
|
||
|
HRESULT _ReleaseAutomationForwarders(void);
|
||
|
HRESULT _OnInPlaceActivate(void);
|
||
|
void _ReleaseWindow(void);
|
||
|
void _ShowWindowLV(HWND hwndLV);
|
||
|
void _UnadviseAll();
|
||
|
|
||
|
class CConnectionPointForwarder : public IConnectionPoint
|
||
|
{
|
||
|
// IUnknown methods
|
||
|
|
||
|
virtual STDMETHODIMP QueryInterface(REFIID riid, LPVOID * ppvObj);
|
||
|
virtual STDMETHODIMP_(ULONG) AddRef(void);
|
||
|
virtual STDMETHODIMP_(ULONG) Release(void);
|
||
|
// IConnectionPoint methods
|
||
|
|
||
|
virtual STDMETHODIMP GetConnectionInterface(IID * pIID);
|
||
|
virtual STDMETHODIMP GetConnectionPointContainer(IConnectionPointContainer ** ppCPC);
|
||
|
virtual STDMETHODIMP Advise(LPUNKNOWN pUnkSink, DWORD * pdwCookie);
|
||
|
virtual STDMETHODIMP Unadvise(DWORD dwCookie);
|
||
|
virtual STDMETHODIMP EnumConnections(LPENUMCONNECTIONS * ppEnum) { return _pcpAuto->EnumConnections(ppEnum); }
|
||
|
|
||
|
IConnectionPoint * _pcpAuto;
|
||
|
HDSA _dsaCookies;
|
||
|
IUnknown* _punkParent;
|
||
|
friend class CWebViewFolderContents;
|
||
|
};
|
||
|
friend class CConnectionPointForwarder;
|
||
|
CConnectionPointForwarder m_cpEvents;
|
||
|
|
||
|
IDefViewFrame2 * _pdvf2; // defview
|
||
|
BOOL _fClientEdge;
|
||
|
BOOL _fTabRecieved;
|
||
|
BOOL _fCalledAutoArrange;
|
||
|
HWND _hwndLV;
|
||
|
HWND _hwndLVParent;
|
||
|
|
||
|
ITypeInfo * _pClassTypeInfo; // ITypeInfo of class
|
||
|
|
||
|
// stuff added to delegate all of our work up to DefViews automation
|
||
|
IShellFolderViewDual *_pdispAuto;
|
||
|
};
|
||
|
|
||
|
|
||
|
LCID g_lcidLocaleUnicpp = MAKELCID(LANG_USER_DEFAULT, SORT_DEFAULT);
|
||
|
|
||
|
// Helper function that can be used here and in main shell folder view window to get the
|
||
|
// current view options...
|
||
|
DWORD GetViewOptionsForDispatch()
|
||
|
{
|
||
|
SHELLSTATE ss = {0};
|
||
|
DWORD dwSetting = 0;
|
||
|
|
||
|
// Get the view options to return...
|
||
|
|
||
|
SHGetSetSettings(&ss,
|
||
|
SSF_SHOWALLOBJECTS|SSF_SHOWEXTENSIONS|SSF_SHOWCOMPCOLOR|
|
||
|
SSF_SHOWSYSFILES|SSF_DOUBLECLICKINWEBVIEW|SSF_DESKTOPHTML|SSF_WIN95CLASSIC,
|
||
|
FALSE);
|
||
|
|
||
|
// Aarg: mnuch the Bool:1 fields into a dword...
|
||
|
if (ss.fShowAllObjects) dwSetting |= SFVVO_SHOWALLOBJECTS;
|
||
|
if (ss.fShowExtensions) dwSetting |= SFVVO_SHOWEXTENSIONS;
|
||
|
if (ss.fShowCompColor) dwSetting |= SFVVO_SHOWCOMPCOLOR;
|
||
|
if (ss.fShowSysFiles) dwSetting |= SFVVO_SHOWSYSFILES;
|
||
|
if (ss.fDoubleClickInWebView) dwSetting |= SFVVO_DOUBLECLICKINWEBVIEW;
|
||
|
if (ss.fDesktopHTML) dwSetting |= SFVVO_DESKTOPHTML;
|
||
|
if (ss.fWin95Classic) dwSetting |= SFVVO_WIN95CLASSIC;
|
||
|
|
||
|
return dwSetting;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
CWebViewFolderContents::CWebViewFolderContents()
|
||
|
{
|
||
|
DllAddRef();
|
||
|
|
||
|
// This allocator should have zero inited the memory, so assert the member variables are empty.
|
||
|
ASSERT(!_pdvf2);
|
||
|
ASSERT(!_hwndLV);
|
||
|
ASSERT(!_hwndLVParent);
|
||
|
ASSERT(!_fClientEdge);
|
||
|
ASSERT(!_fTabRecieved);
|
||
|
ASSERT(!_pClassTypeInfo);
|
||
|
ASSERT(!m_cpEvents._dsaCookies);
|
||
|
ASSERT(!_fCalledAutoArrange);
|
||
|
|
||
|
m_bWindowOnly = TRUE;
|
||
|
m_bEnabled = TRUE;
|
||
|
m_bResizeNatural = TRUE;
|
||
|
m_cpEvents._punkParent = SAFECAST(this, IWebViewOCWinMan *);
|
||
|
GetWndClassInfo().m_wc.style &= ~(CS_HREDRAW|CS_VREDRAW);
|
||
|
}
|
||
|
|
||
|
CWebViewFolderContents::~CWebViewFolderContents()
|
||
|
{
|
||
|
_UnadviseAll();
|
||
|
ASSERT(NULL == _pdvf2);
|
||
|
ASSERT(NULL == _hwndLV);
|
||
|
|
||
|
if (_pClassTypeInfo)
|
||
|
_pClassTypeInfo->Release();
|
||
|
|
||
|
ATOMICRELEASE(_pdvf2);
|
||
|
_ReleaseAutomationForwarders();
|
||
|
|
||
|
DllRelease();
|
||
|
}
|
||
|
|
||
|
|
||
|
// ATL maintainence functions
|
||
|
LRESULT CWebViewFolderContents::_OnMessageForwarder(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL & bHandled)
|
||
|
{
|
||
|
if (_hwndLVParent)
|
||
|
{
|
||
|
bHandled = TRUE;
|
||
|
HWND hwnd = NULL;
|
||
|
|
||
|
// Forward these messages directly to DefView (don't let MSHTML eat them)
|
||
|
return ::SendMessage(_hwndLVParent, uMsg, wParam, lParam);
|
||
|
}
|
||
|
else
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
LRESULT CWebViewFolderContents::_OnEraseBkgndMessage(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL & bHandled)
|
||
|
{
|
||
|
// This function will just tell the default handler not to do anything and we
|
||
|
// will handle it.
|
||
|
|
||
|
// This is done in the case of WM_ERASEBKGND to...
|
||
|
// Avoid flicker by not erasing the background. This OC doesn't care
|
||
|
// about design-time issues - just usage on a Web View page.
|
||
|
bHandled = TRUE;
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
|
||
|
LRESULT CWebViewFolderContents::_OnSizeMessage(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL & bHandled)
|
||
|
{
|
||
|
// Now resize the DefView ListView window because ATL isn't very reliable at it.
|
||
|
if (_hwndLV)
|
||
|
{
|
||
|
// During re-parenting of _hwndLV by Trident, the size of the OC is not known to Trident.
|
||
|
// So, initially we get this message with 1 and 1 for the width and height, from it.
|
||
|
// And once Trident knows it's correct dimensions, we again get this message with the proper dim.
|
||
|
// We get a performance gain by not passing the first no-sense 1 x 1 dimension to _hwndLV.
|
||
|
|
||
|
if (m_rcPos.right - m_rcPos.left != 1 || m_rcPos.bottom - m_rcPos.top != 1)
|
||
|
{
|
||
|
::SetWindowPos(_hwndLV, 0, 0, 0, m_rcPos.right - m_rcPos.left, m_rcPos.bottom - m_rcPos.top, SWP_NOZORDER);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
bHandled = FALSE;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
HRESULT CWebViewFolderContents::DoVerbUIActivate(LPCRECT prcPosRect, HWND hwndParent)
|
||
|
{
|
||
|
HRESULT hr = IOleObjectImpl<CWebViewFolderContents>::DoVerbUIActivate(prcPosRect, hwndParent);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = _OnInPlaceActivate();
|
||
|
}
|
||
|
|
||
|
if (_hwndLV)
|
||
|
{
|
||
|
::SetFocus(_hwndLV);
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
// move from de-active to in-place-active
|
||
|
HRESULT CWebViewFolderContents::_OnInPlaceActivate(void)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
if (_pdvf2 == NULL)
|
||
|
{
|
||
|
hr = IUnknown_QueryService(m_spClientSite, SID_DefView, IID_IDefViewFrame2, (void **)&_pdvf2);
|
||
|
if (EVAL(SUCCEEDED(hr)))
|
||
|
{
|
||
|
hr = _pdvf2->GetWindowLV2(&_hwndLV, SAFECAST(this, IWebViewOCWinMan *));
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
_ShowWindowLV(_hwndLV); // we got it -- show the listview
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::DoVerbInPlaceActivate(LPCRECT prcPosRect, HWND hwndParent)
|
||
|
{
|
||
|
HRESULT hr = IOleObjectImpl<CWebViewFolderContents>::DoVerbInPlaceActivate(prcPosRect, hwndParent);
|
||
|
if (EVAL(SUCCEEDED(hr)))
|
||
|
{
|
||
|
hr = _OnInPlaceActivate();
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::InPlaceDeactivate(void)
|
||
|
{
|
||
|
_ReleaseWindow();
|
||
|
ATOMICRELEASE(_pdvf2);
|
||
|
|
||
|
return IOleInPlaceObject_InPlaceDeactivate();
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::SetObjectRects(LPCRECT prcPosRect, LPCRECT prcClipRect)
|
||
|
{
|
||
|
|
||
|
// WARNING: Do NOT move the EqualRect() comparison to after the
|
||
|
// IOleInPlaceObject_SetObjectRects declaration. The EqualRect()
|
||
|
// will always return an equal result then.
|
||
|
|
||
|
bool bPositionRectDifferent = (EqualRect(&m_rcPos, prcPosRect) == 0);
|
||
|
HRESULT hres = IOleInPlaceObject_SetObjectRects(prcPosRect, prcClipRect);
|
||
|
|
||
|
// 99/02/23 #294278 vtan: Trident did not call this routine when
|
||
|
// marquee selecting but it now does. The jumpy scrolling now
|
||
|
// manifests. Check that posRect has not changed before making
|
||
|
// any scroll position adjustments.
|
||
|
|
||
|
if (_hwndLV && _pdvf2 && (!_fCalledAutoArrange || bPositionRectDifferent))
|
||
|
{
|
||
|
_pdvf2->AutoAutoArrange(0);
|
||
|
_fCalledAutoArrange = TRUE;
|
||
|
}
|
||
|
|
||
|
return hres;
|
||
|
}
|
||
|
|
||
|
// *** IOleInPlaceActiveObject ***
|
||
|
HRESULT CWebViewFolderContents::TranslateAccelerator(LPMSG pMsg)
|
||
|
{
|
||
|
HRESULT hres = S_OK;
|
||
|
if (!_fTabRecieved)
|
||
|
{
|
||
|
hres = IOleInPlaceActiveObjectImpl<CWebViewFolderContents>::TranslateAccelerator(pMsg);
|
||
|
|
||
|
// If we did not handle this and if it is a tab (and we are not getting it in a cycle), forward it to trident, if present.
|
||
|
if (hres != S_OK && pMsg && (pMsg->wParam == VK_TAB || pMsg->wParam == VK_F6) && m_spClientSite)
|
||
|
{
|
||
|
IOleControlSite* pocs = NULL;
|
||
|
if (SUCCEEDED(m_spClientSite->QueryInterface(IID_IOleControlSite, (void **)&pocs)))
|
||
|
{
|
||
|
DWORD grfModifiers = 0;
|
||
|
if (GetKeyState(VK_SHIFT) & 0x8000)
|
||
|
{
|
||
|
grfModifiers |= 0x1; //KEYMOD_SHIFT
|
||
|
}
|
||
|
if (GetKeyState(VK_CONTROL) & 0x8000)
|
||
|
{
|
||
|
grfModifiers |= 0x2; //KEYMOD_CONTROL;
|
||
|
}
|
||
|
if (GetKeyState(VK_MENU) & 0x8000)
|
||
|
{
|
||
|
grfModifiers |= 0x4; //KEYMOD_ALT;
|
||
|
}
|
||
|
_fTabRecieved = TRUE;
|
||
|
hres = pocs->TranslateAccelerator(pMsg, grfModifiers);
|
||
|
_fTabRecieved = FALSE;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return hres;
|
||
|
}
|
||
|
|
||
|
// *** IProvideClassInfo ***
|
||
|
HRESULT CWebViewFolderContents::GetClassInfo(ITypeInfo ** ppTI)
|
||
|
{
|
||
|
if (!_pClassTypeInfo)
|
||
|
Shell32GetTypeInfo(LANGIDFROMLCID(g_lcidLocaleUnicpp), CLSID_WebViewFolderContents, &_pClassTypeInfo);
|
||
|
|
||
|
if (EVAL(_pClassTypeInfo))
|
||
|
{
|
||
|
_pClassTypeInfo->AddRef();
|
||
|
*ppTI = _pClassTypeInfo;
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
*ppTI = NULL;
|
||
|
return E_FAIL;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::GetTypeInfo(UINT itinfo, LCID lcid, ITypeInfo** ppITypeInfo)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
*ppITypeInfo = NULL;
|
||
|
|
||
|
if (0 != itinfo)
|
||
|
return TYPE_E_ELEMENTNOTFOUND;
|
||
|
|
||
|
//Load a type lib if we don't have the information already.
|
||
|
if (NULL == *ppITypeInfo)
|
||
|
{
|
||
|
ITypeInfo * pITIDisp;
|
||
|
|
||
|
hr = Shell32GetTypeInfo(lcid, IID_IShellFolderViewDual, &pITIDisp);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
HRESULT hrT;
|
||
|
HREFTYPE hrefType;
|
||
|
|
||
|
// All our IDispatch implementations are DUAL. GetTypeInfoOfGuid
|
||
|
// returns the ITypeInfo of the IDispatch-part only. We need to
|
||
|
// find the ITypeInfo for the dual interface-part.
|
||
|
|
||
|
hrT = pITIDisp->GetRefTypeOfImplType(0xffffffff, &hrefType);
|
||
|
if (SUCCEEDED(hrT))
|
||
|
hrT = pITIDisp->GetRefTypeInfo(hrefType, ppITypeInfo);
|
||
|
|
||
|
ASSERT(SUCCEEDED(hrT));
|
||
|
pITIDisp->Release();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
HRESULT CWebViewFolderContents::GetIDsOfNames(REFIID /*riid*/, LPOLESTR* rgszNames,
|
||
|
UINT cNames, LCID lcid, DISPID* rgdispid)
|
||
|
{
|
||
|
ITypeInfo* pInfo;
|
||
|
HRESULT hr = GetTypeInfo(0, lcid, &pInfo);
|
||
|
|
||
|
if (pInfo != NULL)
|
||
|
{
|
||
|
hr = pInfo->GetIDsOfNames(rgszNames, cNames, rgdispid);
|
||
|
pInfo->Release();
|
||
|
}
|
||
|
|
||
|
TraceMsg(TF_DEFVIEW, "CWebViewFolderContents::GetIDsOfNames(DISPID=%ls, lcid=%d, cNames=%d) returned hr=%#08lx", *rgszNames, lcid, cNames, hr);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::Invoke(DISPID dispidMember, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS * pdispparams, VARIANT * pvarResult, EXCEPINFO * pexcepinfo, UINT * puArgErr)
|
||
|
{
|
||
|
HRESULT hr = E_FAIL;
|
||
|
DISPPARAMS dispparams = {0};
|
||
|
|
||
|
if (!pdispparams)
|
||
|
pdispparams = &dispparams; // otherwise OLE Fails when passed NULL.
|
||
|
|
||
|
if (dispidMember == DISPID_WINDOWOBJECT)
|
||
|
{
|
||
|
IDispatch * pdisp;
|
||
|
if (SUCCEEDED(get_Script(&pdisp)))
|
||
|
{
|
||
|
hr = pdisp->Invoke(dispidMember, riid, lcid, wFlags, pdispparams, pvarResult, pexcepinfo, puArgErr);
|
||
|
pdisp->Release();
|
||
|
return hr;
|
||
|
}
|
||
|
else
|
||
|
return DISP_E_MEMBERNOTFOUND;
|
||
|
}
|
||
|
|
||
|
// Make sure we are connected up to defviews automation.
|
||
|
hr = _SetupAutomationForwarders();
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = _pdispAuto->Invoke(dispidMember, riid, lcid, wFlags, pdispparams, pvarResult, pexcepinfo, puArgErr);
|
||
|
|
||
|
TraceMsg(TF_DEFVIEW, "CWebViewFolderContents::Invoke(DISPID=%#08lx, lcid=%d, wFlags=%d, pvarResult=%#08lx) returned hr=%#08lx", dispidMember, lcid, wFlags, pvarResult, hr);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
#define DW_MISC_STATUS (OLEMISC_SETCLIENTSITEFIRST | OLEMISC_ACTIVATEWHENVISIBLE | OLEMISC_CANTLINKINSIDE | OLEMISC_INSIDEOUT)
|
||
|
|
||
|
HRESULT CWebViewFolderContents::GetMiscStatus(DWORD dwAspect, DWORD *pdwStatus)
|
||
|
{
|
||
|
*pdwStatus = DW_MISC_STATUS;
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
// IWebViewOCWinMan
|
||
|
|
||
|
HRESULT CWebViewFolderContents::SwapWindow(HWND hwndLV, IWebViewOCWinMan **ppocWinMan)
|
||
|
{
|
||
|
*ppocWinMan = NULL;
|
||
|
|
||
|
_ReleaseWindow();
|
||
|
|
||
|
if (hwndLV)
|
||
|
{
|
||
|
_ShowWindowLV(hwndLV);
|
||
|
|
||
|
_InternalQueryInterface(IID_IWebViewOCWinMan, (void **)ppocWinMan);
|
||
|
}
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
void CWebViewFolderContents::_ShowWindowLV(HWND hwndLV)
|
||
|
{
|
||
|
if (!hwndLV)
|
||
|
return;
|
||
|
_hwndLV = hwndLV;
|
||
|
_hwndLVParent = ::GetParent(_hwndLV);
|
||
|
|
||
|
SHSetParentHwnd(_hwndLV, m_hWnd);
|
||
|
|
||
|
LONG lExStyle = ::GetWindowLong(_hwndLV, GWL_EXSTYLE);
|
||
|
_fClientEdge = lExStyle & WS_EX_CLIENTEDGE ? TRUE : FALSE;
|
||
|
if (_fClientEdge)
|
||
|
{
|
||
|
lExStyle &= ~WS_EX_CLIENTEDGE;
|
||
|
::SetWindowLong(_hwndLV, GWL_EXSTYLE, lExStyle);
|
||
|
}
|
||
|
|
||
|
::SetWindowPos(_hwndLV, 0, 0, 0, m_rcPos.right - m_rcPos.left
|
||
|
, m_rcPos.bottom - m_rcPos.top
|
||
|
, SWP_NOZORDER | SWP_SHOWWINDOW);
|
||
|
|
||
|
// BUGBUG kenwic 060399 #258594 We need to autoautoarrange the sfvext listview after we size it FIXED kenwic 060399
|
||
|
_pdvf2->AutoAutoArrange(0);
|
||
|
}
|
||
|
|
||
|
void CWebViewFolderContents::_ReleaseWindow()
|
||
|
{
|
||
|
if (_hwndLV)
|
||
|
{
|
||
|
if (_fClientEdge)
|
||
|
SetWindowBits(_hwndLV, GWL_EXSTYLE, WS_EX_CLIENTEDGE, WS_EX_CLIENTEDGE);
|
||
|
|
||
|
::ShowWindow(_hwndLV, SW_HIDE); // HIDE IT SO IT doesn't flash before switching.
|
||
|
SHSetParentHwnd(_hwndLV, _hwndLVParent);
|
||
|
::SetWindowPos(_hwndLV, HWND_BOTTOM, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
|
||
|
_pdvf2->ReleaseWindowLV();
|
||
|
_hwndLV = NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
// *** IConnectionPointContainer ***
|
||
|
HRESULT CWebViewFolderContents::EnumConnectionPoints(IEnumConnectionPoints **ppEnum)
|
||
|
{
|
||
|
HRESULT hr = _SetupAutomationForwarders();
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = CreateInstance_IEnumConnectionPoints(ppEnum, 1, SAFECAST(&m_cpEvents, IConnectionPoint*));
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::FindConnectionPoint(REFIID iid, IConnectionPoint **ppCP)
|
||
|
{
|
||
|
*ppCP = NULL;
|
||
|
HRESULT hr = _SetupAutomationForwarders();
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (IsEqualIID(iid, DIID_DShellFolderViewEvents) || IsEqualIID(iid, IID_IDispatch))
|
||
|
{
|
||
|
*ppCP = SAFECAST(&m_cpEvents, IConnectionPoint*);
|
||
|
(*ppCP)->AddRef();
|
||
|
hr = S_OK;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hr = E_NOINTERFACE;
|
||
|
}
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
// *** IConnectionPoint ***
|
||
|
|
||
|
// Our IConnectionPoint is really implemented by defview's CFolder's connection point.
|
||
|
// We just need to keep a count of outstanding references so we can force Unadvise
|
||
|
// calls during "implicit unadvise during shutdown" cases like Trident.
|
||
|
// (Is this really the spec? It doesn't sound very robust...)
|
||
|
|
||
|
HRESULT CWebViewFolderContents::CConnectionPointForwarder::QueryInterface(REFIID riid, void **ppvObj)
|
||
|
{
|
||
|
static const QITAB qit[] = {
|
||
|
QITABENT(CConnectionPointForwarder, IConnectionPoint), // IID_IConnectionPoint
|
||
|
{ 0 }
|
||
|
};
|
||
|
return QISearch(this, qit, riid, ppvObj);
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP_(ULONG) CWebViewFolderContents::CConnectionPointForwarder::AddRef(void)
|
||
|
{
|
||
|
return _punkParent->AddRef();
|
||
|
}
|
||
|
STDMETHODIMP_(ULONG) CWebViewFolderContents::CConnectionPointForwarder::Release(void)
|
||
|
{
|
||
|
return _punkParent->Release();
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::CConnectionPointForwarder::GetConnectionInterface(IID * pIID)
|
||
|
{
|
||
|
HRESULT hr = _pcpAuto->GetConnectionInterface(pIID);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
ASSERT(IsEqualIID(*pIID, DIID_DShellFolderViewEvents));
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::CConnectionPointForwarder::GetConnectionPointContainer(IConnectionPointContainer ** ppCPC)
|
||
|
{
|
||
|
return _punkParent->QueryInterface(IID_IConnectionPointContainer, (void**)ppCPC);
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::CConnectionPointForwarder::Advise(IUnknown * pUnkSink, DWORD * pdwCookie)
|
||
|
{
|
||
|
if (!_dsaCookies)
|
||
|
{
|
||
|
_dsaCookies = DSA_Create(sizeof(*pdwCookie), 4);
|
||
|
if (!_dsaCookies)
|
||
|
{
|
||
|
*pdwCookie = 0;
|
||
|
return E_OUTOFMEMORY;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
HRESULT hr = _pcpAuto->Advise(pUnkSink, pdwCookie);
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (-1 == DSA_AppendItem(_dsaCookies, pdwCookie))
|
||
|
{
|
||
|
_pcpAuto->Unadvise(*pdwCookie);
|
||
|
*pdwCookie = 0;
|
||
|
hr = E_OUTOFMEMORY;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::CConnectionPointForwarder::Unadvise(DWORD dwCookie)
|
||
|
{
|
||
|
int i = 0;
|
||
|
DWORD dw;
|
||
|
|
||
|
if (_dsaCookies)
|
||
|
{
|
||
|
while (DSA_GetItem(_dsaCookies, i++, &dw))
|
||
|
{
|
||
|
if (dw == dwCookie)
|
||
|
{
|
||
|
DSA_DeleteItem(_dsaCookies, --i);
|
||
|
return _pcpAuto->Unadvise(dwCookie);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// BUGBUG: we should probably shassert here
|
||
|
return E_FAIL;
|
||
|
}
|
||
|
|
||
|
void CWebViewFolderContents::_UnadviseAll()
|
||
|
{
|
||
|
if (m_cpEvents._dsaCookies)
|
||
|
{
|
||
|
if (m_cpEvents._pcpAuto)
|
||
|
{
|
||
|
DWORD dw;
|
||
|
for (int i = 0; DSA_GetItem(m_cpEvents._dsaCookies, i, &dw); i++)
|
||
|
{
|
||
|
m_cpEvents._pcpAuto->Unadvise(dw);
|
||
|
}
|
||
|
}
|
||
|
DSA_Destroy(m_cpEvents._dsaCookies);
|
||
|
m_cpEvents._dsaCookies = NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::Close(DWORD dwSaveOption)
|
||
|
{
|
||
|
_UnadviseAll();
|
||
|
HRESULT hr = IOleObjectImpl<CWebViewFolderContents>::Close(dwSaveOption);
|
||
|
_ReleaseAutomationForwarders();
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::_SetupAutomationForwarders(void)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
if (!m_cpEvents._pcpAuto)
|
||
|
{
|
||
|
IShellView *psv;
|
||
|
hr = IUnknown_QueryService(m_spClientSite, SID_DefView, IID_IShellView, (void **)&psv);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
IUnknown *punk;
|
||
|
hr = psv->GetItemObject(SVGIO_BACKGROUND, IID_IDispatch, (void**)&punk);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = punk->QueryInterface(IID_IShellFolderViewDual, (void**)&_pdispAuto);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (SUCCEEDED(MakeSafeForScripting((IUnknown**)&_pdispAuto)))
|
||
|
{
|
||
|
IUnknown_SetSite(_pdispAuto, m_spClientSite);
|
||
|
|
||
|
// Need to get the right interfaces
|
||
|
IConnectionPointContainer* pcpcAuto;
|
||
|
hr = _pdispAuto->QueryInterface(IID_IConnectionPointContainer, (void**)&pcpcAuto);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
hr = pcpcAuto->FindConnectionPoint(IID_IDispatch, &m_cpEvents._pcpAuto);
|
||
|
pcpcAuto->Release();
|
||
|
}
|
||
|
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
IUnknown_SetSite(_pdispAuto, NULL);
|
||
|
ATOMICRELEASE(_pdispAuto);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
punk->Release();
|
||
|
}
|
||
|
psv->Release();
|
||
|
}
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::_ReleaseAutomationForwarders(void)
|
||
|
{
|
||
|
ATOMICRELEASE(m_cpEvents._pcpAuto);
|
||
|
if (_pdispAuto)
|
||
|
IUnknown_SetSite(_pdispAuto, NULL);
|
||
|
|
||
|
ATOMICRELEASE(_pdispAuto);
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
// IShellFolderViewDual
|
||
|
|
||
|
// We will let the folder object get created and have it maintain that we only have one
|
||
|
// application object (with the site) set properly...
|
||
|
|
||
|
HRESULT CWebViewFolderContents::get_Application(IDispatch **ppid)
|
||
|
{
|
||
|
*ppid = NULL;
|
||
|
HRESULT hr = _SetupAutomationForwarders();
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = _pdispAuto->get_Application(ppid);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::get_Parent(IDispatch **ppid)
|
||
|
{
|
||
|
*ppid = NULL;
|
||
|
HRESULT hr = _SetupAutomationForwarders();
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = _pdispAuto->get_Parent(ppid);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::get_Folder(Folder **ppid)
|
||
|
{
|
||
|
*ppid = NULL;
|
||
|
HRESULT hr = _SetupAutomationForwarders();
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = _pdispAuto->get_Folder(ppid);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::SelectedItems(FolderItems **ppid)
|
||
|
{
|
||
|
// We need to talk to the actual window under us
|
||
|
*ppid = NULL;
|
||
|
HRESULT hr = _SetupAutomationForwarders();
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = _pdispAuto->SelectedItems(ppid);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::get_FocusedItem(FolderItem **ppid)
|
||
|
{
|
||
|
*ppid = NULL;
|
||
|
HRESULT hr = _SetupAutomationForwarders();
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = _pdispAuto->get_FocusedItem(ppid);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::SelectItem(VARIANT *pvfi, int dwFlags)
|
||
|
{
|
||
|
HRESULT hr = _SetupAutomationForwarders();
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = _pdispAuto->SelectItem(pvfi, dwFlags);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::PopupItemMenu(FolderItem *pfi, VARIANT vx, VARIANT vy, BSTR * pbs)
|
||
|
{
|
||
|
HRESULT hr = _SetupAutomationForwarders();
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = _pdispAuto->PopupItemMenu(pfi, vx, vy, pbs);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::get_Script(IDispatch **ppid)
|
||
|
{
|
||
|
*ppid = NULL;
|
||
|
HRESULT hr = _SetupAutomationForwarders();
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = _pdispAuto->get_Script(ppid);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
HRESULT CWebViewFolderContents::get_ViewOptions(long *plSetting)
|
||
|
{
|
||
|
HRESULT hr = _SetupAutomationForwarders();
|
||
|
if (SUCCEEDED(hr))
|
||
|
hr = _pdispAuto->get_ViewOptions(plSetting);
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
STDAPI CWebViewFolderContents_CreateInstance(IUnknown *punkOuter, REFIID riid, void **ppvOut)
|
||
|
{
|
||
|
return CComCreator< CComPolyObject< CWebViewFolderContents > >::CreateInstance((void *) punkOuter, riid, ppvOut);
|
||
|
}
|
||
|
|