4038 lines
112 KiB
C
4038 lines
112 KiB
C
/*
|
|
* Copyright 2015 Jacek Caban for CodeWeavers
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#define COBJMACROS
|
|
|
|
#include "windows.h"
|
|
#include "initguid.h"
|
|
#include "dispex.h"
|
|
#include "ole2.h"
|
|
#include "olectl.h"
|
|
#include "objsafe.h"
|
|
#include "activscp.h"
|
|
#include "rpcproxy.h"
|
|
#include "msscript.h"
|
|
DEFINE_GUID(GUID_NULL,0,0,0,0,0,0,0,0,0,0,0);
|
|
|
|
#include "wine/debug.h"
|
|
#include "wine/heap.h"
|
|
#include "wine/list.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(msscript);
|
|
|
|
#ifdef _WIN64
|
|
|
|
#define IActiveScriptParse_Release IActiveScriptParse64_Release
|
|
#define IActiveScriptParse_InitNew IActiveScriptParse64_InitNew
|
|
#define IActiveScriptParse_ParseScriptText IActiveScriptParse64_ParseScriptText
|
|
|
|
#else
|
|
|
|
#define IActiveScriptParse_Release IActiveScriptParse32_Release
|
|
#define IActiveScriptParse_InitNew IActiveScriptParse32_InitNew
|
|
#define IActiveScriptParse_ParseScriptText IActiveScriptParse32_ParseScriptText
|
|
|
|
#endif
|
|
|
|
struct ScriptControl;
|
|
typedef struct ConnectionPoint ConnectionPoint;
|
|
typedef struct ScriptProcedureCollection ScriptProcedureCollection;
|
|
typedef struct ScriptHost ScriptHost;
|
|
|
|
struct ConnectionPoint {
|
|
IConnectionPoint IConnectionPoint_iface;
|
|
ScriptControl *control;
|
|
const IID *riid;
|
|
ConnectionPoint *next;
|
|
};
|
|
|
|
struct named_item {
|
|
struct list entry;
|
|
BSTR name;
|
|
IDispatch *disp;
|
|
};
|
|
|
|
struct module_enum {
|
|
IEnumVARIANT IEnumVARIANT_iface;
|
|
LONG ref;
|
|
|
|
UINT pos;
|
|
ScriptHost *host;
|
|
ScriptControl *control;
|
|
};
|
|
|
|
typedef struct {
|
|
IScriptModule IScriptModule_iface;
|
|
LONG ref;
|
|
|
|
BSTR name;
|
|
ScriptHost *host;
|
|
IDispatch *script_dispatch;
|
|
ITypeInfo *script_typeinfo;
|
|
ITypeComp *script_typecomp;
|
|
|
|
ScriptProcedureCollection *procedures;
|
|
} ScriptModule;
|
|
|
|
typedef struct {
|
|
IScriptProcedure IScriptProcedure_iface;
|
|
LONG ref;
|
|
|
|
ULONG hash;
|
|
struct list entry;
|
|
|
|
BSTR name;
|
|
USHORT num_args;
|
|
VARTYPE ret_type;
|
|
} ScriptProcedure;
|
|
|
|
struct ScriptProcedureCollection {
|
|
IScriptProcedureCollection IScriptProcedureCollection_iface;
|
|
LONG ref;
|
|
|
|
LONG count;
|
|
ScriptModule *module;
|
|
struct list hash_table[43];
|
|
};
|
|
|
|
struct procedure_enum {
|
|
IEnumVARIANT IEnumVARIANT_iface;
|
|
LONG ref;
|
|
|
|
WORD pos;
|
|
WORD count;
|
|
ScriptProcedureCollection *procedures;
|
|
};
|
|
|
|
typedef struct {
|
|
IScriptError IScriptError_iface;
|
|
IActiveScriptError *object;
|
|
LONG ref;
|
|
|
|
HRESULT number;
|
|
BSTR text;
|
|
BSTR source;
|
|
BSTR desc;
|
|
BSTR help_file;
|
|
DWORD help_context;
|
|
ULONG line;
|
|
LONG column;
|
|
|
|
BOOLEAN info_filled;
|
|
BOOLEAN text_filled;
|
|
BOOLEAN pos_filled;
|
|
} ScriptError;
|
|
|
|
struct ScriptHost {
|
|
IActiveScriptSite IActiveScriptSite_iface;
|
|
IActiveScriptSiteWindow IActiveScriptSiteWindow_iface;
|
|
IServiceProvider IServiceProvider_iface;
|
|
LONG ref;
|
|
|
|
IActiveScript *script;
|
|
IActiveScriptParse *parse;
|
|
ScriptError *error;
|
|
SCRIPTSTATE script_state;
|
|
CLSID clsid;
|
|
|
|
unsigned int module_count;
|
|
struct list named_items;
|
|
};
|
|
|
|
struct ScriptControl {
|
|
IScriptControl IScriptControl_iface;
|
|
IPersistStreamInit IPersistStreamInit_iface;
|
|
IOleObject IOleObject_iface;
|
|
IOleControl IOleControl_iface;
|
|
IQuickActivate IQuickActivate_iface;
|
|
IViewObjectEx IViewObjectEx_iface;
|
|
IPointerInactive IPointerInactive_iface;
|
|
IConnectionPointContainer IConnectionPointContainer_iface;
|
|
LONG ref;
|
|
IOleClientSite *site;
|
|
SIZEL extent;
|
|
LONG timeout;
|
|
VARIANT_BOOL allow_ui;
|
|
VARIANT_BOOL use_safe_subset;
|
|
|
|
/* connection points */
|
|
ConnectionPoint *cp_list;
|
|
ConnectionPoint cp_scsource;
|
|
ConnectionPoint cp_propnotif;
|
|
|
|
/* IViewObject sink */
|
|
IAdviseSink *view_sink;
|
|
DWORD view_sink_flags;
|
|
|
|
/* modules */
|
|
ScriptModule **modules;
|
|
IScriptModuleCollection IScriptModuleCollection_iface;
|
|
|
|
ScriptHost *host;
|
|
ScriptError *error;
|
|
};
|
|
|
|
static HINSTANCE msscript_instance;
|
|
|
|
typedef enum tid_t {
|
|
IScriptControl_tid,
|
|
IScriptError_tid,
|
|
IScriptModuleCollection_tid,
|
|
IScriptModule_tid,
|
|
IScriptProcedureCollection_tid,
|
|
IScriptProcedure_tid,
|
|
LAST_tid
|
|
} tid_t;
|
|
|
|
static ITypeLib *typelib;
|
|
static ITypeInfo *typeinfos[LAST_tid];
|
|
|
|
static REFIID tid_ids[] = {
|
|
&IID_IScriptControl,
|
|
&IID_IScriptError,
|
|
&IID_IScriptModuleCollection,
|
|
&IID_IScriptModule,
|
|
&IID_IScriptProcedureCollection,
|
|
&IID_IScriptProcedure
|
|
};
|
|
|
|
static HRESULT load_typelib(void)
|
|
{
|
|
HRESULT hres;
|
|
ITypeLib *tl;
|
|
|
|
hres = LoadRegTypeLib(&LIBID_MSScriptControl, 1, 0, LOCALE_SYSTEM_DEFAULT, &tl);
|
|
if(FAILED(hres)) {
|
|
ERR("LoadRegTypeLib failed: %08x\n", hres);
|
|
return hres;
|
|
}
|
|
|
|
if(InterlockedCompareExchangePointer((void**)&typelib, tl, NULL))
|
|
ITypeLib_Release(tl);
|
|
return hres;
|
|
}
|
|
|
|
static HRESULT get_typeinfo(tid_t tid, ITypeInfo **typeinfo)
|
|
{
|
|
HRESULT hres;
|
|
|
|
if (!typelib)
|
|
hres = load_typelib();
|
|
if (!typelib)
|
|
return hres;
|
|
|
|
if(!typeinfos[tid]) {
|
|
ITypeInfo *ti;
|
|
|
|
hres = ITypeLib_GetTypeInfoOfGuid(typelib, tid_ids[tid], &ti);
|
|
if(FAILED(hres)) {
|
|
ERR("GetTypeInfoOfGuid(%s) failed: %08x\n", debugstr_guid(tid_ids[tid]), hres);
|
|
return hres;
|
|
}
|
|
|
|
if(InterlockedCompareExchangePointer((void**)(typeinfos+tid), ti, NULL))
|
|
ITypeInfo_Release(ti);
|
|
}
|
|
|
|
*typeinfo = typeinfos[tid];
|
|
ITypeInfo_AddRef(typeinfos[tid]);
|
|
return S_OK;
|
|
}
|
|
|
|
static void release_typelib(void)
|
|
{
|
|
unsigned i;
|
|
|
|
if(!typelib)
|
|
return;
|
|
|
|
for(i = 0; i < ARRAY_SIZE(typeinfos); i++)
|
|
if(typeinfos[i])
|
|
ITypeInfo_Release(typeinfos[i]);
|
|
|
|
ITypeLib_Release(typelib);
|
|
}
|
|
|
|
static inline BOOL is_power_of_2(unsigned x)
|
|
{
|
|
return !(x & (x - 1));
|
|
}
|
|
|
|
static void clear_named_items(ScriptHost *host)
|
|
{
|
|
struct named_item *item, *item1;
|
|
LIST_FOR_EACH_ENTRY_SAFE(item, item1, &host->named_items, struct named_item, entry) {
|
|
list_remove(&item->entry);
|
|
SysFreeString(item->name);
|
|
IDispatch_Release(item->disp);
|
|
heap_free(item);
|
|
}
|
|
}
|
|
|
|
static struct named_item *host_get_named_item(ScriptHost *host, const WCHAR *nameW)
|
|
{
|
|
struct named_item *item;
|
|
|
|
LIST_FOR_EACH_ENTRY(item, &host->named_items, struct named_item, entry) {
|
|
if (!wcscmp(item->name, nameW))
|
|
return item;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static HRESULT get_script_dispatch(ScriptModule *module, IDispatch **disp)
|
|
{
|
|
if (!module->script_dispatch)
|
|
{
|
|
HRESULT hr = IActiveScript_GetScriptDispatch(module->host->script,
|
|
module->name, &module->script_dispatch);
|
|
if (FAILED(hr)) return hr;
|
|
}
|
|
*disp = module->script_dispatch;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT get_script_typeinfo(ScriptModule *module, ITypeInfo **typeinfo)
|
|
{
|
|
IDispatch *disp;
|
|
HRESULT hr;
|
|
|
|
if (!module->script_typeinfo)
|
|
{
|
|
hr = get_script_dispatch(module, &disp);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
hr = IDispatch_GetTypeInfo(disp, 0, LOCALE_USER_DEFAULT, &module->script_typeinfo);
|
|
if (FAILED(hr)) return hr;
|
|
}
|
|
*typeinfo = module->script_typeinfo;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT get_script_typecomp(ScriptModule *module, ITypeInfo *typeinfo, ITypeComp **typecomp)
|
|
{
|
|
HRESULT hr;
|
|
|
|
if (!module->script_typecomp)
|
|
{
|
|
hr = ITypeInfo_QueryInterface(typeinfo, &IID_ITypeComp, (void**)&module->script_typecomp);
|
|
if (FAILED(hr)) return hr;
|
|
}
|
|
*typecomp = module->script_typecomp;
|
|
return S_OK;
|
|
}
|
|
|
|
static void uncache_module_objects(ScriptModule *module)
|
|
{
|
|
if (module->script_dispatch)
|
|
{
|
|
IDispatch_Release(module->script_dispatch);
|
|
module->script_dispatch = NULL;
|
|
}
|
|
if (module->script_typeinfo)
|
|
{
|
|
ITypeInfo_Release(module->script_typeinfo);
|
|
module->script_typeinfo = NULL;
|
|
}
|
|
if (module->script_typecomp)
|
|
{
|
|
ITypeComp_Release(module->script_typecomp);
|
|
module->script_typecomp = NULL;
|
|
}
|
|
if (module->procedures)
|
|
module->procedures->count = -1;
|
|
}
|
|
|
|
static HRESULT set_script_state(ScriptHost *host, SCRIPTSTATE state)
|
|
{
|
|
HRESULT hr;
|
|
|
|
hr = IActiveScript_SetScriptState(host->script, state);
|
|
if (SUCCEEDED(hr))
|
|
host->script_state = state;
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT start_script(ScriptHost *host)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if (host->script_state != SCRIPTSTATE_STARTED)
|
|
hr = set_script_state(host, SCRIPTSTATE_STARTED);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT add_script_object(ScriptHost *host, BSTR name, IDispatch *object, DWORD flags)
|
|
{
|
|
struct named_item *item;
|
|
HRESULT hr;
|
|
|
|
if (host_get_named_item(host, name))
|
|
return E_INVALIDARG;
|
|
|
|
item = heap_alloc(sizeof(*item));
|
|
if (!item)
|
|
return E_OUTOFMEMORY;
|
|
|
|
item->name = SysAllocString(name);
|
|
if (!item->name)
|
|
{
|
|
heap_free(item);
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
IDispatch_AddRef(item->disp = object);
|
|
list_add_tail(&host->named_items, &item->entry);
|
|
|
|
hr = IActiveScript_AddNamedItem(host->script, name, flags);
|
|
if (FAILED(hr))
|
|
{
|
|
list_remove(&item->entry);
|
|
IDispatch_Release(item->disp);
|
|
SysFreeString(item->name);
|
|
heap_free(item);
|
|
return hr;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT parse_script_text(ScriptModule *module, BSTR script_text, DWORD flag, VARIANT *res)
|
|
{
|
|
EXCEPINFO excepinfo;
|
|
HRESULT hr;
|
|
|
|
hr = start_script(module->host);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
uncache_module_objects(module);
|
|
|
|
hr = IActiveScriptParse_ParseScriptText(module->host->parse, script_text, module->name,
|
|
NULL, NULL, 0, 1, flag, res, &excepinfo);
|
|
/* FIXME: more error handling */
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT run_procedure(ScriptModule *module, BSTR procedure_name, SAFEARRAY *args, VARIANT *res)
|
|
{
|
|
IDispatchEx *dispex;
|
|
IDispatch *disp;
|
|
DISPPARAMS dp;
|
|
DISPID dispid;
|
|
HRESULT hr;
|
|
UINT i;
|
|
|
|
hr = start_script(module->host);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
hr = get_script_dispatch(module, &disp);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
hr = IDispatch_GetIDsOfNames(disp, &IID_NULL, &procedure_name, 1, LOCALE_USER_DEFAULT, &dispid);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
dp.cArgs = args->rgsabound[0].cElements;
|
|
dp.rgdispidNamedArgs = NULL;
|
|
dp.cNamedArgs = 0;
|
|
dp.rgvarg = heap_alloc(dp.cArgs * sizeof(*dp.rgvarg));
|
|
if (!dp.rgvarg) return E_OUTOFMEMORY;
|
|
|
|
hr = SafeArrayLock(args);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
/* The DISPPARAMS are stored in reverse order */
|
|
for (i = 0; i < dp.cArgs; i++)
|
|
dp.rgvarg[i] = *(VARIANT*)((char*)(args->pvData) + (dp.cArgs - i - 1) * args->cbElements);
|
|
SafeArrayUnlock(args);
|
|
|
|
hr = IDispatch_QueryInterface(disp, &IID_IDispatchEx, (void**)&dispex);
|
|
if (FAILED(hr))
|
|
{
|
|
hr = IDispatch_Invoke(disp, dispid, &IID_NULL, LOCALE_USER_DEFAULT,
|
|
DISPATCH_METHOD, &dp, res, NULL, NULL);
|
|
}
|
|
else
|
|
{
|
|
hr = IDispatchEx_InvokeEx(dispex, dispid, LOCALE_USER_DEFAULT,
|
|
DISPATCH_METHOD, &dp, res, NULL, NULL);
|
|
IDispatchEx_Release(dispex);
|
|
}
|
|
}
|
|
heap_free(dp.rgvarg);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static inline ScriptControl *impl_from_IScriptControl(IScriptControl *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptControl, IScriptControl_iface);
|
|
}
|
|
|
|
static inline ScriptControl *impl_from_IOleObject(IOleObject *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptControl, IOleObject_iface);
|
|
}
|
|
|
|
static inline ScriptControl *impl_from_IPersistStreamInit(IPersistStreamInit *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptControl, IPersistStreamInit_iface);
|
|
}
|
|
|
|
static inline ScriptControl *impl_from_IOleControl(IOleControl *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptControl, IOleControl_iface);
|
|
}
|
|
|
|
static inline ScriptControl *impl_from_IQuickActivate(IQuickActivate *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptControl, IQuickActivate_iface);
|
|
}
|
|
|
|
static inline ScriptControl *impl_from_IViewObjectEx(IViewObjectEx *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptControl, IViewObjectEx_iface);
|
|
}
|
|
|
|
static inline ScriptControl *impl_from_IPointerInactive(IPointerInactive *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptControl, IPointerInactive_iface);
|
|
}
|
|
|
|
static inline ScriptControl *impl_from_IConnectionPointContainer(IConnectionPointContainer *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptControl, IConnectionPointContainer_iface);
|
|
}
|
|
|
|
static inline ScriptProcedure *impl_from_IScriptProcedure(IScriptProcedure *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptProcedure, IScriptProcedure_iface);
|
|
}
|
|
|
|
static inline ScriptProcedureCollection *impl_from_IScriptProcedureCollection(IScriptProcedureCollection *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptProcedureCollection, IScriptProcedureCollection_iface);
|
|
}
|
|
|
|
static inline ScriptControl *impl_from_IScriptModuleCollection(IScriptModuleCollection *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptControl, IScriptModuleCollection_iface);
|
|
}
|
|
|
|
static inline ScriptModule *impl_from_IScriptModule(IScriptModule *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptModule, IScriptModule_iface);
|
|
}
|
|
|
|
static inline ScriptError *impl_from_IScriptError(IScriptError *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptError, IScriptError_iface);
|
|
}
|
|
|
|
static inline ConnectionPoint *impl_from_IConnectionPoint(IConnectionPoint *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ConnectionPoint, IConnectionPoint_iface);
|
|
}
|
|
|
|
static inline ScriptHost *impl_from_IActiveScriptSite(IActiveScriptSite *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptHost, IActiveScriptSite_iface);
|
|
}
|
|
|
|
static inline ScriptHost *impl_from_IActiveScriptSiteWindow(IActiveScriptSiteWindow *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptHost, IActiveScriptSiteWindow_iface);
|
|
}
|
|
|
|
static inline ScriptHost *impl_from_IServiceProvider(IServiceProvider *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, ScriptHost, IServiceProvider_iface);
|
|
}
|
|
|
|
static inline struct module_enum *module_enum_from_IEnumVARIANT(IEnumVARIANT *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct module_enum, IEnumVARIANT_iface);
|
|
}
|
|
|
|
static inline struct procedure_enum *procedure_enum_from_IEnumVARIANT(IEnumVARIANT *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct procedure_enum, IEnumVARIANT_iface);
|
|
}
|
|
|
|
/* IActiveScriptSite */
|
|
static HRESULT WINAPI ActiveScriptSite_QueryInterface(IActiveScriptSite *iface, REFIID riid, void **ppv)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSite(iface);
|
|
|
|
if(IsEqualGUID(&IID_IUnknown, riid)) {
|
|
TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
|
|
*ppv = &This->IActiveScriptSite_iface;
|
|
}else if(IsEqualGUID(&IID_IActiveScriptSite, riid)) {
|
|
TRACE("(%p)->(IID_IActiveScriptSite %p)\n", This, ppv);
|
|
*ppv = &This->IActiveScriptSite_iface;
|
|
}else if(IsEqualGUID(&IID_IActiveScriptSiteWindow, riid)) {
|
|
TRACE("(%p)->(IID_IActiveScriptSiteWindow %p)\n", This, ppv);
|
|
*ppv = &This->IActiveScriptSiteWindow_iface;
|
|
}else if(IsEqualGUID(&IID_IServiceProvider, riid)) {
|
|
TRACE("(%p)->(IID_IServiceProvider %p)\n", This, ppv);
|
|
*ppv = &This->IServiceProvider_iface;
|
|
}else {
|
|
FIXME("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI ActiveScriptSite_AddRef(IActiveScriptSite *iface)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSite(iface);
|
|
LONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI ActiveScriptSite_Release(IActiveScriptSite *iface)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSite(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
if(!ref) {
|
|
clear_named_items(This);
|
|
heap_free(This);
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_GetLCID(IActiveScriptSite *iface, LCID *lcid)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSite(iface);
|
|
|
|
TRACE("(%p, %p)\n", This, lcid);
|
|
|
|
*lcid = GetUserDefaultLCID();
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_GetItemInfo(IActiveScriptSite *iface, LPCOLESTR name, DWORD mask,
|
|
IUnknown **unk, ITypeInfo **ti)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSite(iface);
|
|
struct named_item *item;
|
|
|
|
TRACE("(%p, %s, %#x, %p, %p)\n", This, debugstr_w(name), mask, unk, ti);
|
|
|
|
item = host_get_named_item(This, name);
|
|
if (!item)
|
|
return TYPE_E_ELEMENTNOTFOUND;
|
|
|
|
if (mask != SCRIPTINFO_IUNKNOWN) {
|
|
FIXME("mask %#x is not supported\n", mask);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
*unk = (IUnknown*)item->disp;
|
|
IUnknown_AddRef(*unk);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_GetDocVersionString(IActiveScriptSite *iface, BSTR *version)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSite(iface);
|
|
|
|
FIXME("(%p, %p)\n", This, version);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_OnScriptTerminate(IActiveScriptSite *iface, const VARIANT *result,
|
|
const EXCEPINFO *ei)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSite(iface);
|
|
|
|
FIXME("(%p, %s, %p)\n", This, debugstr_variant(result), ei);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_OnStateChange(IActiveScriptSite *iface, SCRIPTSTATE state)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSite(iface);
|
|
|
|
FIXME("(%p, %d)\n", This, state);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_OnScriptError(IActiveScriptSite *iface, IActiveScriptError *script_error)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSite(iface);
|
|
|
|
TRACE("(%p, %p)\n", This, script_error);
|
|
|
|
if (This->error)
|
|
{
|
|
IScriptError_Clear(&This->error->IScriptError_iface);
|
|
IActiveScriptError_AddRef(script_error);
|
|
This->error->object = script_error;
|
|
}
|
|
return S_FALSE;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_OnEnterScript(IActiveScriptSite *iface)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSite(iface);
|
|
|
|
FIXME("(%p)\n", This);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSite_OnLeaveScript(IActiveScriptSite *iface)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSite(iface);
|
|
|
|
FIXME("(%p)\n", This);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IActiveScriptSiteVtbl ActiveScriptSiteVtbl = {
|
|
ActiveScriptSite_QueryInterface,
|
|
ActiveScriptSite_AddRef,
|
|
ActiveScriptSite_Release,
|
|
ActiveScriptSite_GetLCID,
|
|
ActiveScriptSite_GetItemInfo,
|
|
ActiveScriptSite_GetDocVersionString,
|
|
ActiveScriptSite_OnScriptTerminate,
|
|
ActiveScriptSite_OnStateChange,
|
|
ActiveScriptSite_OnScriptError,
|
|
ActiveScriptSite_OnEnterScript,
|
|
ActiveScriptSite_OnLeaveScript
|
|
};
|
|
|
|
/* IActiveScriptSiteWindow */
|
|
static HRESULT WINAPI ActiveScriptSiteWindow_QueryInterface(IActiveScriptSiteWindow *iface, REFIID riid, void **obj)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSiteWindow(iface);
|
|
return IActiveScriptSite_QueryInterface(&This->IActiveScriptSite_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI ActiveScriptSiteWindow_AddRef(IActiveScriptSiteWindow *iface)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSiteWindow(iface);
|
|
return IActiveScriptSite_AddRef(&This->IActiveScriptSite_iface);
|
|
}
|
|
|
|
static ULONG WINAPI ActiveScriptSiteWindow_Release(IActiveScriptSiteWindow *iface)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSiteWindow(iface);
|
|
return IActiveScriptSite_Release(&This->IActiveScriptSite_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSiteWindow_GetWindow(IActiveScriptSiteWindow *iface, HWND *hwnd)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSiteWindow(iface);
|
|
|
|
FIXME("(%p, %p)\n", This, hwnd);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ActiveScriptSiteWindow_EnableModeless(IActiveScriptSiteWindow *iface, BOOL enable)
|
|
{
|
|
ScriptHost *This = impl_from_IActiveScriptSiteWindow(iface);
|
|
|
|
FIXME("(%p, %d)\n", This, enable);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IActiveScriptSiteWindowVtbl ActiveScriptSiteWindowVtbl = {
|
|
ActiveScriptSiteWindow_QueryInterface,
|
|
ActiveScriptSiteWindow_AddRef,
|
|
ActiveScriptSiteWindow_Release,
|
|
ActiveScriptSiteWindow_GetWindow,
|
|
ActiveScriptSiteWindow_EnableModeless
|
|
};
|
|
|
|
/* IServiceProvider */
|
|
static HRESULT WINAPI ServiceProvider_QueryInterface(IServiceProvider *iface, REFIID riid, void **obj)
|
|
{
|
|
ScriptHost *This = impl_from_IServiceProvider(iface);
|
|
return IActiveScriptSite_QueryInterface(&This->IActiveScriptSite_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI ServiceProvider_AddRef(IServiceProvider *iface)
|
|
{
|
|
ScriptHost *This = impl_from_IServiceProvider(iface);
|
|
return IActiveScriptSite_AddRef(&This->IActiveScriptSite_iface);
|
|
}
|
|
|
|
static ULONG WINAPI ServiceProvider_Release(IServiceProvider *iface)
|
|
{
|
|
ScriptHost *This = impl_from_IServiceProvider(iface);
|
|
return IActiveScriptSite_Release(&This->IActiveScriptSite_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI ServiceProvider_QueryService(IServiceProvider *iface, REFGUID service,
|
|
REFIID riid, void **obj)
|
|
{
|
|
ScriptHost *This = impl_from_IServiceProvider(iface);
|
|
|
|
FIXME("(%p)->(%s %s %p)\n", This, debugstr_guid(service), debugstr_guid(riid), obj);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IServiceProviderVtbl ServiceProviderVtbl = {
|
|
ServiceProvider_QueryInterface,
|
|
ServiceProvider_AddRef,
|
|
ServiceProvider_Release,
|
|
ServiceProvider_QueryService
|
|
};
|
|
|
|
static HRESULT WINAPI ScriptProcedure_QueryInterface(IScriptProcedure *iface, REFIID riid, void **ppv)
|
|
{
|
|
ScriptProcedure *This = impl_from_IScriptProcedure(iface);
|
|
|
|
if (IsEqualGUID(&IID_IDispatch, riid) || IsEqualGUID(&IID_IUnknown, riid) ||
|
|
IsEqualGUID(&IID_IScriptProcedure, riid))
|
|
{
|
|
*ppv = &This->IScriptProcedure_iface;
|
|
}
|
|
else
|
|
{
|
|
WARN("unsupported interface: (%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI ScriptProcedure_AddRef(IScriptProcedure *iface)
|
|
{
|
|
ScriptProcedure *This = impl_from_IScriptProcedure(iface);
|
|
LONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI ScriptProcedure_Release(IScriptProcedure *iface)
|
|
{
|
|
ScriptProcedure *This = impl_from_IScriptProcedure(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
if (!ref)
|
|
{
|
|
list_remove(&This->entry);
|
|
SysFreeString(This->name);
|
|
heap_free(This);
|
|
}
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedure_GetTypeInfoCount(IScriptProcedure *iface, UINT *pctinfo)
|
|
{
|
|
ScriptProcedure *This = impl_from_IScriptProcedure(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pctinfo);
|
|
|
|
*pctinfo = 1;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedure_GetTypeInfo(IScriptProcedure *iface, UINT iTInfo,
|
|
LCID lcid, ITypeInfo **ppTInfo)
|
|
{
|
|
ScriptProcedure *This = impl_from_IScriptProcedure(iface);
|
|
|
|
TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
|
|
|
|
return get_typeinfo(IScriptProcedure_tid, ppTInfo);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedure_GetIDsOfNames(IScriptProcedure *iface, REFIID riid,
|
|
LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
|
|
{
|
|
ScriptProcedure *This = impl_from_IScriptProcedure(iface);
|
|
ITypeInfo *typeinfo;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames, lcid, rgDispId);
|
|
|
|
hr = get_typeinfo(IScriptProcedure_tid, &typeinfo);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
|
|
ITypeInfo_Release(typeinfo);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedure_Invoke(IScriptProcedure *iface, DISPID dispIdMember,
|
|
REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult,
|
|
EXCEPINFO *pExcepInfo, UINT *puArgErr)
|
|
{
|
|
ScriptProcedure *This = impl_from_IScriptProcedure(iface);
|
|
ITypeInfo *typeinfo;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
|
|
lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
|
|
hr = get_typeinfo(IScriptProcedure_tid, &typeinfo);
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
hr = ITypeInfo_Invoke(typeinfo, iface, dispIdMember, wFlags,
|
|
pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
ITypeInfo_Release(typeinfo);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedure_get_Name(IScriptProcedure *iface, BSTR *pbstrName)
|
|
{
|
|
ScriptProcedure *This = impl_from_IScriptProcedure(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pbstrName);
|
|
|
|
if (!pbstrName) return E_POINTER;
|
|
|
|
*pbstrName = SysAllocString(This->name);
|
|
return *pbstrName ? S_OK : E_OUTOFMEMORY;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedure_get_NumArgs(IScriptProcedure *iface, LONG *pcArgs)
|
|
{
|
|
ScriptProcedure *This = impl_from_IScriptProcedure(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pcArgs);
|
|
|
|
if (!pcArgs) return E_POINTER;
|
|
|
|
*pcArgs = This->num_args;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedure_get_HasReturnValue(IScriptProcedure *iface, VARIANT_BOOL *pfHasReturnValue)
|
|
{
|
|
ScriptProcedure *This = impl_from_IScriptProcedure(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pfHasReturnValue);
|
|
|
|
if (!pfHasReturnValue) return E_POINTER;
|
|
|
|
*pfHasReturnValue = (This->ret_type == VT_VOID) ? VARIANT_FALSE : VARIANT_TRUE;
|
|
return S_OK;
|
|
}
|
|
|
|
static const IScriptProcedureVtbl ScriptProcedureVtbl = {
|
|
ScriptProcedure_QueryInterface,
|
|
ScriptProcedure_AddRef,
|
|
ScriptProcedure_Release,
|
|
ScriptProcedure_GetTypeInfoCount,
|
|
ScriptProcedure_GetTypeInfo,
|
|
ScriptProcedure_GetIDsOfNames,
|
|
ScriptProcedure_Invoke,
|
|
ScriptProcedure_get_Name,
|
|
ScriptProcedure_get_NumArgs,
|
|
ScriptProcedure_get_HasReturnValue
|
|
};
|
|
|
|
/* This function always releases the FUNCDESC passed in */
|
|
static HRESULT get_script_procedure(ScriptProcedureCollection *procedures, ITypeInfo *typeinfo,
|
|
FUNCDESC *desc, IScriptProcedure **procedure)
|
|
{
|
|
struct list *proc_list;
|
|
ScriptProcedure *proc;
|
|
ULONG hash;
|
|
HRESULT hr;
|
|
BSTR str;
|
|
UINT len;
|
|
|
|
hr = ITypeInfo_GetNames(typeinfo, desc->memid, &str, 1, &len);
|
|
if (FAILED(hr)) goto done;
|
|
|
|
len = SysStringLen(str);
|
|
hash = LHashValOfNameSys(sizeof(void*) == 8 ? SYS_WIN64 : SYS_WIN32, LOCALE_USER_DEFAULT, str);
|
|
proc_list = &procedures->hash_table[hash % ARRAY_SIZE(procedures->hash_table)];
|
|
|
|
/* Try to find it in the hash table */
|
|
LIST_FOR_EACH_ENTRY(proc, proc_list, ScriptProcedure, entry)
|
|
{
|
|
if (proc->hash == hash && SysStringLen(proc->name) == len &&
|
|
!memcmp(proc->name, str, len * sizeof(*str)))
|
|
{
|
|
SysFreeString(str);
|
|
IScriptProcedure_AddRef(&proc->IScriptProcedure_iface);
|
|
*procedure = &proc->IScriptProcedure_iface;
|
|
goto done;
|
|
}
|
|
}
|
|
|
|
if (!(proc = heap_alloc(sizeof(*proc))))
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
SysFreeString(str);
|
|
goto done;
|
|
}
|
|
|
|
proc->IScriptProcedure_iface.lpVtbl = &ScriptProcedureVtbl;
|
|
proc->ref = 1;
|
|
proc->hash = hash;
|
|
proc->name = str;
|
|
proc->num_args = desc->cParams + desc->cParamsOpt;
|
|
proc->ret_type = desc->elemdescFunc.tdesc.vt;
|
|
list_add_tail(proc_list, &proc->entry);
|
|
|
|
*procedure = &proc->IScriptProcedure_iface;
|
|
|
|
done:
|
|
ITypeInfo_ReleaseFuncDesc(typeinfo, desc);
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI procedure_enum_QueryInterface(IEnumVARIANT *iface, REFIID riid, void **ppv)
|
|
{
|
|
struct procedure_enum *This = procedure_enum_from_IEnumVARIANT(iface);
|
|
|
|
if (IsEqualGUID(&IID_IUnknown, riid) || IsEqualGUID(&IID_IEnumVARIANT, riid))
|
|
{
|
|
*ppv = &This->IEnumVARIANT_iface;
|
|
}
|
|
else
|
|
{
|
|
WARN("unsupported interface: (%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI procedure_enum_AddRef(IEnumVARIANT *iface)
|
|
{
|
|
struct procedure_enum *This = procedure_enum_from_IEnumVARIANT(iface);
|
|
LONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI procedure_enum_Release(IEnumVARIANT *iface)
|
|
{
|
|
struct procedure_enum *This = procedure_enum_from_IEnumVARIANT(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
if (!ref)
|
|
{
|
|
IScriptProcedureCollection_Release(&This->procedures->IScriptProcedureCollection_iface);
|
|
heap_free(This);
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI procedure_enum_Next(IEnumVARIANT *iface, ULONG celt, VARIANT *rgVar, ULONG *pCeltFetched)
|
|
{
|
|
struct procedure_enum *This = procedure_enum_from_IEnumVARIANT(iface);
|
|
FUNCDESC *desc;
|
|
ITypeInfo *ti;
|
|
UINT i, num;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%u %p %p)\n", This, celt, rgVar, pCeltFetched);
|
|
|
|
if (!rgVar) return E_POINTER;
|
|
if (!This->procedures->module->host) return E_FAIL;
|
|
|
|
hr = start_script(This->procedures->module->host);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
hr = get_script_typeinfo(This->procedures->module, &ti);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
num = min(celt, This->count - This->pos);
|
|
for (i = 0; i < num; i++)
|
|
{
|
|
hr = ITypeInfo_GetFuncDesc(ti, This->pos + i, &desc);
|
|
if (FAILED(hr)) break;
|
|
|
|
hr = get_script_procedure(This->procedures, ti, desc, (IScriptProcedure**)&V_DISPATCH(rgVar + i));
|
|
if (FAILED(hr)) break;
|
|
|
|
V_VT(rgVar + i) = VT_DISPATCH;
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
while (i--)
|
|
VariantClear(rgVar + i);
|
|
if (pCeltFetched) *pCeltFetched = 0;
|
|
return hr;
|
|
}
|
|
|
|
This->pos += i;
|
|
|
|
if (pCeltFetched) *pCeltFetched = i;
|
|
return i == celt ? S_OK : S_FALSE;
|
|
}
|
|
|
|
static HRESULT WINAPI procedure_enum_Skip(IEnumVARIANT *iface, ULONG celt)
|
|
{
|
|
struct procedure_enum *This = procedure_enum_from_IEnumVARIANT(iface);
|
|
|
|
TRACE("(%p)->(%u)\n", This, celt);
|
|
|
|
if (This->count - This->pos < celt)
|
|
{
|
|
This->pos = This->count;
|
|
return S_FALSE;
|
|
}
|
|
This->pos += celt;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI procedure_enum_Reset(IEnumVARIANT *iface)
|
|
{
|
|
struct procedure_enum *This = procedure_enum_from_IEnumVARIANT(iface);
|
|
|
|
TRACE("(%p)\n", This);
|
|
|
|
This->pos = 0;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI procedure_enum_Clone(IEnumVARIANT *iface, IEnumVARIANT **ppEnum)
|
|
{
|
|
struct procedure_enum *This = procedure_enum_from_IEnumVARIANT(iface);
|
|
struct procedure_enum *clone;
|
|
|
|
TRACE("(%p)->(%p)\n", This, ppEnum);
|
|
|
|
if (!ppEnum) return E_POINTER;
|
|
|
|
if (!(clone = heap_alloc(sizeof(*clone))))
|
|
return E_OUTOFMEMORY;
|
|
|
|
*clone = *This;
|
|
clone->ref = 1;
|
|
IScriptProcedureCollection_AddRef(&This->procedures->IScriptProcedureCollection_iface);
|
|
|
|
*ppEnum = &clone->IEnumVARIANT_iface;
|
|
return S_OK;
|
|
}
|
|
|
|
static const IEnumVARIANTVtbl procedure_enum_vtbl = {
|
|
procedure_enum_QueryInterface,
|
|
procedure_enum_AddRef,
|
|
procedure_enum_Release,
|
|
procedure_enum_Next,
|
|
procedure_enum_Skip,
|
|
procedure_enum_Reset,
|
|
procedure_enum_Clone
|
|
};
|
|
|
|
static HRESULT WINAPI ScriptProcedureCollection_QueryInterface(IScriptProcedureCollection *iface, REFIID riid, void **ppv)
|
|
{
|
|
ScriptProcedureCollection *This = impl_from_IScriptProcedureCollection(iface);
|
|
|
|
if (IsEqualGUID(&IID_IDispatch, riid) || IsEqualGUID(&IID_IUnknown, riid) ||
|
|
IsEqualGUID(&IID_IScriptProcedureCollection, riid))
|
|
{
|
|
*ppv = &This->IScriptProcedureCollection_iface;
|
|
}
|
|
else
|
|
{
|
|
WARN("unsupported interface: (%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI ScriptProcedureCollection_AddRef(IScriptProcedureCollection *iface)
|
|
{
|
|
ScriptProcedureCollection *This = impl_from_IScriptProcedureCollection(iface);
|
|
LONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI ScriptProcedureCollection_Release(IScriptProcedureCollection *iface)
|
|
{
|
|
ScriptProcedureCollection *This = impl_from_IScriptProcedureCollection(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
UINT i;
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
if (!ref)
|
|
{
|
|
/* Unlink any dangling items from the hash table */
|
|
for (i = 0; i < ARRAY_SIZE(This->hash_table); i++)
|
|
list_remove(&This->hash_table[i]);
|
|
|
|
This->module->procedures = NULL;
|
|
IScriptModule_Release(&This->module->IScriptModule_iface);
|
|
heap_free(This);
|
|
}
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedureCollection_GetTypeInfoCount(IScriptProcedureCollection *iface, UINT *pctinfo)
|
|
{
|
|
ScriptProcedureCollection *This = impl_from_IScriptProcedureCollection(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pctinfo);
|
|
|
|
*pctinfo = 1;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedureCollection_GetTypeInfo(IScriptProcedureCollection *iface, UINT iTInfo,
|
|
LCID lcid, ITypeInfo **ppTInfo)
|
|
{
|
|
ScriptProcedureCollection *This = impl_from_IScriptProcedureCollection(iface);
|
|
|
|
TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
|
|
|
|
return get_typeinfo(IScriptProcedureCollection_tid, ppTInfo);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedureCollection_GetIDsOfNames(IScriptProcedureCollection *iface, REFIID riid,
|
|
LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
|
|
{
|
|
ScriptProcedureCollection *This = impl_from_IScriptProcedureCollection(iface);
|
|
ITypeInfo *typeinfo;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames, lcid, rgDispId);
|
|
|
|
hr = get_typeinfo(IScriptProcedureCollection_tid, &typeinfo);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
|
|
ITypeInfo_Release(typeinfo);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedureCollection_Invoke(IScriptProcedureCollection *iface, DISPID dispIdMember,
|
|
REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult,
|
|
EXCEPINFO *pExcepInfo, UINT *puArgErr)
|
|
{
|
|
ScriptProcedureCollection *This = impl_from_IScriptProcedureCollection(iface);
|
|
ITypeInfo *typeinfo;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
|
|
lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
|
|
hr = get_typeinfo(IScriptProcedureCollection_tid, &typeinfo);
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
hr = ITypeInfo_Invoke(typeinfo, iface, dispIdMember, wFlags,
|
|
pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
ITypeInfo_Release(typeinfo);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedureCollection_get__NewEnum(IScriptProcedureCollection *iface, IUnknown **ppenumProcedures)
|
|
{
|
|
ScriptProcedureCollection *This = impl_from_IScriptProcedureCollection(iface);
|
|
struct procedure_enum *proc_enum;
|
|
TYPEATTR *attr;
|
|
ITypeInfo *ti;
|
|
UINT count;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%p)\n", This, ppenumProcedures);
|
|
|
|
if (!ppenumProcedures) return E_POINTER;
|
|
if (!This->module->host) return E_FAIL;
|
|
|
|
hr = start_script(This->module->host);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
hr = get_script_typeinfo(This->module, &ti);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
hr = ITypeInfo_GetTypeAttr(ti, &attr);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
count = attr->cFuncs;
|
|
ITypeInfo_ReleaseTypeAttr(ti, attr);
|
|
|
|
if (!(proc_enum = heap_alloc(sizeof(*proc_enum))))
|
|
return E_OUTOFMEMORY;
|
|
|
|
proc_enum->IEnumVARIANT_iface.lpVtbl = &procedure_enum_vtbl;
|
|
proc_enum->ref = 1;
|
|
proc_enum->pos = 0;
|
|
proc_enum->count = count;
|
|
proc_enum->procedures = This;
|
|
IScriptProcedureCollection_AddRef(&This->IScriptProcedureCollection_iface);
|
|
|
|
*ppenumProcedures = (IUnknown*)&proc_enum->IEnumVARIANT_iface;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedureCollection_get_Item(IScriptProcedureCollection *iface, VARIANT index,
|
|
IScriptProcedure **ppdispProcedure)
|
|
{
|
|
ScriptProcedureCollection *This = impl_from_IScriptProcedureCollection(iface);
|
|
ITypeInfo *typeinfo;
|
|
FUNCDESC *desc;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%s %p)\n", This, wine_dbgstr_variant(&index), ppdispProcedure);
|
|
|
|
if (!ppdispProcedure) return E_POINTER;
|
|
if (!This->module->host) return E_FAIL;
|
|
|
|
hr = start_script(This->module->host);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
hr = get_script_typeinfo(This->module, &typeinfo);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
if (V_VT(&index) == VT_BSTR)
|
|
{
|
|
struct list *proc_list;
|
|
ScriptProcedure *proc;
|
|
ITypeComp *comp;
|
|
BINDPTR bindptr;
|
|
DESCKIND kind;
|
|
ULONG hash;
|
|
UINT len;
|
|
|
|
len = SysStringLen(V_BSTR(&index));
|
|
hash = LHashValOfNameSys(sizeof(void*) == 8 ? SYS_WIN64 : SYS_WIN32, LOCALE_USER_DEFAULT, V_BSTR(&index));
|
|
proc_list = &This->hash_table[hash % ARRAY_SIZE(This->hash_table)];
|
|
|
|
/* Try to find it in the hash table */
|
|
LIST_FOR_EACH_ENTRY(proc, proc_list, ScriptProcedure, entry)
|
|
{
|
|
if (proc->hash == hash && SysStringLen(proc->name) == len &&
|
|
!memcmp(proc->name, V_BSTR(&index), len * sizeof(WCHAR)))
|
|
{
|
|
IScriptProcedure_AddRef(&proc->IScriptProcedure_iface);
|
|
*ppdispProcedure = &proc->IScriptProcedure_iface;
|
|
return S_OK;
|
|
}
|
|
}
|
|
|
|
hr = get_script_typecomp(This->module, typeinfo, &comp);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
hr = ITypeComp_Bind(comp, V_BSTR(&index), hash, INVOKE_FUNC, &typeinfo, &kind, &bindptr);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
switch (kind)
|
|
{
|
|
case DESCKIND_FUNCDESC:
|
|
hr = get_script_procedure(This, typeinfo, bindptr.lpfuncdesc, ppdispProcedure);
|
|
ITypeInfo_Release(typeinfo);
|
|
return hr;
|
|
case DESCKIND_IMPLICITAPPOBJ:
|
|
case DESCKIND_VARDESC:
|
|
ITypeInfo_ReleaseVarDesc(typeinfo, bindptr.lpvardesc);
|
|
ITypeInfo_Release(typeinfo);
|
|
break;
|
|
case DESCKIND_TYPECOMP:
|
|
ITypeComp_Release(bindptr.lptcomp);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return CTL_E_ILLEGALFUNCTIONCALL;
|
|
}
|
|
|
|
hr = VariantChangeType(&index, &index, 0, VT_INT);
|
|
if (FAILED(hr)) return hr;
|
|
if (V_INT(&index) <= 0) return 0x800a0009;
|
|
|
|
hr = ITypeInfo_GetFuncDesc(typeinfo, V_INT(&index) - 1, &desc);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
return get_script_procedure(This, typeinfo, desc, ppdispProcedure);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptProcedureCollection_get_Count(IScriptProcedureCollection *iface, LONG *plCount)
|
|
{
|
|
ScriptProcedureCollection *This = impl_from_IScriptProcedureCollection(iface);
|
|
TYPEATTR *attr;
|
|
ITypeInfo *ti;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%p)\n", This, plCount);
|
|
|
|
if (!plCount) return E_POINTER;
|
|
if (!This->module->host) return E_FAIL;
|
|
|
|
hr = start_script(This->module->host);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
if (This->count == -1)
|
|
{
|
|
hr = get_script_typeinfo(This->module, &ti);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
hr = ITypeInfo_GetTypeAttr(ti, &attr);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
This->count = attr->cFuncs;
|
|
ITypeInfo_ReleaseTypeAttr(ti, attr);
|
|
}
|
|
|
|
*plCount = This->count;
|
|
return S_OK;
|
|
}
|
|
|
|
static const IScriptProcedureCollectionVtbl ScriptProcedureCollectionVtbl = {
|
|
ScriptProcedureCollection_QueryInterface,
|
|
ScriptProcedureCollection_AddRef,
|
|
ScriptProcedureCollection_Release,
|
|
ScriptProcedureCollection_GetTypeInfoCount,
|
|
ScriptProcedureCollection_GetTypeInfo,
|
|
ScriptProcedureCollection_GetIDsOfNames,
|
|
ScriptProcedureCollection_Invoke,
|
|
ScriptProcedureCollection_get__NewEnum,
|
|
ScriptProcedureCollection_get_Item,
|
|
ScriptProcedureCollection_get_Count
|
|
};
|
|
|
|
static void detach_script_host(ScriptHost *host)
|
|
{
|
|
if (--host->module_count)
|
|
return;
|
|
|
|
if (host->script) {
|
|
IActiveScript_Close(host->script);
|
|
IActiveScript_Release(host->script);
|
|
}
|
|
|
|
if (host->parse)
|
|
IActiveScriptParse_Release(host->parse);
|
|
|
|
if (host->error)
|
|
IScriptError_Release(&host->error->IScriptError_iface);
|
|
|
|
host->parse = NULL;
|
|
host->error = NULL;
|
|
host->script = NULL;
|
|
}
|
|
|
|
static void detach_module(ScriptModule *module)
|
|
{
|
|
ScriptHost *host = module->host;
|
|
|
|
if (host) {
|
|
module->host = NULL;
|
|
detach_script_host(host);
|
|
IActiveScriptSite_Release(&host->IActiveScriptSite_iface);
|
|
}
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModule_QueryInterface(IScriptModule *iface, REFIID riid, void **ppv)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
|
|
if (IsEqualGUID(&IID_IDispatch, riid) || IsEqualGUID(&IID_IUnknown, riid) ||
|
|
IsEqualGUID(&IID_IScriptModule, riid))
|
|
{
|
|
*ppv = &This->IScriptModule_iface;
|
|
}
|
|
else
|
|
{
|
|
WARN("unsupported interface: (%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI ScriptModule_AddRef(IScriptModule *iface)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
LONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI ScriptModule_Release(IScriptModule *iface)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
if (!ref)
|
|
{
|
|
detach_module(This);
|
|
SysFreeString(This->name);
|
|
uncache_module_objects(This);
|
|
heap_free(This);
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModule_GetTypeInfoCount(IScriptModule *iface, UINT *pctinfo)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pctinfo);
|
|
|
|
*pctinfo = 1;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModule_GetTypeInfo(IScriptModule *iface, UINT iTInfo,
|
|
LCID lcid, ITypeInfo **ppTInfo)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
|
|
TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
|
|
|
|
return get_typeinfo(IScriptModule_tid, ppTInfo);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModule_GetIDsOfNames(IScriptModule *iface, REFIID riid,
|
|
LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
ITypeInfo *typeinfo;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames, lcid, rgDispId);
|
|
|
|
hr = get_typeinfo(IScriptModule_tid, &typeinfo);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
|
|
ITypeInfo_Release(typeinfo);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModule_Invoke(IScriptModule *iface, DISPID dispIdMember,
|
|
REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult,
|
|
EXCEPINFO *pExcepInfo, UINT *puArgErr)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
ITypeInfo *typeinfo;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
|
|
lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
|
|
hr = get_typeinfo(IScriptModule_tid, &typeinfo);
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
hr = ITypeInfo_Invoke(typeinfo, iface, dispIdMember, wFlags,
|
|
pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
ITypeInfo_Release(typeinfo);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModule_get_Name(IScriptModule *iface, BSTR *pbstrName)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pbstrName);
|
|
|
|
if (!pbstrName) return E_POINTER;
|
|
if (!This->host) return E_FAIL;
|
|
|
|
*pbstrName = SysAllocString(This->name ? This->name : L"Global");
|
|
return *pbstrName ? S_OK : E_OUTOFMEMORY;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModule_get_CodeObject(IScriptModule *iface, IDispatch **ppdispObject)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%p)\n", This, ppdispObject);
|
|
|
|
if (!This->host) return E_FAIL;
|
|
|
|
hr = start_script(This->host);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
hr = get_script_dispatch(This, ppdispObject);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
IDispatch_AddRef(*ppdispObject);
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModule_get_Procedures(IScriptModule *iface, IScriptProcedureCollection **ppdispProcedures)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, ppdispProcedures);
|
|
|
|
if (!This->host)
|
|
return E_FAIL;
|
|
|
|
if (This->procedures)
|
|
IScriptProcedureCollection_AddRef(&This->procedures->IScriptProcedureCollection_iface);
|
|
else
|
|
{
|
|
ScriptProcedureCollection *procs;
|
|
UINT i;
|
|
|
|
if (!(procs = heap_alloc(sizeof(*procs))))
|
|
return E_OUTOFMEMORY;
|
|
|
|
procs->IScriptProcedureCollection_iface.lpVtbl = &ScriptProcedureCollectionVtbl;
|
|
procs->ref = 1;
|
|
procs->count = -1;
|
|
procs->module = This;
|
|
for (i = 0; i < ARRAY_SIZE(procs->hash_table); i++)
|
|
list_init(&procs->hash_table[i]);
|
|
|
|
This->procedures = procs;
|
|
IScriptModule_AddRef(&This->IScriptModule_iface);
|
|
}
|
|
|
|
*ppdispProcedures = &This->procedures->IScriptProcedureCollection_iface;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModule_AddCode(IScriptModule *iface, BSTR code)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
|
|
TRACE("(%p)->(%s)\n", This, debugstr_w(code));
|
|
|
|
if (!This->host)
|
|
return E_FAIL;
|
|
|
|
return parse_script_text(This, code, SCRIPTTEXT_ISVISIBLE, NULL);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModule_Eval(IScriptModule *iface, BSTR expression, VARIANT *res)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
|
|
TRACE("(%p)->(%s, %p)\n", This, debugstr_w(expression), res);
|
|
|
|
if (!res)
|
|
return E_POINTER;
|
|
V_VT(res) = VT_EMPTY;
|
|
if (!This->host)
|
|
return E_FAIL;
|
|
|
|
return parse_script_text(This, expression, SCRIPTTEXT_ISEXPRESSION, res);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModule_ExecuteStatement(IScriptModule *iface, BSTR statement)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
|
|
TRACE("(%p)->(%s)\n", This, debugstr_w(statement));
|
|
|
|
if (!This->host)
|
|
return E_FAIL;
|
|
|
|
return parse_script_text(This, statement, 0, NULL);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModule_Run(IScriptModule *iface, BSTR procedure_name, SAFEARRAY **parameters, VARIANT *res)
|
|
{
|
|
ScriptModule *This = impl_from_IScriptModule(iface);
|
|
SAFEARRAY *sa;
|
|
|
|
TRACE("(%p)->(%s %p %p)\n", This, debugstr_w(procedure_name), parameters, res);
|
|
|
|
if (!parameters || !res) return E_POINTER;
|
|
if (!(sa = *parameters)) return E_POINTER;
|
|
|
|
V_VT(res) = VT_EMPTY;
|
|
if (sa->cDims == 0) return DISP_E_BADINDEX;
|
|
if (!(sa->fFeatures & FADF_VARIANT)) return DISP_E_BADVARTYPE;
|
|
if (!This->host) return E_FAIL;
|
|
|
|
return run_procedure(This, procedure_name, sa, res);
|
|
}
|
|
|
|
static const IScriptModuleVtbl ScriptModuleVtbl = {
|
|
ScriptModule_QueryInterface,
|
|
ScriptModule_AddRef,
|
|
ScriptModule_Release,
|
|
ScriptModule_GetTypeInfoCount,
|
|
ScriptModule_GetTypeInfo,
|
|
ScriptModule_GetIDsOfNames,
|
|
ScriptModule_Invoke,
|
|
ScriptModule_get_Name,
|
|
ScriptModule_get_CodeObject,
|
|
ScriptModule_get_Procedures,
|
|
ScriptModule_AddCode,
|
|
ScriptModule_Eval,
|
|
ScriptModule_ExecuteStatement,
|
|
ScriptModule_Run
|
|
};
|
|
|
|
static HRESULT WINAPI module_enum_QueryInterface(IEnumVARIANT *iface, REFIID riid, void **ppv)
|
|
{
|
|
struct module_enum *This = module_enum_from_IEnumVARIANT(iface);
|
|
|
|
if (IsEqualGUID(&IID_IUnknown, riid) || IsEqualGUID(&IID_IEnumVARIANT, riid))
|
|
{
|
|
*ppv = &This->IEnumVARIANT_iface;
|
|
}
|
|
else
|
|
{
|
|
WARN("unsupported interface: (%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI module_enum_AddRef(IEnumVARIANT *iface)
|
|
{
|
|
struct module_enum *This = module_enum_from_IEnumVARIANT(iface);
|
|
LONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI module_enum_Release(IEnumVARIANT *iface)
|
|
{
|
|
struct module_enum *This = module_enum_from_IEnumVARIANT(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
if (!ref)
|
|
{
|
|
IActiveScriptSite_Release(&This->host->IActiveScriptSite_iface);
|
|
IScriptControl_Release(&This->control->IScriptControl_iface);
|
|
heap_free(This);
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI module_enum_Next(IEnumVARIANT *iface, ULONG celt, VARIANT *rgVar, ULONG *pCeltFetched)
|
|
{
|
|
struct module_enum *This = module_enum_from_IEnumVARIANT(iface);
|
|
unsigned int i, num;
|
|
|
|
TRACE("(%p)->(%u %p %p)\n", This, celt, rgVar, pCeltFetched);
|
|
|
|
if (!rgVar) return E_POINTER;
|
|
if (This->host != This->control->host) return E_FAIL;
|
|
|
|
num = min(celt, This->host->module_count - This->pos);
|
|
for (i = 0; i < num; i++)
|
|
{
|
|
V_VT(rgVar + i) = VT_DISPATCH;
|
|
V_DISPATCH(rgVar + i) = (IDispatch*)(&This->control->modules[This->pos++]->IScriptModule_iface);
|
|
IDispatch_AddRef(V_DISPATCH(rgVar + i));
|
|
}
|
|
|
|
if (pCeltFetched) *pCeltFetched = i;
|
|
return i == celt ? S_OK : S_FALSE;
|
|
}
|
|
|
|
static HRESULT WINAPI module_enum_Skip(IEnumVARIANT *iface, ULONG celt)
|
|
{
|
|
struct module_enum *This = module_enum_from_IEnumVARIANT(iface);
|
|
|
|
TRACE("(%p)->(%u)\n", This, celt);
|
|
|
|
if (This->host != This->control->host) return E_FAIL;
|
|
|
|
if (This->host->module_count - This->pos < celt)
|
|
{
|
|
This->pos = This->host->module_count;
|
|
return S_FALSE;
|
|
}
|
|
This->pos += celt;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI module_enum_Reset(IEnumVARIANT *iface)
|
|
{
|
|
struct module_enum *This = module_enum_from_IEnumVARIANT(iface);
|
|
|
|
TRACE("(%p)\n", This);
|
|
|
|
if (This->host != This->control->host) return E_FAIL;
|
|
|
|
This->pos = 0;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI module_enum_Clone(IEnumVARIANT *iface, IEnumVARIANT **ppEnum)
|
|
{
|
|
struct module_enum *This = module_enum_from_IEnumVARIANT(iface);
|
|
struct module_enum *clone;
|
|
|
|
TRACE("(%p)->(%p)\n", This, ppEnum);
|
|
|
|
if (!ppEnum) return E_POINTER;
|
|
if (This->host != This->control->host) return E_FAIL;
|
|
|
|
if (!(clone = heap_alloc(sizeof(*clone))))
|
|
return E_OUTOFMEMORY;
|
|
|
|
*clone = *This;
|
|
clone->ref = 1;
|
|
IActiveScriptSite_AddRef(&This->host->IActiveScriptSite_iface);
|
|
IScriptControl_AddRef(&This->control->IScriptControl_iface);
|
|
|
|
*ppEnum = &clone->IEnumVARIANT_iface;
|
|
return S_OK;
|
|
}
|
|
|
|
static const IEnumVARIANTVtbl module_enum_vtbl = {
|
|
module_enum_QueryInterface,
|
|
module_enum_AddRef,
|
|
module_enum_Release,
|
|
module_enum_Next,
|
|
module_enum_Skip,
|
|
module_enum_Reset,
|
|
module_enum_Clone
|
|
};
|
|
|
|
static ScriptModule *create_module(ScriptHost *host, BSTR name)
|
|
{
|
|
ScriptModule *module;
|
|
|
|
if (!(module = heap_alloc_zero(sizeof(*module)))) return NULL;
|
|
|
|
module->IScriptModule_iface.lpVtbl = &ScriptModuleVtbl;
|
|
module->ref = 1;
|
|
if (name && !(module->name = SysAllocString(name)))
|
|
{
|
|
heap_free(module);
|
|
return NULL;
|
|
}
|
|
module->host = host;
|
|
IActiveScriptSite_AddRef(&host->IActiveScriptSite_iface);
|
|
return module;
|
|
}
|
|
|
|
static void release_modules(ScriptControl *control, BOOL force_detach)
|
|
{
|
|
unsigned int i, module_count = control->host->module_count;
|
|
|
|
for (i = 0; i < module_count; i++) {
|
|
if (force_detach) detach_module(control->modules[i]);
|
|
IScriptModule_Release(&control->modules[i]->IScriptModule_iface);
|
|
}
|
|
|
|
heap_free(control->modules);
|
|
}
|
|
|
|
static ScriptModule *find_module(ScriptControl *control, BSTR name)
|
|
{
|
|
unsigned int i;
|
|
|
|
if (!wcsicmp(name, L"Global"))
|
|
return control->modules[0];
|
|
|
|
for (i = 1; i < control->host->module_count; i++)
|
|
{
|
|
if (!wcsicmp(name, control->modules[i]->name))
|
|
return control->modules[i];
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModuleCollection_QueryInterface(IScriptModuleCollection *iface, REFIID riid, void **ppv)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptModuleCollection(iface);
|
|
|
|
if (IsEqualGUID(&IID_IDispatch, riid) || IsEqualGUID(&IID_IUnknown, riid) ||
|
|
IsEqualGUID(&IID_IScriptModuleCollection, riid))
|
|
{
|
|
*ppv = &This->IScriptModuleCollection_iface;
|
|
}
|
|
else
|
|
{
|
|
WARN("unsupported interface: (%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI ScriptModuleCollection_AddRef(IScriptModuleCollection *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptModuleCollection(iface);
|
|
return IScriptControl_AddRef(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static ULONG WINAPI ScriptModuleCollection_Release(IScriptModuleCollection *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptModuleCollection(iface);
|
|
return IScriptControl_Release(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModuleCollection_GetTypeInfoCount(IScriptModuleCollection *iface, UINT *pctinfo)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptModuleCollection(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pctinfo);
|
|
|
|
*pctinfo = 1;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModuleCollection_GetTypeInfo(IScriptModuleCollection *iface, UINT iTInfo,
|
|
LCID lcid, ITypeInfo **ppTInfo)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptModuleCollection(iface);
|
|
|
|
TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
|
|
|
|
return get_typeinfo(IScriptModuleCollection_tid, ppTInfo);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModuleCollection_GetIDsOfNames(IScriptModuleCollection *iface, REFIID riid,
|
|
LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptModuleCollection(iface);
|
|
ITypeInfo *typeinfo;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames, lcid, rgDispId);
|
|
|
|
hr = get_typeinfo(IScriptModuleCollection_tid, &typeinfo);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
|
|
ITypeInfo_Release(typeinfo);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModuleCollection_Invoke(IScriptModuleCollection *iface, DISPID dispIdMember,
|
|
REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult,
|
|
EXCEPINFO *pExcepInfo, UINT *puArgErr)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptModuleCollection(iface);
|
|
ITypeInfo *typeinfo;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
|
|
lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
|
|
hr = get_typeinfo(IScriptModuleCollection_tid, &typeinfo);
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
hr = ITypeInfo_Invoke(typeinfo, iface, dispIdMember, wFlags,
|
|
pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
ITypeInfo_Release(typeinfo);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModuleCollection_get__NewEnum(IScriptModuleCollection *iface, IUnknown **ppenumContexts)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptModuleCollection(iface);
|
|
struct module_enum *module_enum;
|
|
|
|
TRACE("(%p)->(%p)\n", This, ppenumContexts);
|
|
|
|
if (!ppenumContexts) return E_POINTER;
|
|
if (!This->host) return E_FAIL;
|
|
|
|
if (!(module_enum = heap_alloc(sizeof(*module_enum))))
|
|
return E_OUTOFMEMORY;
|
|
|
|
module_enum->IEnumVARIANT_iface.lpVtbl = &module_enum_vtbl;
|
|
module_enum->ref = 1;
|
|
module_enum->pos = 0;
|
|
module_enum->host = This->host;
|
|
module_enum->control = This;
|
|
IActiveScriptSite_AddRef(&This->host->IActiveScriptSite_iface);
|
|
IScriptControl_AddRef(&This->IScriptControl_iface);
|
|
|
|
*ppenumContexts = (IUnknown*)&module_enum->IEnumVARIANT_iface;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModuleCollection_get_Item(IScriptModuleCollection *iface, VARIANT index,
|
|
IScriptModule **ppmod)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptModuleCollection(iface);
|
|
ScriptModule *module;
|
|
unsigned int i;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%s %p)\n", This, wine_dbgstr_variant(&index), ppmod);
|
|
|
|
if (!ppmod) return E_POINTER;
|
|
if (!This->host) return E_FAIL;
|
|
|
|
if (V_VT(&index) == VT_BSTR)
|
|
{
|
|
module = find_module(This, V_BSTR(&index));
|
|
if (!module) return CTL_E_ILLEGALFUNCTIONCALL;
|
|
}
|
|
else
|
|
{
|
|
hr = VariantChangeType(&index, &index, 0, VT_INT);
|
|
if (FAILED(hr)) return hr;
|
|
|
|
i = V_INT(&index) - 1;
|
|
if (i >= This->host->module_count) return 0x800a0009;
|
|
|
|
module = This->modules[i];
|
|
}
|
|
|
|
*ppmod = &module->IScriptModule_iface;
|
|
IScriptModule_AddRef(*ppmod);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModuleCollection_get_Count(IScriptModuleCollection *iface, LONG *plCount)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptModuleCollection(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, plCount);
|
|
|
|
if (!plCount) return E_POINTER;
|
|
if (!This->host) return E_FAIL;
|
|
|
|
*plCount = This->host->module_count;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptModuleCollection_Add(IScriptModuleCollection *iface, BSTR name,
|
|
VARIANT *object, IScriptModule **ppmod)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptModuleCollection(iface);
|
|
ScriptModule *module, **modules;
|
|
ScriptHost *host = This->host;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%s %s %p)\n", This, wine_dbgstr_w(name), wine_dbgstr_variant(object), ppmod);
|
|
|
|
if (!ppmod) return E_POINTER;
|
|
if (!name || V_VT(object) != VT_DISPATCH) return E_INVALIDARG;
|
|
if (!host) return E_FAIL;
|
|
if (find_module(This, name)) return E_INVALIDARG;
|
|
|
|
/* See if we need to grow the array */
|
|
if (is_power_of_2(host->module_count))
|
|
{
|
|
modules = heap_realloc(This->modules, host->module_count * 2 * sizeof(*This->modules));
|
|
if (!modules) return E_OUTOFMEMORY;
|
|
This->modules = modules;
|
|
}
|
|
|
|
if (!(module = create_module(host, name)))
|
|
return E_OUTOFMEMORY;
|
|
|
|
/* If no object, Windows only calls AddNamedItem without adding a NULL object */
|
|
if (V_DISPATCH(object))
|
|
hr = add_script_object(host, name, V_DISPATCH(object), 0);
|
|
else
|
|
hr = IActiveScript_AddNamedItem(host->script, name, SCRIPTITEM_CODEONLY);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
IScriptModule_Release(&module->IScriptModule_iface);
|
|
return hr;
|
|
}
|
|
This->modules[host->module_count++] = module;
|
|
|
|
*ppmod = &module->IScriptModule_iface;
|
|
IScriptModule_AddRef(*ppmod);
|
|
return S_OK;
|
|
}
|
|
|
|
static const IScriptModuleCollectionVtbl ScriptModuleCollectionVtbl = {
|
|
ScriptModuleCollection_QueryInterface,
|
|
ScriptModuleCollection_AddRef,
|
|
ScriptModuleCollection_Release,
|
|
ScriptModuleCollection_GetTypeInfoCount,
|
|
ScriptModuleCollection_GetTypeInfo,
|
|
ScriptModuleCollection_GetIDsOfNames,
|
|
ScriptModuleCollection_Invoke,
|
|
ScriptModuleCollection_get__NewEnum,
|
|
ScriptModuleCollection_get_Item,
|
|
ScriptModuleCollection_get_Count,
|
|
ScriptModuleCollection_Add
|
|
};
|
|
|
|
static void fill_error_info(ScriptError *error)
|
|
{
|
|
EXCEPINFO info;
|
|
|
|
if (error->info_filled) return;
|
|
error->info_filled = TRUE;
|
|
|
|
if (!error->object)
|
|
return;
|
|
if (FAILED(IActiveScriptError_GetExceptionInfo(error->object, &info)))
|
|
return;
|
|
if (info.pfnDeferredFillIn)
|
|
info.pfnDeferredFillIn(&info);
|
|
|
|
error->number = info.scode;
|
|
error->source = info.bstrSource;
|
|
error->desc = info.bstrDescription;
|
|
error->help_file = info.bstrHelpFile;
|
|
error->help_context = info.dwHelpContext;
|
|
}
|
|
|
|
static void fill_error_text(ScriptError *error)
|
|
{
|
|
if (error->text_filled) return;
|
|
error->text_filled = TRUE;
|
|
|
|
if (error->object)
|
|
IActiveScriptError_GetSourceLineText(error->object, &error->text);
|
|
}
|
|
|
|
static void fill_error_pos(ScriptError *error)
|
|
{
|
|
DWORD context;
|
|
LONG column;
|
|
ULONG line;
|
|
|
|
if (error->pos_filled) return;
|
|
error->pos_filled = TRUE;
|
|
|
|
if (!error->object)
|
|
return;
|
|
if (FAILED(IActiveScriptError_GetSourcePosition(error->object, &context, &line, &column)))
|
|
return;
|
|
|
|
error->line = line;
|
|
error->column = column;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_QueryInterface(IScriptError *iface, REFIID riid, void **ppv)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
|
|
if (IsEqualGUID(&IID_IDispatch, riid) || IsEqualGUID(&IID_IUnknown, riid) ||
|
|
IsEqualGUID(&IID_IScriptError, riid))
|
|
{
|
|
*ppv = &This->IScriptError_iface;
|
|
}
|
|
else
|
|
{
|
|
WARN("unsupported interface: (%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI ScriptError_AddRef(IScriptError *iface)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
LONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI ScriptError_Release(IScriptError *iface)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
if (!ref)
|
|
{
|
|
IScriptError_Clear(&This->IScriptError_iface);
|
|
heap_free(This);
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_GetTypeInfoCount(IScriptError *iface, UINT *pctinfo)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pctinfo);
|
|
|
|
*pctinfo = 1;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_GetTypeInfo(IScriptError *iface, UINT iTInfo,
|
|
LCID lcid, ITypeInfo **ppTInfo)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
|
|
TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
|
|
|
|
return get_typeinfo(IScriptError_tid, ppTInfo);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_GetIDsOfNames(IScriptError *iface, REFIID riid,
|
|
LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
ITypeInfo *typeinfo;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames, lcid, rgDispId);
|
|
|
|
hr = get_typeinfo(IScriptError_tid, &typeinfo);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
|
|
ITypeInfo_Release(typeinfo);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_Invoke(IScriptError *iface, DISPID dispIdMember,
|
|
REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult,
|
|
EXCEPINFO *pExcepInfo, UINT *puArgErr)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
ITypeInfo *typeinfo;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
|
|
lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
|
|
hr = get_typeinfo(IScriptError_tid, &typeinfo);
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
hr = ITypeInfo_Invoke(typeinfo, iface, dispIdMember, wFlags,
|
|
pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
ITypeInfo_Release(typeinfo);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_get_Number(IScriptError *iface, LONG *plNumber)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, plNumber);
|
|
|
|
fill_error_info(This);
|
|
*plNumber = This->number;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_get_Source(IScriptError *iface, BSTR *pbstrSource)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pbstrSource);
|
|
|
|
fill_error_info(This);
|
|
*pbstrSource = SysAllocString(This->source);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_get_Description(IScriptError *iface, BSTR *pbstrDescription)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pbstrDescription);
|
|
|
|
fill_error_info(This);
|
|
*pbstrDescription = SysAllocString(This->desc);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_get_HelpFile(IScriptError *iface, BSTR *pbstrHelpFile)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pbstrHelpFile);
|
|
|
|
fill_error_info(This);
|
|
*pbstrHelpFile = SysAllocString(This->help_file);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_get_HelpContext(IScriptError *iface, LONG *plHelpContext)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, plHelpContext);
|
|
|
|
fill_error_info(This);
|
|
*plHelpContext = This->help_context;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_get_Text(IScriptError *iface, BSTR *pbstrText)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, pbstrText);
|
|
|
|
fill_error_text(This);
|
|
*pbstrText = SysAllocString(This->text);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_get_Line(IScriptError *iface, LONG *plLine)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, plLine);
|
|
|
|
fill_error_pos(This);
|
|
*plLine = This->line;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_get_Column(IScriptError *iface, LONG *plColumn)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, plColumn);
|
|
|
|
fill_error_pos(This);
|
|
*plColumn = This->column;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptError_Clear(IScriptError *iface)
|
|
{
|
|
ScriptError *This = impl_from_IScriptError(iface);
|
|
|
|
TRACE("(%p)->()\n", This);
|
|
|
|
if (This->object)
|
|
{
|
|
IActiveScriptError_Release(This->object);
|
|
This->object = NULL;
|
|
}
|
|
SysFreeString(This->text);
|
|
SysFreeString(This->source);
|
|
SysFreeString(This->desc);
|
|
SysFreeString(This->help_file);
|
|
|
|
This->number = 0;
|
|
This->text = NULL;
|
|
This->source = NULL;
|
|
This->desc = NULL;
|
|
This->help_file = NULL;
|
|
This->help_context = 0;
|
|
This->line = 0;
|
|
This->column = 0;
|
|
|
|
This->info_filled = FALSE;
|
|
This->text_filled = FALSE;
|
|
This->pos_filled = FALSE;
|
|
return S_OK;
|
|
}
|
|
|
|
static const IScriptErrorVtbl ScriptErrorVtbl = {
|
|
ScriptError_QueryInterface,
|
|
ScriptError_AddRef,
|
|
ScriptError_Release,
|
|
ScriptError_GetTypeInfoCount,
|
|
ScriptError_GetTypeInfo,
|
|
ScriptError_GetIDsOfNames,
|
|
ScriptError_Invoke,
|
|
ScriptError_get_Number,
|
|
ScriptError_get_Source,
|
|
ScriptError_get_Description,
|
|
ScriptError_get_HelpFile,
|
|
ScriptError_get_HelpContext,
|
|
ScriptError_get_Text,
|
|
ScriptError_get_Line,
|
|
ScriptError_get_Column,
|
|
ScriptError_Clear
|
|
};
|
|
|
|
static HRESULT init_script_host(ScriptControl *control, const CLSID *clsid, ScriptHost **ret)
|
|
{
|
|
IObjectSafety *objsafety;
|
|
ScriptHost *host;
|
|
HRESULT hr;
|
|
|
|
*ret = NULL;
|
|
|
|
host = heap_alloc(sizeof(*host));
|
|
if (!host)
|
|
return E_OUTOFMEMORY;
|
|
|
|
host->IActiveScriptSite_iface.lpVtbl = &ActiveScriptSiteVtbl;
|
|
host->IActiveScriptSiteWindow_iface.lpVtbl = &ActiveScriptSiteWindowVtbl;
|
|
host->IServiceProvider_iface.lpVtbl = &ServiceProviderVtbl;
|
|
host->ref = 1;
|
|
host->script = NULL;
|
|
host->parse = NULL;
|
|
host->clsid = *clsid;
|
|
host->module_count = 1;
|
|
list_init(&host->named_items);
|
|
|
|
hr = CoCreateInstance(&host->clsid, NULL, CLSCTX_INPROC_SERVER|CLSCTX_INPROC_HANDLER,
|
|
&IID_IActiveScript, (void**)&host->script);
|
|
if (FAILED(hr)) {
|
|
WARN("Failed to create an instance for %s, %#x\n", debugstr_guid(clsid), hr);
|
|
goto failed;
|
|
}
|
|
|
|
hr = IActiveScript_QueryInterface(host->script, &IID_IObjectSafety, (void**)&objsafety);
|
|
if (FAILED(hr)) {
|
|
FIXME("Could not get IObjectSafety, %#x\n", hr);
|
|
goto failed;
|
|
}
|
|
|
|
hr = IObjectSafety_SetInterfaceSafetyOptions(objsafety, &IID_IActiveScriptParse, INTERFACESAFE_FOR_UNTRUSTED_DATA, 0);
|
|
IObjectSafety_Release(objsafety);
|
|
if (FAILED(hr)) {
|
|
FIXME("SetInterfaceSafetyOptions failed, %#x\n", hr);
|
|
goto failed;
|
|
}
|
|
|
|
hr = IActiveScript_SetScriptSite(host->script, &host->IActiveScriptSite_iface);
|
|
if (FAILED(hr)) {
|
|
WARN("SetScriptSite failed, %#x\n", hr);
|
|
goto failed;
|
|
}
|
|
|
|
hr = IActiveScript_QueryInterface(host->script, &IID_IActiveScriptParse, (void**)&host->parse);
|
|
if (FAILED(hr)) {
|
|
WARN("Failed to get IActiveScriptParse, %#x\n", hr);
|
|
goto failed;
|
|
}
|
|
|
|
hr = IActiveScriptParse_InitNew(host->parse);
|
|
if (FAILED(hr)) {
|
|
WARN("InitNew failed, %#x\n", hr);
|
|
goto failed;
|
|
}
|
|
host->script_state = SCRIPTSTATE_INITIALIZED;
|
|
host->error = control->error;
|
|
IScriptError_AddRef(&host->error->IScriptError_iface);
|
|
|
|
*ret = host;
|
|
return S_OK;
|
|
|
|
failed:
|
|
detach_script_host(host);
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_QueryInterface(IScriptControl *iface, REFIID riid, void **ppv)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
|
|
if(IsEqualGUID(&IID_IUnknown, riid)) {
|
|
TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
|
|
*ppv = &This->IScriptControl_iface;
|
|
}else if(IsEqualGUID(&IID_IDispatch, riid)) {
|
|
TRACE("(%p)->(IID_IDispatch %p)\n", This, ppv);
|
|
*ppv = &This->IScriptControl_iface;
|
|
}else if(IsEqualGUID(&IID_IScriptControl, riid)) {
|
|
TRACE("(%p)->(IID_IScriptControl %p)\n", This, ppv);
|
|
*ppv = &This->IScriptControl_iface;
|
|
}else if(IsEqualGUID(&IID_IOleObject, riid)) {
|
|
TRACE("(%p)->(IID_IOleObject %p)\n", This, ppv);
|
|
*ppv = &This->IOleObject_iface;
|
|
}else if(IsEqualGUID(&IID_IPersistStreamInit, riid)) {
|
|
TRACE("(%p)->(IID_IPersistStreamInit %p)\n", This, ppv);
|
|
*ppv = &This->IPersistStreamInit_iface;
|
|
}else if(IsEqualGUID(&IID_IPersist, riid)) {
|
|
TRACE("(%p)->(IID_IPersist %p)\n", This, ppv);
|
|
*ppv = &This->IPersistStreamInit_iface;
|
|
}else if(IsEqualGUID(&IID_IOleControl, riid)) {
|
|
TRACE("(%p)->(IID_IOleControl %p)\n", This, ppv);
|
|
*ppv = &This->IOleControl_iface;
|
|
}else if(IsEqualGUID(&IID_IQuickActivate, riid)) {
|
|
TRACE("(%p)->(IID_IQuickActivate %p)\n", This, ppv);
|
|
*ppv = &This->IQuickActivate_iface;
|
|
}else if(IsEqualGUID(&IID_IViewObject, riid)) {
|
|
TRACE("(%p)->(IID_IViewObject %p)\n", This, ppv);
|
|
*ppv = &This->IViewObjectEx_iface;
|
|
}else if(IsEqualGUID(&IID_IViewObject2, riid)) {
|
|
TRACE("(%p)->(IID_IViewObject2 %p)\n", This, ppv);
|
|
*ppv = &This->IViewObjectEx_iface;
|
|
}else if(IsEqualGUID(&IID_IViewObjectEx, riid)) {
|
|
TRACE("(%p)->(IID_IViewObjectEx %p)\n", This, ppv);
|
|
*ppv = &This->IViewObjectEx_iface;
|
|
}else if(IsEqualGUID(&IID_IPointerInactive, riid)) {
|
|
TRACE("(%p)->(IID_IPointerInactive %p)\n", This, ppv);
|
|
*ppv = &This->IPointerInactive_iface;
|
|
}else if(IsEqualGUID(&IID_IConnectionPointContainer, riid)) {
|
|
TRACE("(%p)->(IID_IConnectionPointContainer %p)\n", This, ppv);
|
|
*ppv = &This->IConnectionPointContainer_iface;
|
|
}else {
|
|
FIXME("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI ScriptControl_AddRef(IScriptControl *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
LONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI ScriptControl_Release(IScriptControl *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%d\n", This, ref);
|
|
|
|
if(!ref) {
|
|
if (This->site)
|
|
IOleClientSite_Release(This->site);
|
|
if (This->host)
|
|
{
|
|
release_modules(This, FALSE);
|
|
IActiveScriptSite_Release(&This->host->IActiveScriptSite_iface);
|
|
}
|
|
IScriptError_Release(&This->error->IScriptError_iface);
|
|
heap_free(This);
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_GetTypeInfoCount(IScriptControl *iface, UINT *pctinfo)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
TRACE("(%p)->(%p)\n", This, pctinfo);
|
|
*pctinfo = 1;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_GetTypeInfo(IScriptControl *iface, UINT iTInfo,
|
|
LCID lcid, ITypeInfo **ppTInfo)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
TRACE("(%p)->(%u %u %p)\n", This, iTInfo, lcid, ppTInfo);
|
|
return get_typeinfo(IScriptControl_tid, ppTInfo);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_GetIDsOfNames(IScriptControl *iface, REFIID riid,
|
|
LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
ITypeInfo *typeinfo;
|
|
HRESULT hres;
|
|
|
|
TRACE("(%p)->(%s %p %u %u %p)\n", This, debugstr_guid(riid), rgszNames, cNames, lcid, rgDispId);
|
|
|
|
hres = get_typeinfo(IScriptControl_tid, &typeinfo);
|
|
if(SUCCEEDED(hres)) {
|
|
hres = ITypeInfo_GetIDsOfNames(typeinfo, rgszNames, cNames, rgDispId);
|
|
ITypeInfo_Release(typeinfo);
|
|
}
|
|
|
|
return hres;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_Invoke(IScriptControl *iface, DISPID dispIdMember,
|
|
REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult,
|
|
EXCEPINFO *pExcepInfo, UINT *puArgErr)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
ITypeInfo *typeinfo;
|
|
HRESULT hres;
|
|
|
|
TRACE("(%p)->(%d %s %d %d %p %p %p %p)\n", This, dispIdMember, debugstr_guid(riid),
|
|
lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
|
|
hres = get_typeinfo(IScriptControl_tid, &typeinfo);
|
|
if(SUCCEEDED(hres)) {
|
|
hres = ITypeInfo_Invoke(typeinfo, iface, dispIdMember, wFlags,
|
|
pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
ITypeInfo_Release(typeinfo);
|
|
}
|
|
|
|
return hres;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_get_Language(IScriptControl *iface, BSTR *p)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
LPOLESTR progidW;
|
|
HRESULT hr;
|
|
|
|
TRACE("(%p)->(%p)\n", This, p);
|
|
|
|
if (!p)
|
|
return E_POINTER;
|
|
|
|
*p = NULL;
|
|
|
|
if (!This->host)
|
|
return S_OK;
|
|
|
|
hr = ProgIDFromCLSID(&This->host->clsid, &progidW);
|
|
if (FAILED(hr))
|
|
return hr;
|
|
|
|
*p = SysAllocString(progidW);
|
|
CoTaskMemFree(progidW);
|
|
return *p ? S_OK : E_OUTOFMEMORY;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_put_Language(IScriptControl *iface, BSTR language)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
CLSID clsid;
|
|
HRESULT hres;
|
|
|
|
TRACE("(%p)->(%s)\n", This, debugstr_w(language));
|
|
|
|
if (language && FAILED(CLSIDFromProgID(language, &clsid)))
|
|
return CTL_E_INVALIDPROPERTYVALUE;
|
|
|
|
if (This->host) {
|
|
release_modules(This, TRUE);
|
|
IActiveScriptSite_Release(&This->host->IActiveScriptSite_iface);
|
|
This->host = NULL;
|
|
}
|
|
|
|
if (!language)
|
|
return S_OK;
|
|
|
|
hres = init_script_host(This, &clsid, &This->host);
|
|
if (FAILED(hres))
|
|
return hres;
|
|
|
|
/* Alloc global module */
|
|
This->modules = heap_alloc_zero(sizeof(*This->modules));
|
|
if (This->modules) {
|
|
This->modules[0] = create_module(This->host, NULL);
|
|
if (!This->modules[0]) {
|
|
heap_free(This->modules);
|
|
This->modules = NULL;
|
|
hres = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
else
|
|
hres = E_OUTOFMEMORY;
|
|
|
|
if (FAILED(hres)) {
|
|
detach_script_host(This->host);
|
|
This->host = NULL;
|
|
}
|
|
return hres;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_get_State(IScriptControl *iface, ScriptControlStates *p)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
SCRIPTSTATE state;
|
|
HRESULT hres;
|
|
|
|
TRACE("(%p)->(%p)\n", This, p);
|
|
|
|
if(!p)
|
|
return E_POINTER;
|
|
|
|
if(!This->host)
|
|
return E_FAIL;
|
|
|
|
hres = IActiveScript_GetScriptState(This->host->script, &state);
|
|
if (FAILED(hres)) return hres;
|
|
|
|
switch (state)
|
|
{
|
|
case SCRIPTSTATE_INITIALIZED:
|
|
case SCRIPTSTATE_STARTED:
|
|
*p = Initialized;
|
|
break;
|
|
case SCRIPTSTATE_CONNECTED:
|
|
*p = Connected;
|
|
break;
|
|
default:
|
|
WARN("unexpected state %d\n", state);
|
|
return E_FAIL;
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_put_State(IScriptControl *iface, ScriptControlStates state)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
TRACE("(%p)->(%x)\n", This, state);
|
|
|
|
if(!This->host)
|
|
return E_FAIL;
|
|
|
|
if(state != Initialized && state != Connected)
|
|
return CTL_E_INVALIDPROPERTYVALUE;
|
|
|
|
return IActiveScript_SetScriptState(This->host->script, state == Connected ? SCRIPTSTATE_CONNECTED : SCRIPTSTATE_STARTED);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_put_SitehWnd(IScriptControl *iface, LONG hwnd)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
|
|
FIXME("(%p)->(%x)\n", This, hwnd);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_get_SitehWnd(IScriptControl *iface, LONG *p)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
FIXME("(%p)->(%p)\n", This, p);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_get_Timeout(IScriptControl *iface, LONG *p)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, p);
|
|
|
|
if (!p)
|
|
return E_POINTER;
|
|
|
|
*p = This->timeout;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_put_Timeout(IScriptControl *iface, LONG timeout)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
|
|
TRACE("(%p)->(%d)\n", This, timeout);
|
|
|
|
if (timeout < -1)
|
|
return CTL_E_INVALIDPROPERTYVALUE;
|
|
|
|
if (timeout != -1)
|
|
FIXME("execution timeout ignored\n");
|
|
|
|
This->timeout = timeout;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_get_AllowUI(IScriptControl *iface, VARIANT_BOOL *p)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
TRACE("(%p)->(%p)\n", This, p);
|
|
|
|
if(!p)
|
|
return E_POINTER;
|
|
|
|
*p = This->allow_ui;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_put_AllowUI(IScriptControl *iface, VARIANT_BOOL allow_ui)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
TRACE("(%p)->(%x)\n", This, allow_ui);
|
|
|
|
This->allow_ui = allow_ui;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_get_UseSafeSubset(IScriptControl *iface, VARIANT_BOOL *p)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
TRACE("(%p)->(%p)\n", This, p);
|
|
|
|
if(!p)
|
|
return E_POINTER;
|
|
|
|
*p = This->use_safe_subset;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_put_UseSafeSubset(IScriptControl *iface, VARIANT_BOOL use_safe_subset)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
TRACE("(%p)->(%x)\n", This, use_safe_subset);
|
|
|
|
This->use_safe_subset = use_safe_subset;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_get_Modules(IScriptControl *iface, IScriptModuleCollection **p)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, p);
|
|
|
|
if (!This->host) return E_FAIL;
|
|
|
|
*p = &This->IScriptModuleCollection_iface;
|
|
IScriptControl_AddRef(iface);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_get_Error(IScriptControl *iface, IScriptError **p)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, p);
|
|
|
|
if (!p) return E_POINTER;
|
|
|
|
*p = &This->error->IScriptError_iface;
|
|
IScriptError_AddRef(*p);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_get_CodeObject(IScriptControl *iface, IDispatch **p)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, p);
|
|
|
|
if (!This->host) return E_FAIL;
|
|
|
|
return IScriptModule_get_CodeObject(&This->modules[0]->IScriptModule_iface, p);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_get_Procedures(IScriptControl *iface, IScriptProcedureCollection **p)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, p);
|
|
|
|
if (!This->host) return E_FAIL;
|
|
|
|
return IScriptModule_get_Procedures(&This->modules[0]->IScriptModule_iface, p);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl__AboutBox(IScriptControl *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
FIXME("(%p)\n", This);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_AddObject(IScriptControl *iface, BSTR name, IDispatch *object, VARIANT_BOOL add_members)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
DWORD flags = SCRIPTITEM_ISVISIBLE | SCRIPTITEM_ISSOURCE;
|
|
|
|
TRACE("(%p)->(%s %p %x)\n", This, debugstr_w(name), object, add_members);
|
|
|
|
if (!object)
|
|
return E_INVALIDARG;
|
|
|
|
if (!This->host)
|
|
return E_FAIL;
|
|
|
|
if (add_members)
|
|
flags |= SCRIPTITEM_GLOBALMEMBERS;
|
|
return add_script_object(This->host, name, object, flags);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_Reset(IScriptControl *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
|
|
TRACE("(%p)\n", This);
|
|
|
|
if (!This->host)
|
|
return E_FAIL;
|
|
|
|
clear_named_items(This->host);
|
|
return set_script_state(This->host, SCRIPTSTATE_INITIALIZED);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_AddCode(IScriptControl *iface, BSTR code)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
|
|
TRACE("(%p)->(%s).\n", This, debugstr_w(code));
|
|
|
|
if (!This->host)
|
|
return E_FAIL;
|
|
|
|
return parse_script_text(This->modules[0], code, SCRIPTTEXT_ISVISIBLE, NULL);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_Eval(IScriptControl *iface, BSTR expression, VARIANT *res)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
|
|
TRACE("(%p)->(%s, %p).\n", This, debugstr_w(expression), res);
|
|
|
|
if (!res)
|
|
return E_POINTER;
|
|
V_VT(res) = VT_EMPTY;
|
|
if (!This->host)
|
|
return E_FAIL;
|
|
|
|
return parse_script_text(This->modules[0], expression, SCRIPTTEXT_ISEXPRESSION, res);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_ExecuteStatement(IScriptControl *iface, BSTR statement)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
|
|
TRACE("(%p)->(%s)\n", This, debugstr_w(statement));
|
|
|
|
if (!This->host)
|
|
return E_FAIL;
|
|
|
|
return parse_script_text(This->modules[0], statement, 0, NULL);
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_Run(IScriptControl *iface, BSTR procedure_name, SAFEARRAY **parameters, VARIANT *res)
|
|
{
|
|
ScriptControl *This = impl_from_IScriptControl(iface);
|
|
SAFEARRAY *sa;
|
|
|
|
TRACE("(%p)->(%s %p %p)\n", This, debugstr_w(procedure_name), parameters, res);
|
|
|
|
if (!parameters || !res) return E_POINTER;
|
|
if (!(sa = *parameters)) return E_POINTER;
|
|
|
|
V_VT(res) = VT_EMPTY;
|
|
if (sa->cDims == 0) return DISP_E_BADINDEX;
|
|
if (!(sa->fFeatures & FADF_VARIANT)) return DISP_E_BADVARTYPE;
|
|
if (!This->host) return E_FAIL;
|
|
|
|
return run_procedure(This->modules[0], procedure_name, sa, res);
|
|
}
|
|
|
|
static const IScriptControlVtbl ScriptControlVtbl = {
|
|
ScriptControl_QueryInterface,
|
|
ScriptControl_AddRef,
|
|
ScriptControl_Release,
|
|
ScriptControl_GetTypeInfoCount,
|
|
ScriptControl_GetTypeInfo,
|
|
ScriptControl_GetIDsOfNames,
|
|
ScriptControl_Invoke,
|
|
ScriptControl_get_Language,
|
|
ScriptControl_put_Language,
|
|
ScriptControl_get_State,
|
|
ScriptControl_put_State,
|
|
ScriptControl_put_SitehWnd,
|
|
ScriptControl_get_SitehWnd,
|
|
ScriptControl_get_Timeout,
|
|
ScriptControl_put_Timeout,
|
|
ScriptControl_get_AllowUI,
|
|
ScriptControl_put_AllowUI,
|
|
ScriptControl_get_UseSafeSubset,
|
|
ScriptControl_put_UseSafeSubset,
|
|
ScriptControl_get_Modules,
|
|
ScriptControl_get_Error,
|
|
ScriptControl_get_CodeObject,
|
|
ScriptControl_get_Procedures,
|
|
ScriptControl__AboutBox,
|
|
ScriptControl_AddObject,
|
|
ScriptControl_Reset,
|
|
ScriptControl_AddCode,
|
|
ScriptControl_Eval,
|
|
ScriptControl_ExecuteStatement,
|
|
ScriptControl_Run
|
|
};
|
|
|
|
static HRESULT WINAPI OleObject_QueryInterface(IOleObject *iface, REFIID riid, void **obj)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
return IScriptControl_QueryInterface(&This->IScriptControl_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI OleObject_AddRef(IOleObject *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
return IScriptControl_AddRef(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static ULONG WINAPI OleObject_Release(IOleObject *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
return IScriptControl_Release(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_SetClientSite(IOleObject *iface, IOleClientSite *site)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, site);
|
|
|
|
if (This->site)
|
|
IOleClientSite_Release(This->site);
|
|
|
|
if ((This->site = site))
|
|
IOleClientSite_AddRef(site);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_GetClientSite(IOleObject *iface, IOleClientSite **site)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, site);
|
|
|
|
if (!site)
|
|
return E_POINTER;
|
|
|
|
if ((*site = This->site))
|
|
IOleClientSite_AddRef(*site);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_SetHostNames(IOleObject *iface, LPCOLESTR containerapp, LPCOLESTR containerobj)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%s %s)\n", This, debugstr_w(containerapp), debugstr_w(containerobj));
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_Close(IOleObject *iface, DWORD save)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%d)\n", This, save);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_SetMoniker(IOleObject *iface, DWORD which, IMoniker *moniker)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%d %p)\n", This, which, moniker);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_GetMoniker(IOleObject *iface, DWORD assign, DWORD which, IMoniker **moniker)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%d %d %p)\n", This, assign, which, moniker);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_InitFromData(IOleObject *iface, IDataObject *dataobj, BOOL creation,
|
|
DWORD reserved)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%p %d %d)\n", This, dataobj, creation, reserved);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_GetClipboardData(IOleObject *iface, DWORD reserved, IDataObject **dataobj)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%d %p)\n", This, reserved, dataobj);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_DoVerb(IOleObject *iface, LONG verb, LPMSG msg, IOleClientSite *active_site,
|
|
LONG index, HWND hwndParent, LPCRECT rect)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%d %p %p %d %p %p)\n", This, verb, msg, active_site, index, hwndParent, rect);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_EnumVerbs(IOleObject *iface, IEnumOLEVERB **enumoleverb)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%p)\n", This, enumoleverb);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_Update(IOleObject *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)\n", This);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_IsUpToDate(IOleObject *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)\n", This);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_GetUserClassID(IOleObject *iface, CLSID *clsid)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%p)\n", This, clsid);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_GetUserType(IOleObject *iface, DWORD form_of_type, LPOLESTR *usertype)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%d %p)\n", This, form_of_type, usertype);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_SetExtent(IOleObject *iface, DWORD aspect, SIZEL *size)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%d %p)\n", This, aspect, size);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_GetExtent(IOleObject *iface, DWORD aspect, SIZEL *size)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
TRACE("(%p)->(%d %p)\n", This, aspect, size);
|
|
|
|
if (aspect != DVASPECT_CONTENT)
|
|
return DV_E_DVASPECT;
|
|
|
|
*size = This->extent;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_Advise(IOleObject *iface, IAdviseSink *sink, DWORD *connection)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%p %p)\n", This, sink, connection);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_Unadvise(IOleObject *iface, DWORD connection)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%d)\n", This, connection);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_EnumAdvise(IOleObject *iface, IEnumSTATDATA **enumadvise)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%p)\n", This, enumadvise);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_GetMiscStatus(IOleObject *iface, DWORD aspect, DWORD *status)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
TRACE("(%p)->(%d %p)\n", This, aspect, status);
|
|
|
|
return OleRegGetMiscStatus(&CLSID_ScriptControl, aspect, status);
|
|
}
|
|
|
|
static HRESULT WINAPI OleObject_SetColorScheme(IOleObject *iface, LOGPALETTE *logpal)
|
|
{
|
|
ScriptControl *This = impl_from_IOleObject(iface);
|
|
|
|
FIXME("(%p)->(%p)\n", This, logpal);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IOleObjectVtbl OleObjectVtbl = {
|
|
OleObject_QueryInterface,
|
|
OleObject_AddRef,
|
|
OleObject_Release,
|
|
OleObject_SetClientSite,
|
|
OleObject_GetClientSite,
|
|
OleObject_SetHostNames,
|
|
OleObject_Close,
|
|
OleObject_SetMoniker,
|
|
OleObject_GetMoniker,
|
|
OleObject_InitFromData,
|
|
OleObject_GetClipboardData,
|
|
OleObject_DoVerb,
|
|
OleObject_EnumVerbs,
|
|
OleObject_Update,
|
|
OleObject_IsUpToDate,
|
|
OleObject_GetUserClassID,
|
|
OleObject_GetUserType,
|
|
OleObject_SetExtent,
|
|
OleObject_GetExtent,
|
|
OleObject_Advise,
|
|
OleObject_Unadvise,
|
|
OleObject_EnumAdvise,
|
|
OleObject_GetMiscStatus,
|
|
OleObject_SetColorScheme
|
|
};
|
|
|
|
static HRESULT WINAPI PersistStreamInit_QueryInterface(IPersistStreamInit *iface, REFIID riid, void **obj)
|
|
{
|
|
ScriptControl *This = impl_from_IPersistStreamInit(iface);
|
|
return IScriptControl_QueryInterface(&This->IScriptControl_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI PersistStreamInit_AddRef(IPersistStreamInit *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IPersistStreamInit(iface);
|
|
return IScriptControl_AddRef(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static ULONG WINAPI PersistStreamInit_Release(IPersistStreamInit *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IPersistStreamInit(iface);
|
|
return IScriptControl_Release(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI PersistStreamInit_GetClassID(IPersistStreamInit *iface, CLSID *clsid)
|
|
{
|
|
ScriptControl *This = impl_from_IPersistStreamInit(iface);
|
|
|
|
FIXME("(%p)->(%p)\n", This, clsid);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI PersistStreamInit_IsDirty(IPersistStreamInit *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IPersistStreamInit(iface);
|
|
|
|
FIXME("(%p)\n", This);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI PersistStreamInit_Load(IPersistStreamInit *iface, IStream *stream)
|
|
{
|
|
ScriptControl *This = impl_from_IPersistStreamInit(iface);
|
|
|
|
FIXME("(%p)->(%p)\n", This, stream);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI PersistStreamInit_Save(IPersistStreamInit *iface, IStream *stream, BOOL clear_dirty)
|
|
{
|
|
ScriptControl *This = impl_from_IPersistStreamInit(iface);
|
|
|
|
FIXME("(%p)->(%p %d)\n", This, stream, clear_dirty);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI PersistStreamInit_GetSizeMax(IPersistStreamInit *iface, ULARGE_INTEGER *size)
|
|
{
|
|
ScriptControl *This = impl_from_IPersistStreamInit(iface);
|
|
|
|
FIXME("(%p)->(%p)\n", This, size);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI PersistStreamInit_InitNew(IPersistStreamInit *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IPersistStreamInit(iface);
|
|
|
|
FIXME("(%p)\n", This);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static const IPersistStreamInitVtbl PersistStreamInitVtbl = {
|
|
PersistStreamInit_QueryInterface,
|
|
PersistStreamInit_AddRef,
|
|
PersistStreamInit_Release,
|
|
PersistStreamInit_GetClassID,
|
|
PersistStreamInit_IsDirty,
|
|
PersistStreamInit_Load,
|
|
PersistStreamInit_Save,
|
|
PersistStreamInit_GetSizeMax,
|
|
PersistStreamInit_InitNew
|
|
};
|
|
|
|
static HRESULT WINAPI OleControl_QueryInterface(IOleControl *iface, REFIID riid, void **obj)
|
|
{
|
|
ScriptControl *This = impl_from_IOleControl(iface);
|
|
return IScriptControl_QueryInterface(&This->IScriptControl_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI OleControl_AddRef(IOleControl *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IOleControl(iface);
|
|
return IScriptControl_AddRef(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static ULONG WINAPI OleControl_Release(IOleControl *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IOleControl(iface);
|
|
return IScriptControl_Release(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI OleControl_GetControlInfo(IOleControl *iface, CONTROLINFO *info)
|
|
{
|
|
ScriptControl *This = impl_from_IOleControl(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, info);
|
|
|
|
if (!info)
|
|
return E_POINTER;
|
|
|
|
info->hAccel = NULL;
|
|
info->cAccel = 0;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI OleControl_OnMnemonic(IOleControl *iface, MSG *msg)
|
|
{
|
|
ScriptControl *This = impl_from_IOleControl(iface);
|
|
|
|
FIXME("(%p)->(%p)\n", This, msg);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleControl_OnAmbientPropertyChange(IOleControl *iface, DISPID dispid)
|
|
{
|
|
ScriptControl *This = impl_from_IOleControl(iface);
|
|
|
|
FIXME("(%p)->(%#x)\n", This, dispid);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI OleControl_FreezeEvents(IOleControl *iface, BOOL freeze)
|
|
{
|
|
ScriptControl *This = impl_from_IOleControl(iface);
|
|
|
|
FIXME("(%p)->(%d)\n", This, freeze);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IOleControlVtbl OleControlVtbl = {
|
|
OleControl_QueryInterface,
|
|
OleControl_AddRef,
|
|
OleControl_Release,
|
|
OleControl_GetControlInfo,
|
|
OleControl_OnMnemonic,
|
|
OleControl_OnAmbientPropertyChange,
|
|
OleControl_FreezeEvents
|
|
};
|
|
|
|
static HRESULT WINAPI QuickActivate_QueryInterface(IQuickActivate *iface, REFIID riid, void **obj)
|
|
{
|
|
ScriptControl *This = impl_from_IQuickActivate(iface);
|
|
return IScriptControl_QueryInterface(&This->IScriptControl_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI QuickActivate_AddRef(IQuickActivate *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IQuickActivate(iface);
|
|
return IScriptControl_AddRef(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static ULONG WINAPI QuickActivate_Release(IQuickActivate *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IQuickActivate(iface);
|
|
return IScriptControl_Release(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI QuickActivate_QuickActivate(IQuickActivate *iface, QACONTAINER *container, QACONTROL *control)
|
|
{
|
|
ScriptControl *This = impl_from_IQuickActivate(iface);
|
|
|
|
FIXME("(%p)->(%p %p)\n", This, container, control);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI QuickActivate_SetContentExtent(IQuickActivate *iface, SIZEL *size)
|
|
{
|
|
ScriptControl *This = impl_from_IQuickActivate(iface);
|
|
|
|
FIXME("(%p)->(%p)\n", This, size);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI QuickActivate_GetContentExtent(IQuickActivate *iface, SIZEL *size)
|
|
{
|
|
ScriptControl *This = impl_from_IQuickActivate(iface);
|
|
|
|
FIXME("(%p)->(%p)\n", This, size);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IQuickActivateVtbl QuickActivateVtbl = {
|
|
QuickActivate_QueryInterface,
|
|
QuickActivate_AddRef,
|
|
QuickActivate_Release,
|
|
QuickActivate_QuickActivate,
|
|
QuickActivate_SetContentExtent,
|
|
QuickActivate_GetContentExtent
|
|
};
|
|
|
|
static HRESULT WINAPI ViewObject_QueryInterface(IViewObjectEx *iface, REFIID riid, void **obj)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
return IScriptControl_QueryInterface(&This->IScriptControl_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI ViewObject_AddRef(IViewObjectEx *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
return IScriptControl_AddRef(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static ULONG WINAPI ViewObject_Release(IViewObjectEx *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
return IScriptControl_Release(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI ViewObject_Draw(IViewObjectEx *iface, DWORD drawaspect, LONG index, void *aspect,
|
|
DVTARGETDEVICE *device, HDC target_dev, HDC hdc_draw, const RECTL *bounds, const RECTL *win_bounds,
|
|
BOOL (STDMETHODCALLTYPE *fn_continue)(ULONG_PTR cont), ULONG_PTR cont)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
|
|
FIXME("(%p)->(%d %d %p %p %p %p %p %p %p %lu)\n", This, drawaspect, index, aspect, device, target_dev,
|
|
hdc_draw, bounds, win_bounds, fn_continue, cont);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ViewObject_GetColorSet(IViewObjectEx *iface, DWORD drawaspect, LONG index, void *aspect,
|
|
DVTARGETDEVICE *device, HDC hic_target, LOGPALETTE **colorset)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
|
|
FIXME("(%p)->(%d %d %p %p %p %p)\n", This, drawaspect, index, aspect, device, hic_target,
|
|
colorset);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ViewObject_Freeze(IViewObjectEx *iface, DWORD drawaspect, LONG index, void *aspect,
|
|
DWORD *freeze)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
|
|
FIXME("(%p)->(%d %d %p %p)\n", This, drawaspect, index, aspect, freeze);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ViewObject_Unfreeze(IViewObjectEx *iface, DWORD freeze)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
|
|
FIXME("(%p)->(%d)\n", This, freeze);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ViewObject_SetAdvise(IViewObjectEx *iface, DWORD aspects, DWORD flags, IAdviseSink *sink)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
|
|
TRACE("(%p)->(%d %#x %p)\n", This, aspects, flags, sink);
|
|
|
|
if (aspects != DVASPECT_CONTENT)
|
|
return DV_E_DVASPECT;
|
|
|
|
This->view_sink_flags = flags;
|
|
if (This->view_sink)
|
|
IAdviseSink_Release(This->view_sink);
|
|
This->view_sink = sink;
|
|
if (This->view_sink)
|
|
IAdviseSink_AddRef(This->view_sink);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ViewObject_GetAdvise(IViewObjectEx *iface, DWORD *aspects, DWORD *flags, IAdviseSink **sink)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
|
|
TRACE("(%p)->(%p %p %p)\n", This, aspects, flags, sink);
|
|
|
|
if (aspects)
|
|
*aspects = DVASPECT_CONTENT;
|
|
if (flags)
|
|
*flags = This->view_sink_flags;
|
|
if (sink) {
|
|
*sink = This->view_sink;
|
|
if (*sink)
|
|
IAdviseSink_AddRef(*sink);
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ViewObject_GetExtent(IViewObjectEx *iface, DWORD draw_aspect, LONG index,
|
|
DVTARGETDEVICE *device, SIZEL *size)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
|
|
FIXME("(%p)->(%d %d %p %p)\n", This, draw_aspect, index, device, size);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ViewObject_GetRect(IViewObjectEx *iface, DWORD aspect, RECTL *rect)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
|
|
FIXME("(%p)->(%d %p)\n", This, aspect, rect);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ViewObject_GetViewStatus(IViewObjectEx *iface, DWORD *status)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, status);
|
|
|
|
*status = VIEWSTATUS_OPAQUE;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ViewObject_QueryHitPoint(IViewObjectEx *iface, DWORD aspect, const RECT *bounds,
|
|
POINT pt, LONG close_hint, DWORD *hit_result)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
|
|
FIXME("(%p)->(%d %s %s %d %p)\n", This, aspect, wine_dbgstr_rect(bounds), wine_dbgstr_point(&pt), close_hint, hit_result);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ViewObject_QueryHitRect(IViewObjectEx *iface, DWORD aspect, const RECT *bounds,
|
|
const RECT *loc, LONG close_hint, DWORD *hit_result)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
|
|
FIXME("(%p)->(%d %s %s %d %p)\n", This, aspect, wine_dbgstr_rect(bounds), wine_dbgstr_rect(loc), close_hint, hit_result);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ViewObject_GetNaturalExtent(IViewObjectEx *iface, DWORD aspect, LONG index,
|
|
DVTARGETDEVICE *device, HDC target_hdc, DVEXTENTINFO *extent_info, SIZEL *size)
|
|
{
|
|
ScriptControl *This = impl_from_IViewObjectEx(iface);
|
|
|
|
FIXME("(%p)->(%d %d %p %p %p %p)\n", This, aspect, index, device, target_hdc, extent_info, size);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IViewObjectExVtbl ViewObjectExVtbl = {
|
|
ViewObject_QueryInterface,
|
|
ViewObject_AddRef,
|
|
ViewObject_Release,
|
|
ViewObject_Draw,
|
|
ViewObject_GetColorSet,
|
|
ViewObject_Freeze,
|
|
ViewObject_Unfreeze,
|
|
ViewObject_SetAdvise,
|
|
ViewObject_GetAdvise,
|
|
ViewObject_GetExtent,
|
|
ViewObject_GetRect,
|
|
ViewObject_GetViewStatus,
|
|
ViewObject_QueryHitPoint,
|
|
ViewObject_QueryHitRect,
|
|
ViewObject_GetNaturalExtent
|
|
};
|
|
|
|
static HRESULT WINAPI PointerInactive_QueryInterface(IPointerInactive *iface, REFIID riid, void **obj)
|
|
{
|
|
ScriptControl *This = impl_from_IPointerInactive(iface);
|
|
return IScriptControl_QueryInterface(&This->IScriptControl_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI PointerInactive_AddRef(IPointerInactive *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IPointerInactive(iface);
|
|
return IScriptControl_AddRef(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static ULONG WINAPI PointerInactive_Release(IPointerInactive *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IPointerInactive(iface);
|
|
return IScriptControl_Release(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI PointerInactive_GetActivationPolicy(IPointerInactive *iface, DWORD *policy)
|
|
{
|
|
ScriptControl *This = impl_from_IPointerInactive(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, policy);
|
|
|
|
if (!policy)
|
|
return E_POINTER;
|
|
|
|
*policy = 0;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI PointerInactive_OnInactiveMouseMove(IPointerInactive *iface, const RECT *bounds,
|
|
LONG x, LONG y, DWORD key_state)
|
|
{
|
|
ScriptControl *This = impl_from_IPointerInactive(iface);
|
|
|
|
FIXME("(%p)->(%s %d %d %#x)\n", This, wine_dbgstr_rect(bounds), x, y, key_state);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI PointerInactive_OnInactiveSetCursor(IPointerInactive *iface, const RECT *bounds,
|
|
LONG x, LONG y, DWORD msg, BOOL set_always)
|
|
{
|
|
ScriptControl *This = impl_from_IPointerInactive(iface);
|
|
|
|
FIXME("(%p)->(%s %d %d %#x %d)\n", This, wine_dbgstr_rect(bounds), x, y, msg, set_always);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IPointerInactiveVtbl PointerInactiveVtbl = {
|
|
PointerInactive_QueryInterface,
|
|
PointerInactive_AddRef,
|
|
PointerInactive_Release,
|
|
PointerInactive_GetActivationPolicy,
|
|
PointerInactive_OnInactiveMouseMove,
|
|
PointerInactive_OnInactiveSetCursor
|
|
};
|
|
|
|
static HRESULT WINAPI ConnectionPointContainer_QueryInterface(IConnectionPointContainer *iface, REFIID riid, void **obj)
|
|
{
|
|
ScriptControl *This = impl_from_IConnectionPointContainer(iface);
|
|
return IScriptControl_QueryInterface(&This->IScriptControl_iface, riid, obj);
|
|
}
|
|
|
|
static ULONG WINAPI ConnectionPointContainer_AddRef(IConnectionPointContainer *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IConnectionPointContainer(iface);
|
|
return IScriptControl_AddRef(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static ULONG WINAPI ConnectionPointContainer_Release(IConnectionPointContainer *iface)
|
|
{
|
|
ScriptControl *This = impl_from_IConnectionPointContainer(iface);
|
|
return IScriptControl_Release(&This->IScriptControl_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI ConnectionPointContainer_EnumConnectionPoints(IConnectionPointContainer *iface, IEnumConnectionPoints **enum_points)
|
|
{
|
|
ScriptControl *This = impl_from_IConnectionPointContainer(iface);
|
|
|
|
FIXME("(%p)->(%p)\n", This, enum_points);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ConnectionPointContainer_FindConnectionPoint(IConnectionPointContainer *iface, REFIID riid, IConnectionPoint **cp)
|
|
{
|
|
ScriptControl *This = impl_from_IConnectionPointContainer(iface);
|
|
ConnectionPoint *iter;
|
|
|
|
TRACE("(%p)->(%s %p)\n", This, debugstr_guid(riid), cp);
|
|
|
|
*cp = NULL;
|
|
|
|
for (iter = This->cp_list; iter; iter = iter->next) {
|
|
if (IsEqualIID(iter->riid, riid))
|
|
*cp = &iter->IConnectionPoint_iface;
|
|
}
|
|
|
|
if (*cp) {
|
|
IConnectionPoint_AddRef(*cp);
|
|
return S_OK;
|
|
}
|
|
|
|
FIXME("unsupported connection point %s\n", debugstr_guid(riid));
|
|
return CONNECT_E_NOCONNECTION;
|
|
}
|
|
|
|
static const IConnectionPointContainerVtbl ConnectionPointContainerVtbl = {
|
|
ConnectionPointContainer_QueryInterface,
|
|
ConnectionPointContainer_AddRef,
|
|
ConnectionPointContainer_Release,
|
|
ConnectionPointContainer_EnumConnectionPoints,
|
|
ConnectionPointContainer_FindConnectionPoint
|
|
};
|
|
|
|
static HRESULT WINAPI ConnectionPoint_QueryInterface(IConnectionPoint *iface,
|
|
REFIID riid, void **ppv)
|
|
{
|
|
ConnectionPoint *This = impl_from_IConnectionPoint(iface);
|
|
|
|
if(IsEqualGUID(&IID_IUnknown, riid)) {
|
|
TRACE("(%p)->(IID_IUnknown %p)\n", This, ppv);
|
|
*ppv = &This->IConnectionPoint_iface;
|
|
}else if(IsEqualGUID(&IID_IConnectionPoint, riid)) {
|
|
TRACE("(%p)->(IID_IDispatch %p)\n", This, ppv);
|
|
*ppv = &This->IConnectionPoint_iface;
|
|
}else {
|
|
FIXME("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
|
|
*ppv = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI ConnectionPoint_AddRef(IConnectionPoint *iface)
|
|
{
|
|
ConnectionPoint *This = impl_from_IConnectionPoint(iface);
|
|
return IConnectionPointContainer_AddRef(&This->control->IConnectionPointContainer_iface);
|
|
}
|
|
|
|
static ULONG WINAPI ConnectionPoint_Release(IConnectionPoint *iface)
|
|
{
|
|
ConnectionPoint *This = impl_from_IConnectionPoint(iface);
|
|
return IConnectionPointContainer_Release(&This->control->IConnectionPointContainer_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI ConnectionPoint_GetConnectionInterface(IConnectionPoint *iface, IID *iid)
|
|
{
|
|
ConnectionPoint *This = impl_from_IConnectionPoint(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, iid);
|
|
|
|
*iid = *This->riid;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ConnectionPoint_GetConnectionPointContainer(IConnectionPoint *iface,
|
|
IConnectionPointContainer **container)
|
|
{
|
|
ConnectionPoint *This = impl_from_IConnectionPoint(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, container);
|
|
|
|
if (!container)
|
|
return E_POINTER;
|
|
|
|
*container = &This->control->IConnectionPointContainer_iface;
|
|
IConnectionPointContainer_AddRef(*container);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI ConnectionPoint_Advise(IConnectionPoint *iface, IUnknown *unk_sink,
|
|
DWORD *cookie)
|
|
{
|
|
ConnectionPoint *This = impl_from_IConnectionPoint(iface);
|
|
|
|
FIXME("(%p)->(%p %p)\n", This, unk_sink, cookie);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ConnectionPoint_Unadvise(IConnectionPoint *iface, DWORD cookie)
|
|
{
|
|
ConnectionPoint *This = impl_from_IConnectionPoint(iface);
|
|
|
|
FIXME("(%p)->(%d)\n", This, cookie);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ConnectionPoint_EnumConnections(IConnectionPoint *iface,
|
|
IEnumConnections **ppEnum)
|
|
{
|
|
ConnectionPoint *This = impl_from_IConnectionPoint(iface);
|
|
|
|
FIXME("(%p)->(%p): stub\n", This, ppEnum);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IConnectionPointVtbl ConnectionPointVtbl =
|
|
{
|
|
ConnectionPoint_QueryInterface,
|
|
ConnectionPoint_AddRef,
|
|
ConnectionPoint_Release,
|
|
ConnectionPoint_GetConnectionInterface,
|
|
ConnectionPoint_GetConnectionPointContainer,
|
|
ConnectionPoint_Advise,
|
|
ConnectionPoint_Unadvise,
|
|
ConnectionPoint_EnumConnections
|
|
};
|
|
|
|
static void ConnectionPoint_Init(ConnectionPoint *cp, ScriptControl *sc, REFIID riid)
|
|
{
|
|
cp->IConnectionPoint_iface.lpVtbl = &ConnectionPointVtbl;
|
|
cp->control = sc;
|
|
cp->riid = riid;
|
|
|
|
cp->next = sc->cp_list;
|
|
sc->cp_list = cp;
|
|
}
|
|
|
|
static HRESULT WINAPI ScriptControl_CreateInstance(IClassFactory *iface, IUnknown *outer, REFIID riid, void **ppv)
|
|
{
|
|
ScriptControl *script_control;
|
|
DWORD dpi_x, dpi_y;
|
|
HRESULT hres;
|
|
HDC hdc;
|
|
|
|
TRACE("(%p %s %p)\n", outer, debugstr_guid(riid), ppv);
|
|
|
|
script_control = heap_alloc_zero(sizeof(*script_control));
|
|
if(!script_control)
|
|
return E_OUTOFMEMORY;
|
|
|
|
script_control->error = heap_alloc_zero(sizeof(*script_control->error));
|
|
if(!script_control->error)
|
|
{
|
|
heap_free(script_control);
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
script_control->IScriptControl_iface.lpVtbl = &ScriptControlVtbl;
|
|
script_control->IPersistStreamInit_iface.lpVtbl = &PersistStreamInitVtbl;
|
|
script_control->IOleObject_iface.lpVtbl = &OleObjectVtbl;
|
|
script_control->IOleControl_iface.lpVtbl = &OleControlVtbl;
|
|
script_control->IQuickActivate_iface.lpVtbl = &QuickActivateVtbl;
|
|
script_control->IViewObjectEx_iface.lpVtbl = &ViewObjectExVtbl;
|
|
script_control->IPointerInactive_iface.lpVtbl = &PointerInactiveVtbl;
|
|
script_control->IConnectionPointContainer_iface.lpVtbl = &ConnectionPointContainerVtbl;
|
|
script_control->IScriptModuleCollection_iface.lpVtbl = &ScriptModuleCollectionVtbl;
|
|
script_control->ref = 1;
|
|
script_control->timeout = 10000;
|
|
script_control->allow_ui = VARIANT_TRUE;
|
|
script_control->use_safe_subset = VARIANT_FALSE;
|
|
|
|
script_control->error->IScriptError_iface.lpVtbl = &ScriptErrorVtbl;
|
|
script_control->error->ref = 1;
|
|
|
|
ConnectionPoint_Init(&script_control->cp_scsource, script_control, &DIID_DScriptControlSource);
|
|
ConnectionPoint_Init(&script_control->cp_propnotif, script_control, &IID_IPropertyNotifySink);
|
|
|
|
hdc = GetDC(0);
|
|
dpi_x = GetDeviceCaps(hdc, LOGPIXELSX);
|
|
dpi_y = GetDeviceCaps(hdc, LOGPIXELSY);
|
|
ReleaseDC(0, hdc);
|
|
|
|
script_control->extent.cx = MulDiv(38, 2540, dpi_x);
|
|
script_control->extent.cy = MulDiv(38, 2540, dpi_y);
|
|
|
|
hres = IScriptControl_QueryInterface(&script_control->IScriptControl_iface, riid, ppv);
|
|
IScriptControl_Release(&script_control->IScriptControl_iface);
|
|
return hres;
|
|
}
|
|
|
|
/******************************************************************
|
|
* DllMain (msscript.ocx.@)
|
|
*/
|
|
BOOL WINAPI DllMain(HINSTANCE instance, DWORD reason, void *reserved)
|
|
{
|
|
TRACE("(%p %d %p)\n", instance, reason, reserved);
|
|
|
|
switch(reason) {
|
|
case DLL_WINE_PREATTACH:
|
|
return FALSE; /* prefer native version */
|
|
case DLL_PROCESS_ATTACH:
|
|
msscript_instance = instance;
|
|
DisableThreadLibraryCalls(instance);
|
|
break;
|
|
case DLL_PROCESS_DETACH:
|
|
if(!reserved)
|
|
release_typelib();
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static HRESULT WINAPI ClassFactory_QueryInterface(IClassFactory *iface, REFIID riid, void **ppv)
|
|
{
|
|
*ppv = NULL;
|
|
|
|
if(IsEqualGUID(&IID_IUnknown, riid)) {
|
|
TRACE("(%p)->(IID_IUnknown %p)\n", iface, ppv);
|
|
*ppv = iface;
|
|
}else if(IsEqualGUID(&IID_IClassFactory, riid)) {
|
|
TRACE("(%p)->(IID_IClassFactory %p)\n", iface, ppv);
|
|
*ppv = iface;
|
|
}
|
|
|
|
if(*ppv) {
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
WARN("(%p)->(%s %p)\n", iface, debugstr_guid(riid), ppv);
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
static ULONG WINAPI ClassFactory_AddRef(IClassFactory *iface)
|
|
{
|
|
TRACE("(%p)\n", iface);
|
|
return 2;
|
|
}
|
|
|
|
static ULONG WINAPI ClassFactory_Release(IClassFactory *iface)
|
|
{
|
|
TRACE("(%p)\n", iface);
|
|
return 1;
|
|
}
|
|
|
|
static HRESULT WINAPI ClassFactory_LockServer(IClassFactory *iface, BOOL fLock)
|
|
{
|
|
TRACE("(%p)->(%x)\n", iface, fLock);
|
|
return S_OK;
|
|
}
|
|
|
|
static const IClassFactoryVtbl ScriptControlFactoryVtbl = {
|
|
ClassFactory_QueryInterface,
|
|
ClassFactory_AddRef,
|
|
ClassFactory_Release,
|
|
ScriptControl_CreateInstance,
|
|
ClassFactory_LockServer
|
|
};
|
|
|
|
static IClassFactory ScriptControlFactory = { &ScriptControlFactoryVtbl };
|
|
|
|
/***********************************************************************
|
|
* DllGetClassObject (msscript.ocx.@)
|
|
*/
|
|
HRESULT WINAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, LPVOID *ppv)
|
|
{
|
|
if(IsEqualGUID(&CLSID_ScriptControl, rclsid)) {
|
|
TRACE("(CLSID_ScriptControl %s %p)\n", debugstr_guid(riid), ppv);
|
|
return IClassFactory_QueryInterface(&ScriptControlFactory, riid, ppv);
|
|
}
|
|
|
|
FIXME("%s %s %p\n", debugstr_guid(rclsid), debugstr_guid(riid), ppv);
|
|
return CLASS_E_CLASSNOTAVAILABLE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* DllCanUnloadNow (msscript.ocx.@)
|
|
*/
|
|
HRESULT WINAPI DllCanUnloadNow(void)
|
|
{
|
|
TRACE("\n");
|
|
return S_FALSE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* DllRegisterServer (msscript.ocx.@)
|
|
*/
|
|
HRESULT WINAPI DllRegisterServer(void)
|
|
{
|
|
TRACE("()\n");
|
|
return __wine_register_resources(msscript_instance);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* DllUnregisterServer (msscript.ocx.@)
|
|
*/
|
|
HRESULT WINAPI DllUnregisterServer(void)
|
|
{
|
|
TRACE("()\n");
|
|
return __wine_unregister_resources(msscript_instance);
|
|
}
|