mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1019 lines
21 KiB
1019 lines
21 KiB
// File: conflnk.cpp
|
|
|
|
#include "precomp.h"
|
|
|
|
#include <shellapi.h>
|
|
#include <shlstock.h>
|
|
#include <ConfDir.h>
|
|
#include <ConfCli.h>
|
|
|
|
#include "clCnfLnk.hpp"
|
|
#include "resource.h"
|
|
|
|
#include "atlbase.h"
|
|
#include "confevt.h"
|
|
#include "sdkInternal.h"
|
|
#include "nameres.h"
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// Initialize GUIDs
|
|
//
|
|
#pragma data_seg(".text")
|
|
#define INITGUID
|
|
#include <initguid.h>
|
|
#include "CLinkId.h"
|
|
|
|
// from shlguid.h - must keep in sync
|
|
DEFINE_SHLGUID(CLSID_ShellLink, 0x00021401L, 0, 0);
|
|
#ifndef _UNICODE
|
|
DEFINE_SHLGUID(IID_IShellLink, 0x000214EEL, 0, 0);
|
|
DEFINE_SHLGUID(IID_IShellExtInit, 0x000214E8L, 0, 0);
|
|
DEFINE_SHLGUID(IID_IShellPropSheetExt, 0x000214E9L, 0, 0);
|
|
#endif // _UNICODE
|
|
#undef INITGUID
|
|
#pragma data_seg()
|
|
/////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
BOOL IsValidPCConfLink(PCCConfLink pcConfLink)
|
|
{
|
|
return(
|
|
IS_VALID_READ_PTR(pcConfLink, CConfLink) &&
|
|
FLAGS_ARE_VALID(pcConfLink->m_dwFlags, ALL_CONFLNK_FLAGS) &&
|
|
(! pcConfLink->m_pszFile ||
|
|
IS_VALID_STRING_PTR(pcConfLink->m_pszFile, STR)) &&
|
|
(! pcConfLink->m_pszName ||
|
|
IS_VALID_STRING_PTR(pcConfLink->m_pszName, STR)) &&
|
|
IS_VALID_STRUCT_PTR((PCRefCount)pcConfLink, CRefCount) &&
|
|
IS_VALID_INTERFACE_PTR((PCIDataObject)pcConfLink, IDataObject) &&
|
|
IS_VALID_INTERFACE_PTR((PCIPersistFile)pcConfLink, IPersistFile) &&
|
|
IS_VALID_INTERFACE_PTR((PCIPersistStream)pcConfLink, IPersistStream) &&
|
|
IS_VALID_INTERFACE_PTR((PCIShellExtInit)pcConfLink, IShellExtInit) &&
|
|
IS_VALID_INTERFACE_PTR((PCIShellPropSheetExt)pcConfLink, IShellPropSheetExt) &&
|
|
IS_VALID_INTERFACE_PTR((PCIConferenceLink)pcConfLink, IConferenceLink));
|
|
}
|
|
#endif /* DEBUG */
|
|
|
|
|
|
CConfLink::CConfLink(OBJECTDESTROYEDPROC ObjectDestroyed) :
|
|
RefCount (ObjectDestroyed),
|
|
m_dwFlags (0),
|
|
m_dwCallFlags (0),
|
|
m_dwTransport (0),
|
|
m_pszFile (NULL),
|
|
m_pszName (NULL),
|
|
m_pszRemoteConfName (NULL),
|
|
m_pszAddress (NULL)
|
|
{
|
|
DebugEntry(CConfLink::CConfLink);
|
|
|
|
// Don't validate this until after construction.
|
|
|
|
ASSERT( !ObjectDestroyed ||
|
|
IS_VALID_CODE_PTR(ObjectDestroyed, OBJECTDESTROYEDPROC));
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
DebugExitVOID(CConfLink::CConfLink);
|
|
|
|
return;
|
|
}
|
|
|
|
CConfLink::~CConfLink(void)
|
|
{
|
|
DebugEntry(CConfLink::~CConfLink);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
delete m_pszFile;
|
|
m_pszFile = NULL;
|
|
|
|
delete m_pszName;
|
|
m_pszName = NULL;
|
|
|
|
delete m_pszAddress;
|
|
m_pszAddress = NULL;
|
|
|
|
delete m_pszRemoteConfName;
|
|
m_pszRemoteConfName = NULL;
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
DebugExitVOID(CConfLink::~CConfLink);
|
|
|
|
return;
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE CConfLink::SetAddress(PCSTR pcszAddress)
|
|
{
|
|
HRESULT hr;
|
|
PSTR pszNewAddress = NULL;
|
|
|
|
DebugEntry(CConfLink::SetAddress);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT( !pcszAddress ||
|
|
IS_VALID_STRING_PTR(pcszAddress, CSTR));
|
|
|
|
pszNewAddress = new char[lstrlen(pcszAddress) + 1];
|
|
|
|
if (NULL != pszNewAddress)
|
|
{
|
|
lstrcpy(pszNewAddress, pcszAddress);
|
|
hr = S_OK;
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
|
|
if (S_OK == hr)
|
|
{
|
|
if (NULL != m_pszAddress)
|
|
{
|
|
delete m_pszAddress;
|
|
}
|
|
|
|
m_pszAddress = pszNewAddress;
|
|
|
|
Dirty(TRUE);
|
|
|
|
TRACE_OUT(("CConfLink::SetAddress(): Set Address to %s.",
|
|
CHECK_STRING(m_pszAddress)));
|
|
}
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT( hr == S_OK ||
|
|
hr == E_OUTOFMEMORY);
|
|
|
|
DebugExitHRESULT(CConfLink::SetAddress, hr);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE CConfLink::GetAddress(PSTR *ppszAddress)
|
|
{
|
|
HRESULT hr;
|
|
|
|
DebugEntry(CConfLink::GetAddress);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT(IS_VALID_WRITE_PTR(ppszAddress, PSTR));
|
|
|
|
*ppszAddress = NULL;
|
|
|
|
if (NULL != m_pszAddress)
|
|
{
|
|
*ppszAddress = NULL;
|
|
LPMALLOC pMalloc;
|
|
|
|
if (SUCCEEDED(SHGetMalloc(&pMalloc)))
|
|
{
|
|
// (+ 1) for null terminator.
|
|
*ppszAddress = (PSTR) pMalloc->Alloc(lstrlen(m_pszAddress) + 1);
|
|
pMalloc->Release();
|
|
pMalloc = NULL;
|
|
}
|
|
|
|
if (NULL != *ppszAddress)
|
|
{
|
|
lstrcpy(*ppszAddress, m_pszAddress);
|
|
|
|
hr = S_OK;
|
|
|
|
TRACE_OUT(("CConfLink::GetAddress(): Got Address %s.", *ppszAddress));
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// No Name.
|
|
hr = S_FALSE;
|
|
}
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT( (hr == S_OK &&
|
|
IS_VALID_STRING_PTR(*ppszAddress, STR)) ||
|
|
(( hr == S_FALSE ||
|
|
hr == E_OUTOFMEMORY) &&
|
|
! *ppszAddress));
|
|
|
|
DebugExitHRESULT(CConfLink::GetAddress, hr);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE CConfLink::SetRemoteConfName(PCSTR pcszRemoteConfName)
|
|
{
|
|
HRESULT hr;
|
|
|
|
DebugEntry(CConfLink::SetRemoteConfName);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT( !pcszRemoteConfName ||
|
|
IS_VALID_STRING_PTR(pcszRemoteConfName, CSTR));
|
|
|
|
PSTR pszNewRemoteConfName = new char[lstrlen(pcszRemoteConfName) + 1];
|
|
|
|
if (NULL != pszNewRemoteConfName)
|
|
{
|
|
lstrcpy(pszNewRemoteConfName, pcszRemoteConfName);
|
|
hr = S_OK;
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
|
|
if (S_OK == hr)
|
|
{
|
|
if (NULL != m_pszRemoteConfName)
|
|
{
|
|
delete m_pszRemoteConfName;
|
|
}
|
|
|
|
m_pszRemoteConfName = pszNewRemoteConfName;
|
|
|
|
Dirty(TRUE);
|
|
|
|
TRACE_OUT(("CConfLink::SetRemoteConfName(): Set RemoteConfName to %s.",
|
|
CHECK_STRING(m_pszRemoteConfName)));
|
|
}
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT( hr == S_OK ||
|
|
hr == E_OUTOFMEMORY);
|
|
|
|
DebugExitHRESULT(CConfLink::SetRemoteConfName, hr);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE CConfLink::GetRemoteConfName(PSTR *ppszRemoteConfName)
|
|
{
|
|
HRESULT hr;
|
|
|
|
DebugEntry(CConfLink::GetRemoteConfName);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT(IS_VALID_WRITE_PTR(ppszRemoteConfName, PSTR));
|
|
|
|
*ppszRemoteConfName = NULL;
|
|
|
|
if (NULL != m_pszRemoteConfName)
|
|
{
|
|
*ppszRemoteConfName = NULL;
|
|
LPMALLOC pMalloc;
|
|
|
|
if (SUCCEEDED(SHGetMalloc(&pMalloc)))
|
|
{
|
|
// (+ 1) for null terminator.
|
|
*ppszRemoteConfName = (PSTR) pMalloc->Alloc(lstrlen(m_pszRemoteConfName) + 1);
|
|
pMalloc->Release();
|
|
pMalloc = NULL;
|
|
}
|
|
|
|
if (NULL != *ppszRemoteConfName)
|
|
{
|
|
lstrcpy(*ppszRemoteConfName, m_pszRemoteConfName);
|
|
|
|
hr = S_OK;
|
|
|
|
TRACE_OUT(("CConfLink::GetRemoteConfName(): Got RemoteConfName %s.",
|
|
*ppszRemoteConfName));
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// No Name.
|
|
hr = S_FALSE;
|
|
}
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT( (hr == S_OK &&
|
|
IS_VALID_STRING_PTR(*ppszRemoteConfName, STR)) ||
|
|
(( hr == S_FALSE ||
|
|
hr == E_OUTOFMEMORY) &&
|
|
! *ppszRemoteConfName));
|
|
|
|
DebugExitHRESULT(CConfLink::GetRemoteConfName, hr);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE CConfLink::SetName(PCSTR pcszName)
|
|
{
|
|
HRESULT hr;
|
|
BOOL bChanged;
|
|
PSTR pszNewName = NULL;
|
|
|
|
DebugEntry(CConfLink::SetName);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT( !pcszName ||
|
|
IS_VALID_STRING_PTR(pcszName, CSTR));
|
|
|
|
bChanged = ! ((! pcszName && ! m_pszName) ||
|
|
(pcszName && m_pszName &&
|
|
! lstrcmp(pcszName, m_pszName)));
|
|
|
|
if (bChanged && pcszName)
|
|
{
|
|
pszNewName = new(char[lstrlen(pcszName) + 1]);
|
|
|
|
if (pszNewName)
|
|
{
|
|
lstrcpy(pszNewName, pcszName);
|
|
hr = S_OK;
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = S_OK;
|
|
}
|
|
|
|
if (hr == S_OK)
|
|
{
|
|
if (bChanged)
|
|
{
|
|
if (m_pszName)
|
|
{
|
|
delete m_pszName;
|
|
}
|
|
|
|
m_pszName = pszNewName;
|
|
|
|
Dirty(TRUE);
|
|
|
|
TRACE_OUT(("CConfLink::SetName(): Set Name to %s.",
|
|
CHECK_STRING(m_pszName)));
|
|
}
|
|
else
|
|
{
|
|
TRACE_OUT(("CConfLink::SetName(): Name already %s.",
|
|
CHECK_STRING(m_pszName)));
|
|
}
|
|
}
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT( hr == S_OK ||
|
|
hr == E_OUTOFMEMORY);
|
|
|
|
DebugExitHRESULT(CConfLink::SetName, hr);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE CConfLink::GetName(PSTR *ppszName)
|
|
{
|
|
HRESULT hr;
|
|
|
|
DebugEntry(CConfLink::GetName);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT(IS_VALID_WRITE_PTR(ppszName, PSTR));
|
|
|
|
*ppszName = NULL;
|
|
|
|
if (m_pszName)
|
|
{
|
|
*ppszName = NULL;
|
|
LPMALLOC pMalloc;
|
|
|
|
if (SUCCEEDED(SHGetMalloc(&pMalloc)))
|
|
{
|
|
// (+ 1) for null terminator.
|
|
*ppszName = (PSTR) pMalloc->Alloc(lstrlen(m_pszName) + 1);
|
|
pMalloc->Release();
|
|
pMalloc = NULL;
|
|
}
|
|
|
|
// ChrisPi: I changed this to use the above implementation because
|
|
// SHAlloc is not available in the sdk shell32.lib
|
|
|
|
// *ppszName = (PSTR)SHAlloc(lstrlen(m_pszURL) + 1);
|
|
|
|
if (*ppszName)
|
|
{
|
|
lstrcpy(*ppszName, m_pszName);
|
|
|
|
hr = S_OK;
|
|
|
|
TRACE_OUT(("CConfLink::GetName(): Got Name %s.", *ppszName));
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// No Name.
|
|
hr = S_FALSE;
|
|
}
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT( (hr == S_OK &&
|
|
IS_VALID_STRING_PTR(*ppszName, STR)) ||
|
|
(( hr == S_FALSE ||
|
|
hr == E_OUTOFMEMORY) &&
|
|
! *ppszName));
|
|
|
|
DebugExitHRESULT(CConfLink::GetName, hr);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE CConfLink::SetTransport(DWORD dwTransport)
|
|
{
|
|
HRESULT hr;
|
|
|
|
DebugEntry(CConfLink::SetTransport);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
m_dwTransport = dwTransport;
|
|
|
|
Dirty(TRUE);
|
|
|
|
TRACE_OUT(("CConfLink::SetTransport(): Set Transport to %#8x.",
|
|
m_dwTransport));
|
|
|
|
hr = S_OK;
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
DebugExitHRESULT(CConfLink::SetTransport, hr);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE CConfLink::GetTransport(DWORD *pdwTransport)
|
|
{
|
|
HRESULT hr;
|
|
|
|
DebugEntry(CConfLink::GetTransport);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT(IS_VALID_WRITE_PTR(pdwTransport, PDWORD));
|
|
|
|
*pdwTransport = m_dwTransport;
|
|
|
|
hr = S_OK;
|
|
|
|
TRACE_OUT(("CConfLink::GetTransport(): Got Transport %#8x.",
|
|
*pdwTransport));
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
DebugExitHRESULT(CConfLink::GetTransport, hr);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE CConfLink::SetCallFlags(DWORD dwCallFlags)
|
|
{
|
|
HRESULT hr;
|
|
|
|
DebugEntry(CConfLink::SetCallFlags);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
m_dwCallFlags = dwCallFlags;
|
|
|
|
Dirty(TRUE);
|
|
|
|
TRACE_OUT(("CConfLink::SetCallFlags(): Set CallFlags to %#8x.",
|
|
m_dwCallFlags));
|
|
|
|
hr = S_OK;
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
DebugExitHRESULT(CConfLink::SetCallFlags, hr);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
HRESULT STDMETHODCALLTYPE CConfLink::GetCallFlags(DWORD *pdwCallFlags)
|
|
{
|
|
HRESULT hr;
|
|
|
|
DebugEntry(CConfLink::GetCallFlags);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT(IS_VALID_WRITE_PTR(pdwCallFlags, PDWORD));
|
|
|
|
*pdwCallFlags = m_dwCallFlags;
|
|
|
|
hr = S_OK;
|
|
|
|
TRACE_OUT(("CConfLink::GetCallFlags(): Got CallFlags %#8x.",
|
|
*pdwCallFlags));
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
DebugExitHRESULT(CConfLink::GetCallFlags, hr);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
#define STR_CALLTO _T("callto:")
|
|
#define STR_CONFERENCE _T("+conference=")
|
|
#define STR_GATEWAY _T("+gateway=")
|
|
#define STR_SECURE _T("+secure=true")
|
|
#define STR_PHONE _T("+type=phone")
|
|
#define STR_NO_AV _T("+av=false")
|
|
#define STR_NO_DATA _T("+data=false")
|
|
|
|
|
|
//--------------------------------------------------------------------------//
|
|
// CConfLink::InvokeCommand. //
|
|
//--------------------------------------------------------------------------//
|
|
HRESULT
|
|
STDMETHODCALLTYPE
|
|
CConfLink::InvokeCommand
|
|
(
|
|
PCLINVOKECOMMANDINFO pclici
|
|
){
|
|
DebugEntry( CConfLink::InvokeCommand );
|
|
ASSERT( IS_VALID_STRUCT_PTR( this, CConfLink ) );
|
|
ASSERT( IS_VALID_STRUCT_PTR( pclici, CCLINVOKECOMMANDINFO ) );
|
|
|
|
HRESULT result = E_INVALIDARG;
|
|
|
|
if(m_pszAddress)
|
|
{
|
|
if( m_dwTransport == NAMETYPE_CALLTO )
|
|
{
|
|
result = shellCallto( m_pszAddress, IS_FLAG_SET( pclici->dwFlags, ICL_INVOKECOMMAND_FL_ALLOW_UI ) );
|
|
}
|
|
else
|
|
{
|
|
long lMax = lstrlen(STR_CALLTO) +
|
|
lstrlen(STR_CONFERENCE) +
|
|
lstrlen(STR_GATEWAY) +
|
|
lstrlen(STR_PHONE) +
|
|
lstrlen(STR_SECURE) +
|
|
lstrlen(STR_NO_AV) +
|
|
lstrlen(STR_NO_DATA) +
|
|
lstrlen(m_pszAddress) +
|
|
(m_pszRemoteConfName ? lstrlen(m_pszRemoteConfName) : 0) +
|
|
1;
|
|
|
|
LPTSTR url = (TCHAR*) _alloca(lMax);
|
|
|
|
lstrcpy( url, STR_CALLTO );
|
|
|
|
if( m_dwTransport != NAMETYPE_H323GTWY )
|
|
{
|
|
lstrcat( url, m_pszAddress );
|
|
}
|
|
else
|
|
{
|
|
TCHAR * const pSlash = (TCHAR * const) _StrChr( m_pszAddress, '/' );
|
|
|
|
if( pSlash != NULL )
|
|
{
|
|
*pSlash = '\0';
|
|
lstrcat( url, pSlash + 1 );
|
|
lstrcat( url, STR_GATEWAY );
|
|
}
|
|
|
|
lstrcat( url, m_pszAddress );
|
|
lstrcat( url, STR_PHONE );
|
|
}
|
|
|
|
if( m_pszRemoteConfName != NULL )
|
|
{
|
|
lstrcat( url, STR_CONFERENCE );
|
|
lstrcat( url, m_pszRemoteConfName );
|
|
}
|
|
|
|
if( IS_FLAG_SET( m_dwCallFlags, CRPCF_SECURE ) )
|
|
{
|
|
lstrcat( url, STR_SECURE );
|
|
}
|
|
|
|
if( IS_FLAG_CLEAR( m_dwCallFlags, (CRPCF_AUDIO | CRPCF_VIDEO) ) )
|
|
{
|
|
lstrcat( url, STR_NO_AV );
|
|
}
|
|
|
|
if( IS_FLAG_CLEAR( m_dwCallFlags, CRPCF_DATA) )
|
|
{
|
|
lstrcat( url, STR_NO_DATA );
|
|
}
|
|
|
|
result = shellCallto( url, IS_FLAG_SET( pclici->dwFlags, ICL_INVOKECOMMAND_FL_ALLOW_UI ) );
|
|
}
|
|
|
|
ASSERT( IS_VALID_STRUCT_PTR( this, CConfLink ) );
|
|
DebugExitHRESULT( CConfLink::InvokeCommand, result );
|
|
}
|
|
|
|
return( result );
|
|
|
|
} // End of CConfLink::InvokeCommand.
|
|
|
|
|
|
//--------------------------------------------------------------------------//
|
|
// shellCallto. //
|
|
//--------------------------------------------------------------------------//
|
|
HRESULT
|
|
shellCallto
|
|
(
|
|
const TCHAR * const url,
|
|
const bool notifyOnError
|
|
){
|
|
ASSERT( url != NULL );
|
|
ASSERT( url[ 0 ] != '\0' );
|
|
TRACE_OUT( ("shellCallto: url:\"%s\"", url) );
|
|
|
|
HRESULT result;
|
|
|
|
if( url == NULL )
|
|
{
|
|
result = E_POINTER;
|
|
}
|
|
else if( url[ 0 ] == '\0' )
|
|
{
|
|
result = E_INVALIDARG;
|
|
}
|
|
else
|
|
{
|
|
result = ::CoInitializeEx( NULL, COINIT_APARTMENTTHREADED );
|
|
|
|
if( SUCCEEDED( result ) )
|
|
{
|
|
IInternalConfExe * internalConf;
|
|
HRESULT result;
|
|
|
|
result = CoCreateInstance( CLSID_NmManager, NULL, CLSCTX_ALL, IID_IInternalConfExe, (LPVOID *) &internalConf );
|
|
|
|
if( SUCCEEDED( result ) && (internalConf != NULL) )
|
|
{
|
|
result = internalConf->ShellCalltoProtocolHandler( CComBSTR( url ), FALSE );
|
|
|
|
internalConf->Release();
|
|
}
|
|
|
|
::CoUninitialize();
|
|
}
|
|
|
|
if( FAILED( result ) && notifyOnError )
|
|
{
|
|
TCHAR message[ MAX_PATH ];
|
|
TCHAR title[ MAX_PATH ];
|
|
|
|
::LoadString( g_hInst, // handle to module containing string resource
|
|
IDS_COULD_NOT_CALL, // resource identifier
|
|
message, // pointer to buffer for resource
|
|
ARRAY_ELEMENTS( message ) ); // size of buffer
|
|
|
|
::LoadString( g_hInst, // handle to module containing string resource
|
|
IDS_LAUNCH_ERROR, // resource identifier
|
|
title, // pointer to buffer for resource
|
|
ARRAY_ELEMENTS( title ) ); // size of buffer
|
|
|
|
::MessageBox( NULL, message, title, MB_ICONERROR | MB_OK );
|
|
}
|
|
}
|
|
|
|
return( result );
|
|
|
|
} // End of shellCallto.
|
|
|
|
|
|
ULONG STDMETHODCALLTYPE CConfLink::AddRef(void)
|
|
{
|
|
ULONG ulcRef;
|
|
|
|
DebugEntry(CConfLink::AddRef);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
ulcRef = RefCount::AddRef();
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
DebugExitULONG(CConfLink::AddRef, ulcRef);
|
|
|
|
return(ulcRef);
|
|
}
|
|
|
|
|
|
ULONG STDMETHODCALLTYPE CConfLink::Release(void)
|
|
{
|
|
ULONG ulcRef;
|
|
|
|
DebugEntry(CConfLink::Release);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
ulcRef = RefCount::Release();
|
|
|
|
DebugExitULONG(CConfLink::Release, ulcRef);
|
|
|
|
return(ulcRef);
|
|
}
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE CConfLink::QueryInterface(REFIID riid,
|
|
PVOID *ppvObject)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
DebugEntry(CConfLink::QueryInterface);
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
ASSERT(IsValidREFIID(riid));
|
|
ASSERT(IS_VALID_WRITE_PTR(ppvObject, PVOID));
|
|
|
|
if (riid == IID_IDataObject)
|
|
{
|
|
*ppvObject = (PIDataObject)this;
|
|
TRACE_OUT(("CConfLink::QueryInterface(): Returning IDataObject."));
|
|
}
|
|
|
|
#if 0 // These are currently unsupported interfaces:
|
|
|
|
else if (riid == IID_IExtractIcon)
|
|
{
|
|
*ppvObject = NULL;
|
|
hr = E_NOINTERFACE;
|
|
|
|
// *ppvObject = (PIExtractIcon)this;
|
|
TRACE_OUT(("CConfLink::QueryInterface(): NOT Returning IExtractIcon!"));
|
|
}
|
|
else if (riid == IID_INewShortcutHook)
|
|
{
|
|
*ppvObject = NULL;
|
|
hr = E_NOINTERFACE;
|
|
|
|
// *ppvObject = (PINewShortcutHook)this;
|
|
TRACE_OUT(("CConfLink::QueryInterface(): NOT Returning INewShortcutHook!"));
|
|
}
|
|
else if (riid == IID_IShellExecuteHook)
|
|
{
|
|
*ppvObject = NULL;
|
|
hr = E_NOINTERFACE;
|
|
|
|
// *ppvObject = (PIShellExecuteHook)this;
|
|
TRACE_OUT(("CConfLink::QueryInterface(): NOT Returning IShellExecuteHook!"));
|
|
}
|
|
else if (riid == IID_IShellLink)
|
|
{
|
|
*ppvObject = NULL;
|
|
hr = E_NOINTERFACE;
|
|
|
|
// *ppvObject = (PIShellLink)this;
|
|
TRACE_OUT(("CConfLink::QueryInterface(): NOT Returning IShellLink!"));
|
|
}
|
|
#endif /* DEBUG */
|
|
|
|
|
|
// Supported interfaces:
|
|
|
|
else if (riid == IID_IShellExtInit)
|
|
{
|
|
*ppvObject = (PIShellExtInit)this;
|
|
TRACE_OUT(("CConfLink::QueryInterface(): Returning IShellExtInit."));
|
|
}
|
|
else if (riid == IID_IShellPropSheetExt)
|
|
{
|
|
*ppvObject = (PIShellPropSheetExt)this;
|
|
TRACE_OUT(("CConfLink::QueryInterface(): Returning IShellPropSheetExt."));
|
|
}
|
|
else if (riid == IID_IPersist)
|
|
{
|
|
*ppvObject = (PIPersist)(PIPersistStream)this;
|
|
TRACE_OUT(("CConfLink::QueryInterface(): Returning IPersist."));
|
|
}
|
|
else if (riid == IID_IPersistFile)
|
|
{
|
|
*ppvObject = (PIPersistFile)this;
|
|
TRACE_OUT(("CConfLink::QueryInterface(): Returning IPersistFile."));
|
|
}
|
|
else if (riid == IID_IPersistStream)
|
|
{
|
|
*ppvObject = (PIPersistStream)this;
|
|
TRACE_OUT(("CConfLink::QueryInterface(): Returning IPersistStream."));
|
|
}
|
|
else if (riid == IID_IConferenceLink)
|
|
{
|
|
*ppvObject = (PIConferenceLink)this;
|
|
TRACE_OUT(("CConfLink::QueryInterface(): Returning IConferenceLink."));
|
|
}
|
|
else if (riid == IID_IUnknown)
|
|
{
|
|
*ppvObject = (PIUnknown) (PIConferenceLink) this;
|
|
TRACE_OUT(("CConfLink::QueryInterface(): Returning IUnknown."));
|
|
}
|
|
else
|
|
{
|
|
TRACE_OUT(("CConfLink::QueryInterface(): Called on unknown interface."));
|
|
*ppvObject = NULL;
|
|
hr = E_NOINTERFACE;
|
|
}
|
|
|
|
if (hr == S_OK)
|
|
{
|
|
AddRef();
|
|
}
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(this, CConfLink));
|
|
|
|
DebugExitHRESULT(CConfLink::QueryInterface, hr);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
|
|
|
|
/* O P E N C O N F L I N K */
|
|
/*-------------------------------------------------------------------------
|
|
%%Function: OpenConfLink
|
|
|
|
-------------------------------------------------------------------------*/
|
|
void WINAPI OpenConfLink(HWND hwndParent, HINSTANCE hinst,
|
|
PSTR pszCmdLine, int nShowCmd)
|
|
{
|
|
HRESULT hr;
|
|
CConfLink conflink(NULL);
|
|
// int nResult;
|
|
|
|
DebugEntry(OpenConfLink);
|
|
|
|
ASSERT(IS_VALID_HANDLE(hwndParent, WND));
|
|
ASSERT(IS_VALID_HANDLE(hinst, INSTANCE));
|
|
ASSERT(IS_VALID_STRING_PTR(pszCmdLine, STR));
|
|
ASSERT(IsValidShowCmd(nShowCmd));
|
|
|
|
// Assume the entire command line is a Conference Link file path.
|
|
|
|
// TrimWhiteSpace(pszCmdLine);
|
|
|
|
TRACE_OUT(("OpenConfLink(): Trying to open Conference Link %s.",
|
|
pszCmdLine));
|
|
|
|
#if 0
|
|
TCHAR szBuf[256];
|
|
|
|
wsprintf(szBuf, "Launching conference link %s", pszCmdLine);
|
|
|
|
::MessageBox( NULL,
|
|
szBuf,
|
|
"Conference Link Extension",
|
|
MB_ICONINFORMATION);
|
|
#endif
|
|
|
|
hr = conflink.LoadFromFile(pszCmdLine, TRUE);
|
|
|
|
if (hr == S_OK)
|
|
{
|
|
CLINVOKECOMMANDINFO clici;
|
|
|
|
clici.dwcbSize = sizeof(clici);
|
|
clici.hwndParent = hwndParent;
|
|
clici.pcszVerb = NULL;
|
|
clici.dwFlags = ( ICL_INVOKECOMMAND_FL_ALLOW_UI |
|
|
ICL_INVOKECOMMAND_FL_USE_DEFAULT_VERB);
|
|
|
|
hr = conflink.InvokeCommand(&clici);
|
|
}
|
|
else
|
|
{
|
|
// BUGBUG: need resource here INTL
|
|
|
|
::MessageBox( hwndParent,
|
|
_TEXT("There was an error while identifying the conference"),
|
|
_TEXT("Conference Shortcut Error"),
|
|
MB_ICONINFORMATION);
|
|
|
|
// if (MyMsgBox(hwndParent, MAKEINTRESOURCE(IDS_SHORTCUT_ERROR_TITLE),
|
|
// MAKEINTRESOURCE(IDS_LOADFROMFILE_FAILED),
|
|
// (MB_OK | MB_ICONEXCLAMATION), &nResult, pszCmdLine))
|
|
//
|
|
// ASSERT(nResult == IDOK);
|
|
}
|
|
|
|
DebugExitVOID(OpenConfLink);
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
/* A N Y M E A T */
|
|
/*-------------------------------------------------------------------------
|
|
%%Function: AnyMeat
|
|
|
|
Return TRUE if the string has any non-whitespace characters
|
|
-------------------------------------------------------------------------*/
|
|
BOOL AnyMeat(PCSTR pcsz)
|
|
{
|
|
TCHAR ch;
|
|
|
|
ASSERT((!pcsz) || IS_VALID_STRING_PTR(pcsz, CSTR));
|
|
|
|
if (NULL == pcsz)
|
|
return FALSE;
|
|
|
|
while (0 != (ch = *pcsz++))
|
|
{
|
|
switch (ch)
|
|
{
|
|
case _T(' '): // Space
|
|
case _T('\t'): // Tab
|
|
case 0x00A0: // Non-breaking space
|
|
break;
|
|
default:
|
|
return TRUE; // found a non-whitespace character
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
|
|
HRESULT MyReleaseStgMedium(PSTGMEDIUM pstgmed)
|
|
{
|
|
HRESULT hr;
|
|
|
|
ASSERT(IS_VALID_STRUCT_PTR(pstgmed, CSTGMEDIUM));
|
|
|
|
if (pstgmed->pUnkForRelease)
|
|
hr = pstgmed->pUnkForRelease->Release();
|
|
else
|
|
{
|
|
switch(pstgmed->tymed)
|
|
{
|
|
case TYMED_HGLOBAL:
|
|
hr = (! GlobalFree(pstgmed->hGlobal)) ? S_OK : E_HANDLE;
|
|
break;
|
|
|
|
#if 0 /* Only TYMED_HGLOBAL is ever used */
|
|
case TYMED_ISTREAM:
|
|
hr = pstgmed->pstm->Release();
|
|
break;
|
|
|
|
case TYMED_ISTORAGE:
|
|
hr = pstgmed->pstm->Release();
|
|
break;
|
|
|
|
case TYMED_FILE:
|
|
{
|
|
PIMalloc pMalloc = NULL;
|
|
|
|
if (SUCCEEDED(SHGetMalloc(&pMalloc)))
|
|
{
|
|
pMalloc->Free(pstgmed->lpszFileName);
|
|
pMalloc->Release();
|
|
pMalloc = NULL;
|
|
pstgmed->lpszFileName = NULL;
|
|
}
|
|
hr = S_OK;
|
|
break;
|
|
}
|
|
|
|
case TYMED_GDI:
|
|
hr = (DeleteObject(pstgmed->hBitmap)) ? S_OK : E_HANDLE;
|
|
break;
|
|
|
|
case TYMED_MFPICT:
|
|
hr = (DeleteMetaFile((HMETAFILE)(pstgmed->hMetaFilePict)) &&
|
|
! GlobalFree(pstgmed->hMetaFilePict)) ? S_OK : E_HANDLE;
|
|
break;
|
|
|
|
case TYMED_ENHMF:
|
|
hr = (DeleteEnhMetaFile(pstgmed->hEnhMetaFile)) ? S_OK : E_HANDLE;
|
|
break;
|
|
#endif // 0
|
|
|
|
default:
|
|
ASSERT(pstgmed->tymed == TYMED_NULL);
|
|
hr = S_OK;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return(hr);
|
|
}
|
|
|