2020-09-30 16:53:55 +02:00

3011 lines
80 KiB
C++

#include "precomp.h"
#define __MARS_INLINE_FAST_IS_EQUAL_GUID
#define LIGHTWEIGHT_AUTOPERSIST
#define BLOCK_PANEL_RENAVIGATES
#include "mcinc.h"
#include <evilguid.h>
#include "marswin.h"
#include "pandef.h"
#include "panel.h"
#include "place.h"
#include "htiface.h"
#include "mimeinfo.h"
#include <exdispid.h>
#include "dllload.h"
#include <perhist.h>
#include <mshtmcid.h>
#include <strsafe.h>
#include <msiehost.h>
// We include this cpp file because the stuff
// in pandef needs to be included in another project
// namely, parser\comptree
#include "pandef.cpp"
// CLSID strings passed to ATL to create control
static WCHAR wszCLSID_HTMLDocument[] = L"{25336920-03F9-11CF-8FD0-00AA00686F13}";
static WCHAR wszCLSID_WebBrowser[] = L"{8856F961-340A-11D0-A96B-00C04FD705A2}";
static WCHAR wszCLSID_HTADocument[] = L"{3050f5c8-98b5-11cf-bb82-00aa00bdce0b}";
const GUID CLSID_HTADoc = { 0x3050f5c8, 0x98b5, 0x11cf, { 0xbb, 0x82, 0x00, 0xaa, 0x00, 0xbd, 0xce, 0x0b } };
// CLASS_CMarsPanel = {EE0462C2-5CD3-11d3-97FA-00C04F45D0B3}
const GUID CLASS_CMarsPanel = { 0xee0462c2, 0x5cd3, 0x11d3, { 0x97, 0xfa, 0x0, 0xc0, 0x4f, 0x45, 0xd0, 0xb3 } };
CMarsPanel::CMarsPanel(CPanelCollection *pParent, CMarsWindow *pMarsWindow) :
m_MarsExternal(this, pMarsWindow),
m_BrowserEvents(this)
{
m_spPanelCollection = pParent;
m_spMarsDocument = m_spPanelCollection->Document();
m_lReadyState = READYSTATE_COMPLETE;
}
CMarsPanel::~CMarsPanel()
{
}
HRESULT CMarsPanel::Passivate()
{
return CMarsComObject::Passivate();
}
HRESULT CMarsPanel::DoPassivate()
{
m_spMarsDocument->MarsWindow()->NotifyHost(MARSHOST_ON_PANEL_PASSIVATE, SAFECAST(this, IMarsPanel *), 0);
m_spPanelCollection->SetActivePanel(this, FALSE);
// First we unload the document so that script can do work in its "onunload"
// handler, before we become passive
if(IsWebBrowser())
{
CComPtr<IUnknown> spUnk;
if (SUCCEEDED(m_Content.QueryControl(IID_IUnknown, (void **)&spUnk)))
{
m_BrowserEvents.Connect(spUnk, FALSE);
}
}
else if(IsCustomControl())
{
}
else
{
DisconnectCompletionAdviser();
}
m_Content.DestroyWindow();
////////////////////////////////////////
m_spMarsDocument->MarsWindow()->ReleaseOwnedObjects(SAFECAST(this, IDispatch *));
m_BrowserEvents.Passivate();
m_MarsExternal.Passivate();
m_spBrowserService.Release();
m_spPanelCollection.Release();
m_spMarsDocument.Release();
return S_OK;
}
// IUnknown
IMPLEMENT_ADDREF_RELEASE(CMarsPanel);
STDMETHODIMP CMarsPanel::QueryInterface(REFIID iid, void ** ppvObject)
{
ATLASSERT(ppvObject);
if ((iid == IID_IUnknown) ||
(iid == IID_IDispatch) ||
(iid == IID_IMarsPanel))
{
*ppvObject = SAFECAST(this, IMarsPanel *);
}
else if (iid == IID_IHlinkFrame)
{
*ppvObject = SAFECAST(this, IHlinkFrame *);
}
else if (iid == IID_IServiceProvider)
{
*ppvObject = SAFECAST(this, IServiceProvider *);
}
else if (iid == IID_IInternetSecurityManager)
{
*ppvObject = SAFECAST(this, IInternetSecurityManager *);
}
else if ((iid == IID_IOleInPlaceSite) || (iid == IID_IOleWindow))
{
*ppvObject = SAFECAST(this, IOleInPlaceSite *);
}
else if (iid == IID_IOleControlSite)
{
*ppvObject = SAFECAST(this, IOleControlSite *);
}
else if (iid == IID_IPropertyNotifySink)
{
*ppvObject = SAFECAST(this, IPropertyNotifySink *);
}
else if (iid == IID_IProfferService)
{
*ppvObject = SAFECAST(this, IProfferService *);
}
else if (iid == IID_IOleInPlaceUIWindow)
{
*ppvObject = SAFECAST(this, IOleInPlaceUIWindow *);
}
else if (iid == CLASS_CMarsPanel)
{
*ppvObject = SAFECAST(this, CMarsPanel *);
}
else
{
*ppvObject = NULL;
return E_NOINTERFACE;
}
AddRef();
return S_OK;
}
HRESULT CMarsPanel::DoEnableModeless(BOOL fEnable)
{
CComPtr<IOleInPlaceActiveObject> spOleInPlaceActiveObject;
HRESULT hr = m_Content.QueryControl(&spOleInPlaceActiveObject);
if (SUCCEEDED(hr))
{
spOleInPlaceActiveObject->EnableModeless(fEnable);
}
return hr;
}
//==================================================================
// Automation Object Model
//==================================================================
//------------------------------------------------------------------------------
// IMarsPanel::get_name
//
HRESULT CMarsPanel::get_name( BSTR *pbstrName )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidWritePtr( pbstrName ))
{
*pbstrName = SysAllocString( GetName() );
hr = (NULL != *pbstrName) ? S_OK : E_OUTOFMEMORY;
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::get_content
//
HRESULT CMarsPanel::get_content( IDispatch* *ppVal )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidWritePtr( ppVal ))
{
*ppVal = NULL;
if(VerifyNotPassive( &hr ))
{
CreateControl();
hr = m_Content.QueryControl( ppVal );
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::get_visible
//
HRESULT CMarsPanel::get_visible( VARIANT_BOOL *pbVisible )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidWritePtr( pbVisible ))
{
if(VerifyNotPassive( &hr ))
{
*pbVisible = VARIANT_BOOLIFY( m_fVisible );
hr = S_OK;
}
else
{
*pbVisible = VARIANT_FALSE;
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::put_visible
//
HRESULT CMarsPanel::put_visible( VARIANT_BOOL bVisible )
{
ATLASSERT(IsValidVariantBoolVal( bVisible ));
HRESULT hr = S_OK;
if(VerifyNotPassive( &hr ))
{
if(!!m_fVisible != !!bVisible)
{
m_fVisible = (bVisible != VARIANT_FALSE);
if(!m_fVisible)
{
m_spPanelCollection->SetActivePanel( this, FALSE );
}
// If the theme has changed while the panel was invisible,
// now is a good time to automatically update it
if(m_fVisible && IsContentInvalid())
{
refresh();
}
OnLayoutChange();
}
else
{
hr = S_FALSE;
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::get_startUrl
//
HRESULT CMarsPanel::get_startUrl( BSTR *pVal )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidWritePtr( pVal ))
{
*pVal = m_bstrStartUrl.Copy();
hr = S_OK;
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::put_startUrl
//
HRESULT CMarsPanel::put_startUrl( BSTR newVal )
{
HRESULT hr = S_OK;
m_bstrStartUrl = newVal;
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::get_height
//
HRESULT CMarsPanel::get_height( long *plHeight )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidWritePtr( plHeight ))
{
if(VerifyNotPassive( &hr ))
{
hr = S_OK;
if(m_Content.IsWindowVisible())
{
// Use actual position
RECT rc;
m_Content.GetWindowRect( &rc );
*plHeight = rc.bottom - rc.top;
}
else
{
// Panel's hidden. Return requested position.
*plHeight = m_lHeight;
}
}
else
{
*plHeight = 0;
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::put_height
//
HRESULT CMarsPanel::put_height( long lHeight )
{
ATLASSERT(lHeight >= 0);
HRESULT hr = S_OK;
if(VerifyNotPassive( &hr ))
{
if(lHeight >= 0)
{
if(m_lMinHeight >= 0 && lHeight < m_lMinHeight) lHeight = m_lMinHeight;
if(m_lMaxHeight >= 0 && lHeight > m_lMaxHeight) lHeight = m_lMaxHeight;
if(m_lHeight != lHeight)
{
m_lHeight = lHeight;
OnLayoutChange();
}
}
else
{
hr = S_FALSE;
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::get_width
//
HRESULT CMarsPanel::get_width( long *plWidth )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidWritePtr( plWidth ))
{
if(VerifyNotPassive( &hr ))
{
if(m_Content.IsWindowVisible())
{
// Use actual position
RECT rc;
m_Content.GetWindowRect( &rc );
*plWidth = rc.right - rc.left;
}
else
{
// Panel's hidden. Return requested position.
*plWidth = m_lWidth;
}
hr = S_OK;
}
else
{
*plWidth = 0;
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::put_width
//
HRESULT CMarsPanel::put_width( long lWidth )
{
ATLASSERT(lWidth >= 0);
HRESULT hr = S_OK;
if(VerifyNotPassive( &hr ))
{
if(lWidth >= 0)
{
if(m_lMinWidth >= 0 && lWidth < m_lMinWidth) lWidth = m_lMinWidth;
if(m_lMaxWidth >= 0 && lWidth > m_lMaxWidth) lWidth = m_lMaxWidth;
if(m_lWidth != lWidth)
{
m_lWidth = lWidth;
OnLayoutChange();
}
}
else
{
hr = S_FALSE;
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::get_x
//
HRESULT CMarsPanel::get_x( long *plX )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidWritePtr( plX ))
{
if(VerifyNotPassive( &hr ))
{
hr = S_OK;
if(m_Content.IsWindowVisible())
{
// Use actual position
RECT rc;
GetMyClientRectInParentCoords( &rc );
*plX = rc.left;
}
else
{
// Panel's hidden. Return requested position.
*plX = m_lX;
}
}
else
{
*plX = 0;
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::put_x
//
HRESULT CMarsPanel::put_x( long lX )
{
HRESULT hr = S_OK;
if(VerifyNotPassive( &hr ))
{
if(m_lX != lX)
{
m_lX = lX;
if(IsPopup())
{
OnLayoutChange();
}
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::get_y
//
HRESULT CMarsPanel::get_y( long *plY )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidWritePtr( plY ))
{
if(VerifyNotPassive( &hr ))
{
hr = S_OK;
if(m_Content.IsWindowVisible())
{
// Use actual position
RECT rc;
GetMyClientRectInParentCoords( &rc );
*plY = rc.top;
}
else
{
*plY = m_lY;
}
}
else
{
*plY = 0;
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::put_y
//
HRESULT CMarsPanel::put_y( long lY )
{
HRESULT hr = S_OK;
if(VerifyNotPassive( &hr ))
{
if(m_lY != lY)
{
m_lY = lY;
if(IsPopup())
{
OnLayoutChange();
}
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::get_position
//
HRESULT CMarsPanel::get_position( VARIANT *pvarPosition )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidWritePtr( pvarPosition ))
{
if(VerifyNotPassive( &hr ))
{
VariantClear( pvarPosition );
hr = S_FALSE;
if(m_Position <= c_iPositionMapSize)
{
pvarPosition->bstrVal = SysAllocString( s_PositionMap[m_Position].pwszName );
if(pvarPosition->bstrVal)
{
pvarPosition->vt = VT_BSTR;
hr = S_OK;
}
}
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::put_position
//
HRESULT CMarsPanel::put_position( VARIANT varPosition )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidVariantBstr( varPosition ))
{
if(VerifyNotPassive( &hr ))
{
hr = SCRIPT_ERROR;
PANEL_POSITION position;
if(SUCCEEDED(StringToPanelPosition( varPosition.bstrVal, &position )))
{
hr = S_OK;
if(m_Position != position)
{
m_Position = s_PositionMap[position].Position;
if(m_Position == PANEL_POPUP)
{
m_Content.SetWindowPos( HWND_TOP, 0, 0, 0, 0, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOSIZE );
}
OnLayoutChange();
}
}
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::get_autoSize
//
HRESULT CMarsPanel::get_autoSize( VARIANT_BOOL *pbAutoSize )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidWritePtr( pbAutoSize ))
{
*pbAutoSize = VARIANT_BOOLIFY(IsAutoSizing());
if(VerifyNotPassive( &hr ))
{
hr = S_OK;
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::put_autoSize
//
HRESULT CMarsPanel::put_autoSize( VARIANT_BOOL bAutoSize )
{
HRESULT hr = S_OK;
ATLASSERT(IsValidVariantBoolVal( bAutoSize ));
if(VerifyNotPassive( &hr ))
{
if(bAutoSize)
{
if(!IsAutoSizing())
{
m_dwFlags |= PANEL_FLAG_AUTOSIZE;
OnLayoutChange();
}
}
else
{
if(IsAutoSizing())
{
m_dwFlags &= ~PANEL_FLAG_AUTOSIZE;
OnLayoutChange();
}
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::get_contentInvalid
//
HRESULT CMarsPanel::get_contentInvalid( VARIANT_BOOL *pbInvalid )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidWritePtr( pbInvalid ))
{
*pbInvalid = VARIANT_BOOLIFY(m_fContentInvalid);
hr = S_OK;
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::put_contentInvalid
//
HRESULT CMarsPanel::put_contentInvalid( VARIANT_BOOL bInvalid )
{
ATLASSERT(IsValidVariantBoolVal( bInvalid ));
ATLASSERT(!IsPassive());
m_fContentInvalid = BOOLIFY( bInvalid );
return S_OK;
}
//------------------------------------------------------------------------------
// IMarsPanel::get_layoutIndex
//
HRESULT CMarsPanel::get_layoutIndex( long *plIndex )
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidWritePtr( plIndex ))
{
if(VerifyNotPassive( &hr ))
{
hr = m_spPanelCollection->FindPanelIndex( this, plIndex );
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::moveto
//
HRESULT CMarsPanel::moveto( VARIANT vlX, VARIANT vlY, VARIANT vlWidth, VARIANT vlHeight )
{
HRESULT hr = S_OK;
ATLASSERT(vlX.vt == VT_NULL || vlX.vt == VT_I4);
ATLASSERT(vlY.vt == VT_NULL || vlY.vt == VT_I4);
ATLASSERT(vlWidth.vt == VT_NULL || vlWidth.vt == VT_I4);
ATLASSERT(vlHeight.vt == VT_NULL || vlHeight.vt == VT_I4);
if(VerifyNotPassive( &hr ))
{
long lX = VariantToI4( vlX );
long lY = VariantToI4( vlY );
long lWidth = VariantToI4( vlWidth );
long lHeight = VariantToI4( vlHeight );
if(( lX != m_lX ) ||
( lY != m_lY ) ||
(lWidth >= 0 && lWidth != m_lWidth ) ||
(lHeight >= 0 && lHeight != m_lHeight) )
{
m_lX = lX;
m_lY = lY;
if(lWidth >= 0) m_lWidth = lWidth;
if(lHeight >= 0) m_lHeight = lHeight;
OnLayoutChange();
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::restrictWidth
//
HRESULT CMarsPanel::restrictWidth( VARIANT vlMin, VARIANT vlMax, VARIANT vbstrMarker )
{
HRESULT hr = S_OK;
ATLASSERT(vlMin.vt == VT_EMPTY || vlMin.vt == VT_NULL || vlMin.vt == VT_BSTR || vlMin.vt == VT_I4);
ATLASSERT(vlMax.vt == VT_EMPTY || vlMax.vt == VT_NULL || vlMax.vt == VT_BSTR || vlMin.vt == VT_I4);
ATLASSERT((vbstrMarker.vt==VT_NULL) || (vbstrMarker.vt == VT_EMPTY));
if(VerifyNotPassive( &hr ))
{
m_lMaxWidth = m_lMinWidth = -1;
if(vlMin.vt != VT_NULL &&
vlMin.vt != VT_EMPTY )
{
CComVariant vlMin2; vlMin2.ChangeType( VT_I4, &vlMin );
m_lMinWidth = VariantToI4( vlMin2 );
}
if(vlMax.vt != VT_NULL &&
vlMax.vt != VT_EMPTY )
{
CComVariant vlMax2; vlMax2.ChangeType( VT_I4, &vlMax );
m_lMaxWidth = VariantToI4( vlMax2 );
}
if(m_lMaxWidth >= 0 && m_lMaxWidth < m_lMinWidth)
{
m_lMaxWidth = m_lMinWidth;
}
OnLayoutChange();
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::restrictHeight
//
HRESULT CMarsPanel::restrictHeight( VARIANT vlMin, VARIANT vlMax, VARIANT vbstrMarker )
{
HRESULT hr = S_OK;
ATLASSERT(vlMin.vt == VT_EMPTY || vlMin.vt == VT_NULL || vlMin.vt == VT_BSTR || vlMin.vt == VT_I4);
ATLASSERT(vlMax.vt == VT_EMPTY || vlMax.vt == VT_NULL || vlMax.vt == VT_BSTR || vlMin.vt == VT_I4);
ATLASSERT((vbstrMarker.vt==VT_NULL) || (vbstrMarker.vt == VT_EMPTY));
if(VerifyNotPassive( &hr ))
{
m_lMaxHeight = m_lMinHeight = -1;
if(vlMin.vt != VT_NULL &&
vlMin.vt != VT_EMPTY )
{
CComVariant vlMin2; vlMin2.ChangeType( VT_I4, &vlMin );
m_lMinHeight = VariantToI4( vlMin2 );
}
if(vlMax.vt != VT_NULL &&
vlMax.vt != VT_EMPTY )
{
CComVariant vlMax2; vlMax2.ChangeType( VT_I4, &vlMax );
m_lMaxHeight = VariantToI4( vlMax2 );
}
if(m_lMaxHeight >= 0 && m_lMaxHeight < m_lMinHeight) m_lMaxHeight = m_lMinHeight;
OnLayoutChange();
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::canResize
//
HRESULT CMarsPanel::canResize( long lDX, long lDY, VARIANT_BOOL *pVal )
{
long lWidthOld = m_lWidth;
long lHeightOld = m_lHeight;
RECT rcClient;
m_lWidth += lDX;
m_lHeight += lDY;
(void)m_spMarsDocument->Window()->GetClientRect( &rcClient );
if(m_spMarsDocument->MarsWindow()->CanLayout( rcClient ))
{
if(pVal) *pVal = VARIANT_TRUE;
}
else
{
if(pVal) *pVal = VARIANT_FALSE;
}
m_lWidth = lWidthOld;
m_lHeight = lHeightOld;
return S_OK;
}
//------------------------------------------------------------------------------
// IMarsPanel::activate
//
HRESULT CMarsPanel::activate()
{
m_Content.SendMessage(WM_SETFOCUS, 0, 0);
return S_OK;
}
//------------------------------------------------------------------------------
// IMarsPanel::insertBefore
//
HRESULT CMarsPanel::insertBefore(VARIANT varInsertBefore)
{
ATLASSERT(varInsertBefore.vt == VT_I4 || // index to insert before
varInsertBefore.vt == VT_BSTR || // panel name to insert before
varInsertBefore.vt == VT_DISPATCH); // panel object to insert before
HRESULT hr = E_INVALIDARG;
if (API_IsValidVariant(varInsertBefore))
{
if (VerifyNotPassive(&hr))
{
hr = E_FAIL;
long lIndex = -668;
switch(varInsertBefore.vt)
{
case VT_I4:
lIndex = varInsertBefore.lVal;
hr = S_OK;
break;
case VT_DISPATCH:
{
CComClassPtr<CMarsPanel> spPanel;
varInsertBefore.pdispVal->QueryInterface(CLASS_CMarsPanel, (void **)&spPanel);
if (spPanel)
{
hr = m_spPanelCollection->FindPanelIndex(spPanel, &lIndex);
}
break;
}
case VT_BSTR:
hr = m_spPanelCollection->FindPanelIndex(
m_spPanelCollection->FindPanel(varInsertBefore.bstrVal),
&lIndex);
break;
default:
ATLASSERT(0);
}
if (SUCCEEDED(hr))
{
// Successfully got an index to insert before
ATLASSERT(lIndex != -668);
// Get our current index
long lCurrentIndex;
if (SUCCEEDED(m_spPanelCollection->FindPanelIndex(this, &lCurrentIndex)))
{
ATLASSERT(lCurrentIndex != -1);
if (lIndex != lCurrentIndex)
{
m_spPanelCollection->InsertPanelFromTo(lCurrentIndex, lIndex);
OnLayoutChange();
}
}
else
{
ATLASSERT(!"FindPanelIndex failed!");
}
}
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::execMshtml
//
// execMshtml allows script to send a command directly to Trident. This is different from
// IWebBrowser2::ExecWB because ExecWB passes a NULL command group, which prevents Trident
// from responding to commands like IDM_FIND. Usually, we could accomplish this by calling
// the OM method execCommand, but Trident doesn't respond to execCommand("Find")
//
HRESULT CMarsPanel::execMshtml(DWORD nCmdID, DWORD nCmdExecOpt,
VARIANTARG *pvaIn, VARIANTARG *pvaOut)
{
HRESULT hr = E_INVALIDARG;
if (VerifyNotPassive(&hr))
{
CComPtr<IOleCommandTarget> spCmdTarget;
hr = m_Content.QueryControl(IID_IOleCommandTarget, (void **) &spCmdTarget);
if (SUCCEEDED(hr))
{
hr = spCmdTarget->Exec(&CGID_MSHTML, nCmdID, nCmdExecOpt, pvaIn, pvaOut);
}
hr = S_OK;
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::get_isCurrentlyVisible
//
HRESULT CMarsPanel::get_isCurrentlyVisible(VARIANT_BOOL *pbIsVisible)
{
HRESULT hr = E_INVALIDARG;
if (API_IsValidWritePtr(pbIsVisible))
{
if (VerifyNotPassive(&hr))
{
*pbIsVisible = VARIANT_BOOLIFY(m_Content.IsWindowVisible());
hr = S_OK;
}
}
return hr;
}
//------------------------------------------------------------------------------
// IServiceProvider::QueryService
//
// First, we try to handle guidService, then we go to IProfferServiceImpl,
// then we go to the parent window.
//
HRESULT CMarsPanel::QueryService(REFGUID guidService, REFIID riid, void **ppv)
{
HRESULT hr = E_FAIL;
*ppv = NULL;
if (!IsPassive())
{
if (guidService == IID_IInternetProtocol)
{
// Yeah, we don't support this service. Stop asking us for it!
}
else if ((guidService == SID_SHlinkFrame) ||
(guidService == SID_SProfferService) ||
(guidService == SID_SMarsPanel))
{
hr = QueryInterface(riid, ppv);
}
else if (guidService == SID_SInternetSecurityManager)
{
if (IsTrusted())
{
hr = QueryInterface(riid, ppv);
}
}
if (FAILED(hr))
{
hr = IProfferServiceImpl::QueryService(guidService, riid, ppv);
}
}
if (FAILED(hr))
{
hr = m_spMarsDocument->MarsWindow()->QueryService(guidService, riid, ppv);
}
return hr;
}
//------------------------------------------------------------------------------
// IHlinkFrame::Navigate
//
HRESULT CMarsPanel::Navigate(DWORD grfHLNF, LPBC pbc,
IBindStatusCallback *pibsc, IHlink *pihlNavigate)
{
// BUGBUG: Call ReleasedOwnedObjects for this panel
HRESULT hr = E_FAIL;
CComPtr<IMoniker> spMk;
if (VerifyNotPassive())
{
pihlNavigate->GetMonikerReference(grfHLNF, &spMk, NULL);
if (spMk)
{
hr = NavigateMk(spMk);
}
}
return hr;
}
//------------------------------------------------------------------------------
HRESULT CMarsPanel::Create( MarsAppDef_Panel* pLayout)
{
ATLASSERT(!IsPassive());
m_bstrName = pLayout->szName;
m_Position = pLayout->Position;
m_lWidth = pLayout->lWidth;
m_lHeight = pLayout->lHeight;
m_lX = pLayout->lX;
m_lY = pLayout->lY;
m_dwFlags = pLayout->dwFlags;
m_lMinWidth = pLayout->lWidthMin;
m_lMinHeight = pLayout->lHeightMin;
m_lMaxWidth = pLayout->lWidthMax;
m_lMaxHeight = pLayout->lHeightMax;
m_fVisible = BOOLIFY(m_dwFlags & PANEL_FLAG_VISIBLE);
m_dwCookie = 0;
if(!IsAutoSizing())
{
switch(m_Position)
{
case PANEL_BOTTOM:
case PANEL_TOP:
if(m_lMinHeight < 0 && m_lMaxHeight < 0)
{
m_lMinHeight = m_lHeight;
m_lMaxHeight = m_lHeight;
}
break;
case PANEL_LEFT:
case PANEL_RIGHT:
if(m_lMinWidth < 0 && m_lMaxWidth < 0)
{
m_lMinWidth = m_lWidth;
m_lMaxWidth = m_lWidth;
}
break;
}
}
{
DWORD dwStyle = WS_CHILD | WS_CLIPSIBLINGS;
RECT rcCreate = { 0, 0, 0, 0 };
m_Content.Create( m_spMarsDocument->Window()->m_hWnd, &rcCreate, _T(""), dwStyle, 0 );
}
{
CComPtr<IObjectWithSite> spObj;
m_Content.QueryHost(IID_IObjectWithSite, (void **)&spObj);
if (spObj)
{
spObj->SetSite(SAFECAST(this, IMarsPanel *)); // To connect our QueryService
}
}
m_spMarsDocument->MarsWindow()->NotifyHost(MARSHOST_ON_PANEL_INIT, SAFECAST(this, IMarsPanel *), 0);
if(pLayout->szUrl[0])
{
m_bstrStartUrl = pLayout->szUrl;
}
if(!(m_dwFlags & PANEL_FLAG_ONDEMAND))
{
CreateControl();
}
if(m_dwFlags & PANEL_FLAG_VISIBLE)
{
OnLayoutChange();
}
return S_OK;
}
//------------------------------------------------------------------------------
DELAY_LOAD_NAME_HRESULT(g_hinstBorg, mshtml, BorgDllGetClassObject, DllGetClassObject,
(REFCLSID rclsid, REFIID riid, LPVOID* ppv),
(rclsid, riid, ppv));
HRESULT CMarsPanel::CreateControlObject()
{
ATLASSERT(!m_fControlCreated);
CComPtr<IAxWinHostWindow> spHost;
HRESULT hr = m_Content.QueryHost(&spHost);
if (SUCCEEDED(hr))
{
if(IsWebBrowser())
{
hr = spHost->CreateControl(wszCLSID_WebBrowser, m_Content, 0);
}
else if(IsCustomControl())
{
hr = spHost->CreateControl(m_bstrStartUrl, m_Content, 0);
}
else
{
if (IsTrusted())
{
CComPtr<IClassFactory> spCf;
hr = BorgDllGetClassObject(CLSID_HTADoc, IID_IClassFactory, (void **)&spCf);
if (SUCCEEDED(hr))
{
CComPtr<IUnknown> spUnk;
hr = spCf->CreateInstance(NULL, IID_IUnknown, (void **)&spUnk);
if (SUCCEEDED(hr))
{
hr = spHost->AttachControl(spUnk, m_Content);
}
}
}
else
{
hr = spHost->CreateControl(wszCLSID_HTMLDocument, m_Content, 0);
}
}
}
return hr;
}
//------------------------------------------------------------------------------
// Internal function called when panel is first made visible
HRESULT CMarsPanel::CreateControl()
{
HRESULT hr = S_FALSE;
if(!m_fControlCreated)
{
// Create WebBrowser or Trident and navigate to default url
if (SUCCEEDED(CreateControlObject()))
{
m_fControlCreated = TRUE;
if(IsWebBrowser())
{
CComPtr<IUnknown> spUnk;
if (SUCCEEDED(m_Content.QueryControl(IID_IUnknown, (void **)&spUnk)))
{
m_BrowserEvents.Connect(spUnk, TRUE);
// get the browserservice that we will intercept to update the travel log
// only if this is a WebBrowser and not a Popup
if (!m_spBrowserService)
{
HRESULT hrQS = IUnknown_QueryService(spUnk, SID_STopFrameBrowser,
IID_IBrowserService,
(void **)&m_spBrowserService);
if (FAILED(hrQS))
{
ATLASSERT(!m_spBrowserService);
}
}
CComPtr<IOleCommandTarget> spCommandTarget;
if (SUCCEEDED(IUnknown_QueryService(spUnk, SID_STopFrameBrowser, IID_IOleCommandTarget, (void **)&spCommandTarget)))
{
VARIANT var;
var.vt = VT_BOOL;
var.boolVal = TRUE;
spCommandTarget->Exec(&CGID_InternetExplorer,
IECMDID_SET_INVOKE_DEFAULT_BROWSER_ON_NEW_WINDOW,
0,
&var,
NULL);
}
}
}
else if(IsCustomControl())
{
}
else
{
ConnectCompletionAdviser();
}
if(IsTrusted())
{
m_Content.SetExternalDispatch(&m_MarsExternal);
}
if (m_bstrStartUrl)
{
ATLASSERT(m_bstrStartUrl[0]);
NavigateURL(m_bstrStartUrl, FALSE);
m_bstrStartUrl.Empty();
}
m_spMarsDocument->MarsWindow()->NotifyHost(MARSHOST_ON_PANEL_CONTROL_CREATE, SAFECAST(this, IMarsPanel *), 0);
hr = S_OK;
}
else
{
hr = E_FAIL;
}
}
return hr;
}
//------------------------------------------------------------------------------
// This returns void because it calls a script method that might return
// S_FALSE with no URL, so we should force callers to check the BSTR
//
// rbstrUrl = document.location.href
//
void CMarsPanel::GetUrl(CComBSTR& rbstrUrl)
{
CComPtr<IHTMLDocument2> spDoc2;
GetDoc2FromAxWindow(&m_Content, &spDoc2);
if (spDoc2)
{
CComPtr<IHTMLLocation> spLoc;
spDoc2->get_location(&spLoc);
if (spLoc)
{
spLoc->get_href(&rbstrUrl);
}
}
}
//------------------------------------------------------------------------------
HRESULT CMarsPanel::NavigateMk(IMoniker *pmk)
{
HRESULT hr = E_FAIL;
if (VerifyNotPassive())
{
CComPtr<IPersistMoniker> spPersistMk;
if (SUCCEEDED(m_Content.QueryControl(IID_IPersistMoniker, (void **)&spPersistMk)))
{
m_spMarsDocument->MarsWindow()->ReleaseOwnedObjects((IDispatch *)this);
hr = spPersistMk->Load(FALSE, pmk, NULL, 0);
}
else
{
// NavigateMk: QueryControl failed -- the most likely reason for this is that
// you don't have CLSID_HTADocument registered -- upgrade your IE bits
ATLASSERT(FALSE);
// Don't just hang the UI
m_lReadyState = READYSTATE_COMPLETE;
}
}
return hr;
}
//------------------------------------------------------------------------------
HRESULT CMarsPanel::NavigateURL(LPCWSTR lpszURL, BOOL fForceLoad)
{
HRESULT hr = E_FAIL;
if (VerifyNotPassive())
{
CreateControl();
BOOL fIgnoreNavigate = FALSE;
#ifdef BLOCK_PANEL_RENAVIGATES
if (!fForceLoad && IsTrusted() && !IsContentInvalid())
{
CComBSTR sbstrCurrentUrl;
GetUrl(sbstrCurrentUrl);
if (sbstrCurrentUrl && (0 == StrCmpIW(sbstrCurrentUrl, lpszURL)))
{
fIgnoreNavigate = TRUE;
hr = S_FALSE;
}
}
#endif
if(!fIgnoreNavigate)
{
if(IsWebBrowser())
{
CComPtr<IWebBrowser2> spWebBrowser2;
if (SUCCEEDED(m_Content.QueryControl(IID_IWebBrowser2, (void **)&spWebBrowser2)))
{
CComVariant varEmpty;
CComVariant varURL(lpszURL);
m_spMarsDocument->MarsWindow()->ReleaseOwnedObjects((IDispatch *)this);
hr = spWebBrowser2->Navigate2(&varURL, &varEmpty, &varEmpty, &varEmpty, &varEmpty);
}
}
else if(IsCustomControl())
{
}
else
{
CComPtr<IMoniker> spMkUrl;
if (SUCCEEDED(CreateURLMoniker(NULL, lpszURL, &spMkUrl)) && spMkUrl)
{
hr = NavigateMk(spMkUrl);
}
}
}
if (SUCCEEDED(hr))
{
m_fContentInvalid = FALSE;
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::navigate
//
HRESULT CMarsPanel::navigate(VARIANT varTarget, VARIANT varForceLoad)
{
HRESULT hr = S_FALSE;
if (API_IsValidVariant(varTarget) && VerifyNotPassive(&hr))
{
CComBSTR strPath;
if (SUCCEEDED(MarsVariantToPath(varTarget, strPath)))
{
if (!PathIsURLW(strPath) && PathIsURLFileW(strPath))
{
// handle navigate to .url shortcut
CComPtr<IDispatch> spDisp;
CreateControl();
if (SUCCEEDED(m_Content.QueryControl(&spDisp)))
{
if (SUCCEEDED(MarsNavigateShortcut(spDisp, strPath)))
{
m_spMarsDocument->MarsWindow()->ReleaseOwnedObjects((IDispatch *)this);
hr = S_OK;
}
}
}
else
{
// handle navigate to URL
BOOL fForceLoad;
if (varForceLoad.vt == VT_BOOL)
{
// the optional param was specified
fForceLoad = varForceLoad.boolVal;
}
else
{
fForceLoad = FALSE;
}
if (SUCCEEDED(NavigateURL(strPath, fForceLoad)))
{
hr = S_OK;
}
}
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanel::refresh
//
HRESULT CMarsPanel::refresh(void)
{
m_fInRefresh = TRUE;
execMshtml(IDM_REFRESH, OLECMDEXECOPT_DODEFAULT, NULL, NULL);
m_fInRefresh = FALSE;
m_fContentInvalid = FALSE;
return S_OK;
}
//------------------------------------------------------------------------------
void CMarsPanel::OnLayoutChange()
{
ATLASSERT(!IsPassive());
if (!m_fVisible && !m_Content.IsWindowVisible())
{
// We're not even visible. Nothing to do.
return;
}
// Create content if it's not already created
CreateControl();
// Redraw entire Mars window
m_spPanelCollection->Layout();
}
//------------------------------------------------------------------------------
void CMarsPanel::MakeVisible(VARIANT_BOOL bVisible, VARIANT_BOOL bForce)
{
ATLASSERT(!IsPassive());
BOOL fVisible = m_Content.IsWindowVisible();
if (bForce || (!!fVisible != !!bVisible))
{
m_Content.ShowWindow((bVisible) ? SW_SHOW : SW_HIDE);
}
}
//------------------------------------------------------------------------------
void CMarsPanel::OnWindowPosChanging(WINDOWPOS *pWindowPos)
{
if (!IsPassive() && !m_spPanelCollection->IsLayoutLocked())
{
if (pWindowPos->x < 0)
{
pWindowPos->x = 0;
}
if (pWindowPos->y < 0)
{
pWindowPos->y = 0;
}
}
}
//------------------------------------------------------------------------------
inline void DimChange(long& lMember, int iVal, BOOL& bChanged)
{
if (lMember != iVal)
{
lMember = iVal;
bChanged = TRUE;
}
}
//------------------------------------------------------------------------------
void CMarsPanel::OnWindowPosChanged(WINDOWPOS *pWindowPos)
{
if(!IsPassive() && !m_spPanelCollection->IsLayoutLocked())
{
BOOL bChanged = FALSE;
switch(m_Position)
{
case PANEL_POPUP:
if(!(pWindowPos->flags & SWP_NOMOVE)) DimChange(m_lX , pWindowPos->x , bChanged);
if(!(pWindowPos->flags & SWP_NOMOVE)) DimChange(m_lY , pWindowPos->y , bChanged);
if(!(pWindowPos->flags & SWP_NOSIZE)) DimChange(m_lWidth , pWindowPos->cx, bChanged);
if(!(pWindowPos->flags & SWP_NOSIZE)) DimChange(m_lHeight, pWindowPos->cy, bChanged);
break;
case PANEL_BOTTOM:
case PANEL_TOP:
if(!(pWindowPos->flags & SWP_NOSIZE)) DimChange(m_lHeight, pWindowPos->cy, bChanged);
break;
case PANEL_LEFT:
case PANEL_RIGHT:
if(!(pWindowPos->flags & SWP_NOSIZE)) DimChange(m_lWidth, pWindowPos->cx, bChanged);
break;
}
if(bChanged)
{
OnLayoutChange();
}
}
}
//------------------------------------------------------------------------------
void CMarsPanel::GetMinMaxInfo( POINT& ptMin, POINT& ptMax )
{
long lWidth = m_lWidth;
long lHeight = m_lHeight;
long lMinWidth;
long lMinHeight;
long lMaxWidth;
long lMaxHeight;
if(!IsAutoSizing())
{
lMinWidth = m_lMinWidth;
lMinHeight = m_lMinHeight;
lMaxWidth = m_lMaxWidth;
lMaxHeight = m_lMaxHeight;
}
else
{
ComputeDimensionsOfContent( &lMinWidth, &lMinHeight );
if(m_lMinWidth != -1 && lMinWidth < m_lMinWidth ) lMinWidth = m_lMinWidth;
if(m_lMinHeight != -1 && lMinHeight < m_lMinHeight) lMinHeight = m_lMinHeight;
lMaxWidth = m_lMaxWidth;
lMaxHeight = m_lMaxHeight;
if(lMinWidth > lWidth ) lWidth = lMinWidth;
if(lMinHeight > lHeight) lHeight = lMinHeight;
}
switch(m_Position)
{
case PANEL_BOTTOM:
case PANEL_TOP :
if(lMinHeight < 0) lMinHeight = lHeight;
if(lMaxHeight < 0) lMaxHeight = lHeight;
break;
case PANEL_LEFT :
case PANEL_RIGHT:
if(lMinWidth < 0) lMinWidth = lWidth;
if(lMaxWidth < 0) lMaxWidth = lWidth;
break;
}
ptMin.x = lMinWidth;
ptMin.y = lMinHeight;
ptMax.x = lMaxWidth;
ptMax.y = lMaxHeight;
}
bool CMarsPanel::CanLayout( RECT& rcClient, POINT& ptDiff )
{
ptDiff.x = 0;
ptDiff.y = 0;
if(IsVisible())
{
RECT rcClient2;
POINT ptMin;
POINT ptMax;
long lWidth;
long lHeight;
GetRect ( &rcClient, &rcClient2 );
GetMinMaxInfo( ptMin , ptMax );
lWidth = rcClient2.right - rcClient2.left;
lHeight = rcClient2.bottom - rcClient2.top;
if(ptMin.x >= 0 && lWidth < ptMin.x) ptDiff.x -= (lWidth - ptMin.x);
if(ptMax.x >= 0 && lWidth > ptMax.x) ptDiff.x -= (lWidth - ptMax.x);
if(ptMin.y >= 0 && lHeight < ptMin.y) ptDiff.y -= (lHeight - ptMin.y);
if(ptMax.y >= 0 && lHeight > ptMax.y) ptDiff.y -= (lHeight - ptMax.y);
}
return ptDiff.x == 0 && ptDiff.y == 0;
}
//------------------------------------------------------------------------------
// S_FALSE : We used up all remaining client area
// E_INVALIDARG : *prcClient was empty, so we are hidden
HRESULT CMarsPanel::Layout( RECT *prcClient )
{
ATLASSERT(prcClient);
ATLASSERT(!IsPassive());
RECT rcMyClient;
HRESULT hr = S_OK;
if (m_fVisible && !IsRectEmpty(prcClient))
{
hr = GetRect(prcClient, &rcMyClient);
// Optimize out the case that the rect is the same as we already have. This
// is pretty common, and Windows doesn't optimize it out.
RECT rcCurrent;
GetMyClientRectInParentCoords(&rcCurrent);
if(memcmp( &rcCurrent, &rcMyClient, sizeof(RECT) ))
{
m_Content.MoveWindow( &rcMyClient, TRUE );
}
if (IsPopup())
{
// Bring popup windows to top
m_Content.SetWindowPos(HWND_TOP, 0, 0, 0, 0, SWP_NOACTIVATE|SWP_NOMOVE|SWP_NOSIZE);
}
MakeVisible(VARIANT_TRUE, VARIANT_FALSE);
}
else
{
if (m_fVisible)
{
// We wanted to be visible but had no remaining client area
hr = E_INVALIDARG;
}
MakeVisible(VARIANT_FALSE, VARIANT_FALSE);
}
return hr;
}
//------------------------------------------------------------------------------
// IPropertyNotifySink::OnChanged
//
STDMETHODIMP CMarsPanel::OnChanged(DISPID dispID)
{
if (DISPID_READYSTATE == dispID)
{
VARIANT vResult = {0};
EXCEPINFO excepInfo;
UINT uArgErr;
DISPPARAMS dp = {NULL, NULL, 0, 0};
CComPtr<IHTMLDocument2> spDocument;
if (SUCCEEDED(m_Content.QueryControl(IID_IHTMLDocument2, (void **)&spDocument))
&& SUCCEEDED(spDocument->Invoke(DISPID_READYSTATE, IID_NULL, LOCALE_SYSTEM_DEFAULT,
DISPATCH_PROPERTYGET, &dp, &vResult, &excepInfo,
&uArgErr)))
{
m_lReadyState = (READYSTATE)V_I4(&vResult);
switch (m_lReadyState)
{
case READYSTATE_UNINITIALIZED: //= 0,
break;
case READYSTATE_LOADING: //= 1,
break;
case READYSTATE_LOADED: //= 2,
break;
case READYSTATE_INTERACTIVE: //= 3,
break;
case READYSTATE_COMPLETE: //= 4
if (IsAutoSizing())
{
OnLayoutChange();
}
m_spMarsDocument->GetPlaces()->OnPanelReady();
break;
}
}
}
return NOERROR;
}
//------------------------------------------------------------------------------
void CMarsPanel::ConnectCompletionAdviser()
{
if (!m_dwCookie)
{
CComPtr<IConnectionPointContainer> spICPC;
if (SUCCEEDED(m_Content.QueryControl(IID_IConnectionPointContainer, (void **)&spICPC)))
{
CComPtr<IConnectionPoint> spCP;
if (spICPC && SUCCEEDED(spICPC->FindConnectionPoint(IID_IPropertyNotifySink, &spCP)))
{
spCP->Advise((LPUNKNOWN)(IPropertyNotifySink*)this, &m_dwCookie);
ATLASSERT(m_dwCookie);
}
}
}
}
//------------------------------------------------------------------------------
void CMarsPanel::DisconnectCompletionAdviser()
{
if (m_dwCookie)
{
CComPtr<IConnectionPointContainer> spICPC;
if (SUCCEEDED(m_Content.QueryControl(IID_IConnectionPointContainer, (void **)&spICPC)))
{
CComPtr<IConnectionPoint> spCP;
if (spICPC && SUCCEEDED(spICPC->FindConnectionPoint(IID_IPropertyNotifySink, &spCP)))
{
spCP->Unadvise(m_dwCookie);
m_dwCookie = 0;
}
}
}
}
//------------------------------------------------------------------------------
// Given some content, we're going to compute its dimensions and use those.
//
void CMarsPanel::ComputeDimensionsOfContent(long *plWidth, long *plHeight)
{
ATLASSERT(plWidth && plHeight);
ATLASSERT(!IsPassive());
*plWidth = -1;
*plHeight = -1;
if(!IsWebBrowser() && IsAutoSizing())
{
CComPtr<IHTMLDocument2> spDocument;
if(SUCCEEDED(m_Content.QueryControl(IID_IHTMLDocument2, (void **)&spDocument)))
{
CComPtr<IHTMLElement> spBody;
if(spDocument)
{
spDocument->get_body( &spBody );
}
BOOL fScrollBar = FALSE;
BOOL fHeight = (m_Position == PANEL_BOTTOM) ||
(m_Position == PANEL_TOP ) ||
(m_Position == PANEL_POPUP );
BOOL fWidth = (m_Position == PANEL_LEFT ) ||
(m_Position == PANEL_RIGHT) ||
(m_Position == PANEL_POPUP);
CComQIPtr<IHTMLElement2> spBody2(spBody);
if(spBody2 && fHeight)
{
LONG lScrollHeight = 0; spBody2->get_scrollHeight( &lScrollHeight );
if(m_lMaxHeight >= 0 && lScrollHeight > m_lMaxHeight)
{
*plHeight = m_lMaxHeight;
fScrollBar = TRUE;
}
else
{
*plHeight = lScrollHeight;
}
}
if(spBody2 && fWidth)
{
LONG lScrollWidth = 0; spBody2->get_scrollWidth( &lScrollWidth );
if(m_lMaxWidth >= 0 && lScrollWidth > m_lMaxWidth)
{
*plWidth = m_lMaxWidth;
fScrollBar = TRUE;
}
else
{
*plWidth = lScrollWidth;
}
}
CComQIPtr<IHTMLBodyElement> spBody3(spBody);
if(spBody3)
{
spBody3->put_scroll( CComBSTR( fScrollBar ? L"yes" : L"no" ) );
}
}
}
}
//------------------------------------------------------------------------------
// Given the remaining client rect that we are allowed to use, calculate our
// own position in client coordinates, and return the remaining empty client
// rectangle
//
// S_OK : Remaining client area
// S_FALSE : No client area remaining
HRESULT CMarsPanel::GetRect(RECT *prcClient, RECT *prcMyClient)
{
ATLASSERT(!IsPassive());
HRESULT hr = S_OK;
if (!m_fVisible)
{
memset(prcMyClient, 0, sizeof(*prcMyClient));
return S_OK;
}
*prcMyClient = *prcClient;
long lWidth = m_lWidth;
long lHeight = m_lHeight;
if (IsAutoSizing())
{
long lMinWidth;
long lMinHeight;
ComputeDimensionsOfContent(&lMinWidth, &lMinHeight);
if(lMinWidth > lWidth ) lWidth = lMinWidth;
if(lMinHeight > lHeight) lHeight = lMinHeight;
}
if(m_lMinWidth >= 0 && lWidth < m_lMinWidth ) lWidth = m_lMinWidth;
if(m_lMinHeight >= 0 && lHeight < m_lMinHeight) lHeight = m_lMinHeight;
switch (m_Position)
{
case PANEL_POPUP:
// Special case: we exist on top of the other panels.
if (m_lX < 0)
{
prcMyClient->right = prcClient->right + 1 + m_lX;
prcMyClient->left = prcMyClient->right - lWidth;
}
else
{
prcMyClient->left = prcClient->left + m_lX;
prcMyClient->right = prcMyClient->left + lWidth;
}
if (m_lY < 0)
{
prcMyClient->bottom = prcClient->bottom + 1 + m_lY;
prcMyClient->top = prcMyClient->bottom - lHeight;
}
else
{
prcMyClient->top = prcClient->top + m_lY;
prcMyClient->bottom = prcMyClient->top + lHeight;
}
break;
case PANEL_LEFT:
prcClient->left = prcMyClient->right = prcClient->left + lWidth;
if (prcClient->left > prcClient->right)
{
prcClient->left = prcMyClient->right = prcClient->right;
hr = S_FALSE;
}
break;
case PANEL_RIGHT:
prcClient->right = prcMyClient->left = prcClient->right - lWidth;
if (prcClient->right < prcClient->left)
{
prcClient->right = prcMyClient->left = prcClient->left;
hr = S_FALSE;
}
break;
case PANEL_TOP:
prcClient->top = prcMyClient->bottom = prcClient->top + lHeight;
if (prcClient->top > prcClient->bottom)
{
prcClient->top = prcMyClient->bottom = prcClient->bottom;
hr = S_FALSE;
}
break;
case PANEL_BOTTOM:
prcClient->bottom = prcMyClient->top = prcClient->bottom - lHeight;
if (prcClient->bottom < prcClient->top)
{
prcClient->bottom = prcMyClient->top = prcClient->top;
hr = S_FALSE;
}
break;
case PANEL_WINDOW:
hr = S_FALSE;
break;
default:
ATLASSERT(FALSE); // Invalid panel position.
break;
}
return hr;
}
// =========================================================
// CBrowserEvents
// =========================================================
CMarsPanel::CBrowserEvents::CBrowserEvents(CMarsPanel *pParent) :
CMarsPanelSubObject(pParent)
{
ATLASSERT(m_dwCookie == 0);
ATLASSERT(m_dwCookie2 == 0);
}
IMPLEMENT_ADDREF_RELEASE(CMarsPanel::CBrowserEvents);
//------------------------------------------------------------------------------
void CMarsPanel::CBrowserEvents::Connect(IUnknown *punk, BOOL bConnect)
{
CComPtr<IConnectionPointContainer> spCpc;
if (SUCCEEDED(punk->QueryInterface(IID_IConnectionPointContainer, (void **)&spCpc)))
{
CComPtr<IConnectionPoint> spCp;
if (SUCCEEDED(spCpc->FindConnectionPoint(DIID_DWebBrowserEvents, &spCp)))
{
if (bConnect)
{
spCp->Advise(this, &m_dwCookie);
}
else if (m_dwCookie)
{
spCp->Unadvise(m_dwCookie);
m_dwCookie = 0;
}
}
spCp.Release();
if (SUCCEEDED(spCpc->FindConnectionPoint(DIID_DWebBrowserEvents2, &spCp)))
{
if (bConnect)
{
spCp->Advise(this, &m_dwCookie2);
}
else if (m_dwCookie2)
{
spCp->Unadvise(m_dwCookie2);
m_dwCookie2 = 0;
}
}
}
}
//------------------------------------------------------------------------------
// IUnknown::QueryInterface
//
HRESULT CMarsPanel::CBrowserEvents::QueryInterface(REFIID iid, void **ppvObject)
{
HRESULT hr;
if ((iid == IID_IUnknown) ||
(iid == IID_IDispatch))
{
AddRef();
*ppvObject = SAFECAST(this, IDispatch *);
hr = S_OK;
}
else
{
*ppvObject = NULL;
hr = E_NOINTERFACE;
}
return hr;
}
//------------------------------------------------------------------------------
// IDispatch::Invoke
//
HRESULT CMarsPanel::CBrowserEvents::Invoke(DISPID dispidMember, REFIID riid,
LCID lcid, WORD wFlags, DISPPARAMS* pdispparams, VARIANT* pvarResult,
EXCEPINFO* pexcepinfo, UINT* puArgErr)
{
HRESULT hr = S_OK;
if (VerifyNotPassive(&hr))
{
switch (dispidMember)
{
case DISPID_BEFORENAVIGATE:
case DISPID_FRAMEBEFORENAVIGATE:
case DISPID_BEFORENAVIGATE2:
{
VARIANT *pVarCancel = &pdispparams->rgvarg[0];
if (pVarCancel->vt == (VT_BOOL | VT_BYREF))
{
if (VARIANT_TRUE == *pVarCancel->pboolVal)
{
CComPtr<IHTMLDocument2> spDoc2;
GetDoc2FromAxWindow(&Parent()->m_Content, &spDoc2);
if (spDoc2)
{
VARIANT vResult = {0};
EXCEPINFO excepInfo;
UINT uArgErr;
DISPPARAMS dp = {NULL, NULL, 0, 0};
if (SUCCEEDED(spDoc2->Invoke(DISPID_READYSTATE, IID_NULL, LOCALE_SYSTEM_DEFAULT,
DISPATCH_PROPERTYGET, &dp, &vResult, &excepInfo, &uArgErr)))
{
Parent()->m_lReadyState = (READYSTATE)V_I4(&vResult);
}
}
}
else
{
Parent()->m_lReadyState = READYSTATE_LOADING;
}
if (READYSTATE_COMPLETE == Parent()->m_lReadyState)
{
Parent()->m_spMarsDocument->GetPlaces()->OnPanelReady();
}
}
break;
}
case DISPID_DOCUMENTCOMPLETE:
{
Parent()->m_lReadyState = READYSTATE_COMPLETE;
Parent()->m_spMarsDocument->GetPlaces()->OnPanelReady();
break;
}
} // switch(dispidMember)
}
return hr;
}
//------------------------------------------------------------------------------
STDMETHODIMP CMarsPanel::TranslateAccelerator(MSG *pMsg, DWORD grfModifiers)
{
HRESULT hr;
ATLASSERT(!m_fTabCycle);
if (IsVK_TABCycler(pMsg))
{
m_fTabCycle = TRUE;
hr = S_OK;
}
else
{
hr = S_FALSE;
}
return hr;
}
//------------------------------------------------------------------------------
STDMETHODIMP CMarsPanel::OnUIActivate()
{
ATLASSERT(!IsPassive());
m_spPanelCollection->SetActivePanel(this, TRUE);
return S_OK;
}
//------------------------------------------------------------------------------
HRESULT CMarsPanel::UIDeactivate()
{
HRESULT hr;
CComPtr<IOleInPlaceObject> spOleInPlaceObject;
if (SUCCEEDED(m_Content.QueryControl(&spOleInPlaceObject)))
{
hr = spOleInPlaceObject->UIDeactivate();
}
else
{
// What the heck else can we do?
hr = S_FALSE;
}
return hr;
}
//------------------------------------------------------------------------------
// Forwards the message to the hosted control.
void CMarsPanel::ForwardMessage(UINT uMsg, WPARAM wParam, LPARAM lParam)
{
HWND hwnd;
CComPtr<IOleWindow> spOleWindow;
// Note that we send the message directly to the window rather than use
// WM_FORWARDMSG which calls TranslateAccelerator
if (SUCCEEDED(m_Content.QueryControl(&spOleWindow)) &&
SUCCEEDED(spOleWindow->GetWindow(&hwnd)))
{
SendMessage(hwnd, uMsg, wParam, lParam);
}
}
//------------------------------------------------------------------------------
HRESULT CMarsPanel::TranslateAccelerator(MSG *pMsg)
{
ATLASSERT(!IsPassive());
HRESULT hr = S_FALSE;
BOOL fGlobal = IsGlobalKeyMessage( pMsg );
if((S_OK == SHIsChildOrSelf(m_Content.m_hWnd, pMsg->hwnd)) || fGlobal)
{
if(m_spActiveObject && (this == m_spPanelCollection->ActivePanel()))
{
hr = m_spActiveObject->TranslateAccelerator(pMsg);
}
else
{
CComPtr<IOleInPlaceActiveObject> obj;
if(SUCCEEDED(m_Content.QueryControl( IID_IOleInPlaceActiveObject, (void **)&obj )))
{
hr = obj->TranslateAccelerator(pMsg);
}
//
// If it's a WebBrowser, forward the accelerator directly to the Document object, otherwise accesskeys won't be resolved.
//
if(hr == S_FALSE && fGlobal)
{
CComPtr<IWebBrowser2> obj2;
if(SUCCEEDED(m_Content.QueryControl( IID_IWebBrowser2, (void **)&obj2 )))
{
CComPtr<IDispatch> disp;
if(SUCCEEDED(obj2->get_Document( &disp )) && disp)
{
CComPtr<IOleInPlaceActiveObject> obj3;
if(SUCCEEDED(disp.QueryInterface( &obj3 )))
{
hr = obj3->TranslateAccelerator(pMsg);
}
}
}
}
}
}
return hr;
}
//----------------------------------------------------------------------------
// Returns the Screen coordinates of this panel
//----------------------------------------------------------------------------
void CMarsPanel::GetMyClientRectInParentCoords(RECT *prc)
{
ATLASSERT(!IsPassive());
POINT ptParent = {0, 0}, ptMe = {0, 0}, ptOffset;
m_Content.ClientToScreen(&ptMe);
m_spMarsDocument->Window()->ClientToScreen(&ptParent);
ptOffset.x = ptMe.x - ptParent.x;
ptOffset.y = ptMe.y - ptParent.y;
m_Content.GetClientRect(prc);
OffsetRect(prc, ptOffset.x, ptOffset.y);
}
// IInternetSecurityManager
// This interface is used to override default security settings for our panels.
// These panels are trusted.
//------------------------------------------------------------------------------
// IInternetSecurityManager::SetSecuritySite
//
HRESULT CMarsPanel::SetSecuritySite(IInternetSecurityMgrSite *pSite)
{
return E_NOTIMPL;
}
//------------------------------------------------------------------------------
// IInternetSecurityManager::GetSecuritySite
//
HRESULT CMarsPanel::GetSecuritySite(IInternetSecurityMgrSite **ppSite)
{
return E_NOTIMPL;
}
//------------------------------------------------------------------------------
// IInternetSecurityManager::MapUrlToZone
//
HRESULT CMarsPanel::MapUrlToZone(LPCWSTR pwszUrl, DWORD *pdwZone, DWORD dwFlags)
{
return INET_E_DEFAULT_ACTION;
}
//------------------------------------------------------------------------------
// IInternetSecurityManager::GetSecurityId
//
HRESULT CMarsPanel::GetSecurityId(LPCWSTR pwszUrl, BYTE *pbSecurityId, DWORD *pcbSecurityId, DWORD_PTR dwReserved)
{
return INET_E_DEFAULT_ACTION;
}
//------------------------------------------------------------------------------
// IInternetSecurityManager::ProcessUrlAction
//
// Be as permissive as we can
//
HRESULT CMarsPanel::ProcessUrlAction(LPCWSTR pwszUrl, DWORD dwAction,
BYTE __RPC_FAR *pPolicy, DWORD cbPolicy, BYTE *pContext,
DWORD cbContext, DWORD dwFlags, DWORD dwReserved)
{
ATLASSERT(IsTrusted());
if (cbPolicy >= sizeof(DWORD))
{
*((DWORD *)pPolicy) = URLPOLICY_ALLOW;
}
return S_OK;
}
//------------------------------------------------------------------------------
// IInternetSecurityManager::QueryCustomPolicy
//
// Be as permissive as we can
//
HRESULT CMarsPanel::QueryCustomPolicy(LPCWSTR pwszUrl, REFGUID guidKey, BYTE **ppPolicy,
DWORD *pcbPolicy, BYTE *pContext, DWORD cbContext, DWORD dwReserved)
{
ATLASSERT(IsTrusted());
ATLASSERT(ppPolicy && !*ppPolicy);
ATLASSERT(pcbPolicy);
if (ppPolicy && pcbPolicy)
{
*ppPolicy = (BYTE *)CoTaskMemAlloc(sizeof(DWORD));
if (*ppPolicy)
{
*pcbPolicy = sizeof(DWORD);
*(DWORD *)*ppPolicy = URLPOLICY_ALLOW;
return S_OK;
}
}
return INET_E_DEFAULT_ACTION;
}
//------------------------------------------------------------------------------
// IInternetSecurityManager::SetZoneMapping
//
HRESULT CMarsPanel::SetZoneMapping(DWORD dwZone, LPCWSTR lpszPattern, DWORD dwFlags)
{
return INET_E_DEFAULT_ACTION;
}
//------------------------------------------------------------------------------
// IInternetSecurityManager::GetZoneMappings
//
HRESULT CMarsPanel::GetZoneMappings(DWORD dwZone, IEnumString **ppenumString, DWORD dwFlags)
{
return INET_E_DEFAULT_ACTION;
}
//------------------------------------------------------------------------------
// OnDocHostUIExec
//
// When we get an Exec from Trident on CGID_DocHostCommandHandler, we can return S_OK
// to indicate that we handled the command and Trident should take no further action
// We'll delegate the processing to script by firing events
//
// TODO: once Mars accelerators are implemented, we should just block Trident from
// taking action, and not fire any events.
//
HRESULT CMarsPanel::OnDocHostUIExec(const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt,
VARIANTARG *pvarargIn, VARIANTARG *pvarargOut)
{
// HACK: Shdocvw/Trident sometimes tests specifically for a value like
// OLECMDERR_E_NOTSUPPORTED and will not perform an essential action
// if we return something more generic like E_FAIL.
HRESULT hr = OLECMDERR_E_NOTSUPPORTED;
if(!IsPassive() && pguidCmdGroup && (*pguidCmdGroup == CGID_DocHostCommandHandler))
{
switch (nCmdID)
{
case OLECMDID_SHOWFIND:
hr = S_OK;
break;
case IDM_NEW_TOPLEVELWINDOW:
// Shdocvw gives us this command when Ctrl+N or the localized equivalent
// is translated. We return S_OK to stop it from opening an IE window.
hr = S_OK;
break;
case IDM_REFRESH:
case IDM_REFRESH_TOP:
case IDM_REFRESH_TOP_FULL:
case IDM_REFRESH_THIS:
case IDM_REFRESH_THIS_FULL:
if(!m_fInRefresh)
{
hr = S_OK;
}
break;
case OLECMDID_SHOWSCRIPTERROR:
if(SUCCEEDED(m_spMarsDocument->MarsWindow()->NotifyHost( MARSHOST_ON_SCRIPT_ERROR, V_VT(pvarargIn) == VT_UNKNOWN ? V_UNKNOWN(pvarargIn) : NULL, 0 )))
{
V_VT (pvarargOut) = VT_BOOL;
V_BOOL(pvarargOut) = VARIANT_FALSE;
hr = S_OK;
}
break;
}
}
return hr;
}
//
// CPanelCollection implementation
//
//------------------------------------------------------------------------------
CPanelCollection::CPanelCollection(CMarsDocument *pMarsDocument)
{
m_spMarsDocument = pMarsDocument;
}
//------------------------------------------------------------------------------
CPanelCollection::~CPanelCollection()
{
ATLASSERT(GetSize() == 0);
FreePanels();
}
//------------------------------------------------------------------------------
void CPanelCollection::FreePanels()
{
for (int i=0; i<GetSize(); i++)
{
(*this)[i].PassivateAndRelease();
}
RemoveAll();
}
//------------------------------------------------------------------------------
HRESULT CPanelCollection::DoPassivate()
{
FreePanels();
m_spMarsDocument.Release();
return S_OK;
}
IMPLEMENT_ADDREF_RELEASE(CPanelCollection);
//------------------------------------------------------------------------------
// IUnknown::QueryInterface
//
STDMETHODIMP CPanelCollection::QueryInterface(REFIID iid, void **ppvObject)
{
HRESULT hr;
if (API_IsValidWritePtr(ppvObject))
{
if ((iid == IID_IUnknown) ||
(iid == IID_IDispatch) ||
(iid == IID_IMarsPanelCollection))
{
AddRef();
*ppvObject = SAFECAST(this, IMarsPanelCollection *);
hr = S_OK;
}
else
{
*ppvObject = NULL;
hr = E_NOINTERFACE;
}
}
else
{
hr = E_INVALIDARG;
}
return hr;
}
//------------------------------------------------------------------------------
HRESULT CPanelCollection::DoEnableModeless(BOOL fEnable)
{
for (int i=0; i<GetSize(); i++)
{
(*this)[i]->DoEnableModeless(fEnable);
}
return S_OK;
}
//////////////////////////////
// IMarsPanelCollection
//------------------------------------------------------------------------------
// IMarsPanelCollection::get_panel
//
// TODO: (PaulNash, 9/19/99) This method is outdated.
// Remove once content is switched over.
//
STDMETHODIMP CPanelCollection::get_panel(LPWSTR pwszName, IMarsPanel **ppPanel)
{
CComVariant var(pwszName);
return get_item(var, ppPanel);
}
//------------------------------------------------------------------------------
// IMarsPanelCollection::addPanel
//
STDMETHODIMP CPanelCollection::addPanel(
BSTR bstrName,
VARIANT varType,
BSTR bstrStartUrl,
VARIANT varCreate,
long lFlags,
IMarsPanel **ppPanel)
{
HRESULT hr = E_INVALIDARG;
if (API_IsValidString(bstrName) &&
(VT_NULL == varType.vt || API_IsValidVariantBstr(varType)) &&
(NULL == bstrStartUrl || API_IsValidString(bstrStartUrl)) &&
(VT_NULL == varCreate.vt || API_IsValidVariantBstr(varCreate)) &&
API_IsValidFlag(lFlags, PANEL_FLAG_ALL) &&
API_IsValidWritePtr(ppPanel) )
{
*ppPanel = NULL;
if (VerifyNotPassive(&hr))
{
MarsAppDef_Panel Layout;
BSTR bstrType = VariantToBSTR( varType );
BSTR bstrCreate = VariantToBSTR( varCreate );
DWORD dwFlags = DEFAULT_PANEL_FLAGS | (DWORD)lFlags;
StringToPanelFlags( bstrType , dwFlags );
StringToPanelFlags( bstrCreate, dwFlags );
StringCchCopyW ( Layout.szName, ARRAYSIZE(Layout.szName), bstrName );
ExpandEnvironmentStringsW( bstrStartUrl , Layout.szUrl, ARRAYSIZE(Layout.szUrl ) );
Layout.dwFlags = dwFlags;
AddPanel(&Layout, NULL);
hr = get_item( CComVariant( bstrName ), ppPanel );
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanelCollection::removePanel
//
STDMETHODIMP CPanelCollection::removePanel(LPWSTR pwszName)
{
HRESULT hr = E_INVALIDARG;
if(API_IsValidString(pwszName))
{
if(VerifyNotPassive(&hr))
{
hr = S_FALSE;
for(int i=0; i<GetSize(); i++)
{
if(!StrCmpI(pwszName, (*this)[i]->GetName()))
{
BOOL fVisible = (*this)[i]->IsVisible();
(*this)[i].PassivateAndRelease();
RemoveAt(i);
if(fVisible)
{
Layout();
}
hr = S_OK;
break;
}
}
}
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanelCollection::lockLayout
//
STDMETHODIMP CPanelCollection::lockLayout()
{
HRESULT hr = S_OK;
if (VerifyNotPassive(&hr))
{
m_iLockLayout++;
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanelCollection::unlockLayout
//
STDMETHODIMP CPanelCollection::unlockLayout()
{
HRESULT hr = S_OK;
if (VerifyNotPassive(&hr))
{
if (IsLayoutLocked())
{
if (0 == --m_iLockLayout)
{
// TODO: clear lock timeout
if (m_fPendingLayout)
{
m_fPendingLayout = FALSE;
Layout();
}
}
}
}
return hr;
}
//------------------------------------------------------------------------------
void CPanelCollection::Layout()
{
if(!IsPassive())
{
if(!IsLayoutLocked())
{
RECT rcClient;
lockLayout();
m_spMarsDocument->Window()->GetClientRect( &rcClient );
if(m_spMarsDocument->MarsWindow()->CanLayout( rcClient ) == false)
{
m_spMarsDocument->MarsWindow()->FixLayout( rcClient );
}
m_fPendingLayout = FALSE;
for (int i=0; i<GetSize(); i++)
{
if (S_OK != (*this)[i]->Layout( &rcClient ))
{
// We're out of client area; won't be able to show all panels.
// Keep calling Layout for remaining panels so they can hide themselves.
}
}
ATLASSERT(!m_fPendingLayout);
unlockLayout();
}
else
{
// We'll do the layout once we get unlocked
m_fPendingLayout = TRUE;
}
}
}
//------------------------------------------------------------------------------
void CPanelCollection::SetActivePanel(CMarsPanel *pPanel, BOOL bActive)
{
if (bActive)
{
if (m_spActivePanel != pPanel)
{
if (m_spActivePanel)
{
m_spActivePanel->UIDeactivate();
m_spActivePanel.Release();
}
m_spActivePanel = pPanel;
}
}
else
{
// A panel is telling us it doesn't want to be the active
// panel anymore.
if (pPanel == m_spActivePanel)
{
m_spActivePanel->UIDeactivate();
m_spActivePanel.Release();
}
}
}
//------------------------------------------------------------------------------
// IMarsPanelCollection::get_activePanel
//
STDMETHODIMP CPanelCollection::get_activePanel(IMarsPanel **ppPanel)
{
HRESULT hr = E_INVALIDARG;
if (API_IsValidWritePtr(ppPanel))
{
*ppPanel = NULL;
if (VerifyNotPassive(&hr))
{
hr = S_OK;
*ppPanel = ActivePanel();
if (*ppPanel)
{
(*ppPanel)->AddRef();
}
}
}
return hr;
}
//------------------------------------------------------------------------------
HRESULT CPanelCollection::AddPanel( MarsAppDef_Panel* pLayout, /*optional*/ IMarsPanel **ppPanel)
{
ATLASSERT(pLayout);
ATLASSERT(!IsPassive());
HRESULT hr = E_FAIL;
if (ppPanel)
{
*ppPanel = NULL;
}
// If it's a duplicate panel name, just fail.
if (!FindPanel(pLayout->szName))
{
CComClassPtr<CMarsPanel> spPanel;
spPanel.Attach(new CMarsPanel(this, m_spMarsDocument->MarsWindow()));
if (spPanel)
{
if (Add(spPanel))
{
spPanel->Create(pLayout);
if (ppPanel)
{
hr = spPanel->QueryInterface(IID_IMarsPanel, (void **)ppPanel);
}
hr = S_OK;
}
else
{
hr = E_OUTOFMEMORY;
}
}
else
{
hr = E_OUTOFMEMORY;
}
}
return hr;
}
//------------------------------------------------------------------------------
CMarsPanel *CPanelCollection::FindPanel(LPCWSTR pwszName)
{
ATLASSERT(!IsPassive());
CMarsPanel *pPanel = NULL;
int iLen = GetSize();
for (int i = 0; i < iLen; ++i)
{
if (!StrCmpIW(pwszName, (*this)[i]->GetName()))
{
pPanel = (*this)[i];
break;
}
}
return pPanel;
}
//------------------------------------------------------------------------------
HRESULT CPanelCollection::FindPanelIndex(CMarsPanel *pPanel, long *plIndex)
{
ATLASSERT(plIndex);
ATLASSERT(!IsPassive());
HRESULT hr = E_FAIL;
*plIndex = -1;
if (pPanel)
{
long lSize = GetSize();
for (long i = 0; i < lSize; ++i)
{
if (pPanel == (*this)[i])
{
*plIndex = i;
hr = S_OK;
break;
}
}
}
return hr;
}
//------------------------------------------------------------------------------
HRESULT CPanelCollection::InsertPanelFromTo(long lOldIndex, long lNewIndex)
{
ATLASSERT((lOldIndex >= 0) && (lOldIndex < GetSize()));
ATLASSERT(!IsPassive());
HRESULT hr = S_FALSE;
if (lNewIndex < 0)
{
lNewIndex = 0;
}
if (lNewIndex > GetSize() - 1)
{
lNewIndex = GetSize() - 1;
}
// If this is something that is done very often, we should optimize this better
// and probably not use an array
if (lOldIndex != lNewIndex)
{
CComClassPtr<CMarsPanel> spPanel = (*this)[lOldIndex];
RemoveAt(lOldIndex);
InsertAt(lNewIndex, spPanel);
hr = S_OK;
}
return hr;
}
//------------------------------------------------------------------------------
// Sets the dirty bit on all panels after a theme switch
//
void CPanelCollection::InvalidatePanels()
{
for (int i=0; i < GetSize(); i++)
{
(*this)[i]->put_contentInvalid(VARIANT_TRUE);
}
}
//------------------------------------------------------------------------------
// Called after the theme switch event has fired, to refresh any panels
// that are visible but still haven't been updated with the new theme
//
void CPanelCollection::RefreshInvalidVisiblePanels()
{
for (int i=0; i < GetSize(); i++)
{
CMarsPanel *pPanel = (*this)[i];
if (pPanel->IsVisible() && pPanel->IsContentInvalid())
{
pPanel->refresh();
}
}
}
//------------------------------------------------------------------------------
// IMarsPanelCollection::get_length
//
// standard collection method (gets instantaneous length of coll)
//
STDMETHODIMP CPanelCollection::get_length(LONG *plNumPanels)
{
HRESULT hr = E_INVALIDARG;
if (API_IsValidWritePtr(plNumPanels))
{
(*plNumPanels) = GetSize();
hr = S_OK;
}
return hr;
}
//------------------------------------------------------------------------------
// IMarsPanelCollection::get_item
//
// standard collection method (gets an theme given index or name)
//
STDMETHODIMP CPanelCollection::get_item(/*[in]*/ VARIANT varIndexOrName,
/*[out, retval]*/ IMarsPanel **ppPanel)
{
ATLASSERT(VT_BSTR == varIndexOrName.vt || VT_I4 == varIndexOrName.vt);
HRESULT hr = E_INVALIDARG;
//
// We can't use the API_IsValid varieties because they rip, and we actually don't want that.
// It's only valid to RIP on IsValidFailure if only a single type is allowed, but here we
// allow two types in the Variant, so we would only want to RIP if both are false (already
// handled by above RIP).
//
if ((IsValidVariantI4(varIndexOrName) || IsValidVariantBstr(varIndexOrName)) &&
API_IsValidWritePtr(ppPanel))
{
*ppPanel= NULL;
if (VerifyNotPassive(&hr))
{
CMarsPanel *pPanel = NULL;
if (VT_BSTR == varIndexOrName.vt)
{
pPanel = FindPanel(V_BSTR(&varIndexOrName));
if (pPanel)
{
hr = pPanel->QueryInterface(IID_IMarsPanel, (void **)ppPanel);
}
else
{
hr = S_FALSE;
}
}
else if (VT_I4 == varIndexOrName.vt)
{
long idxPanel = V_I4(&varIndexOrName);
if (idxPanel >= 0 && idxPanel < GetSize())
{
pPanel = (*this)[idxPanel];
if (pPanel)
{
hr = pPanel->QueryInterface(IID_IMarsPanel, (void **)ppPanel);
}
else
{
hr = S_FALSE;
}
}
}
else
{
// We only accept VT_BSTR and VT_I4 and we should have already
// detected any other invalid params higher up in the function.
ATLASSERT(false);
}
}
}
return hr;
} // get_item
//------------------------------------------------------------------------------
// IMarsPanelCollection::get__newEnum
// standard collection method (gets a new IEnumVARIANT)
//
STDMETHODIMP CPanelCollection::get__newEnum(/*[out, retval]*/ IUnknown **ppEnumPanels)
{
HRESULT hr = E_INVALIDARG;
if (API_IsValidWritePtr(ppEnumPanels))
{
*ppEnumPanels = NULL;
if (VerifyNotPassive(&hr))
{
// This helper takes a CMarsSimpleArray and does all
// the work of creating a CComEnum for us. Neat!
hr = CMarsComEnumVariant< CMarsPanel >::CreateFromMarsSimpleArray(*this, ppEnumPanels);
}
}
return hr;
} // get__newEnum