1144 lines
33 KiB
C
1144 lines
33 KiB
C
/*
|
|
* Copyright 2005-2006 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
|
|
*/
|
|
|
|
#include "ieframe.h"
|
|
|
|
#include "exdispid.h"
|
|
#include "mshtml.h"
|
|
#include "perhist.h"
|
|
#include "initguid.h"
|
|
|
|
#include "wine/debug.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(ieframe);
|
|
|
|
DEFINE_OLEGUID(CGID_DocHostCmdPriv, 0x000214D4L, 0, 0);
|
|
|
|
#define DOCHOST_DOCCANNAVIGATE 0
|
|
|
|
/* Undocumented notification, see mshtml tests */
|
|
#define CMDID_EXPLORER_UPDATEHISTORY 38
|
|
|
|
static ATOM doc_view_atom = 0;
|
|
|
|
void push_dochost_task(DocHost *This, task_header_t *task, task_proc_t proc, task_destr_t destr, BOOL send)
|
|
{
|
|
BOOL is_empty;
|
|
|
|
task->proc = proc;
|
|
task->destr = destr;
|
|
|
|
is_empty = list_empty(&This->task_queue);
|
|
list_add_tail(&This->task_queue, &task->entry);
|
|
|
|
if(send)
|
|
SendMessageW(This->frame_hwnd, WM_DOCHOSTTASK, 0, 0);
|
|
else if(is_empty)
|
|
PostMessageW(This->frame_hwnd, WM_DOCHOSTTASK, 0, 0);
|
|
}
|
|
|
|
LRESULT process_dochost_tasks(DocHost *This)
|
|
{
|
|
task_header_t *task;
|
|
|
|
while(!list_empty(&This->task_queue)) {
|
|
task = LIST_ENTRY(This->task_queue.next, task_header_t, entry);
|
|
list_remove(&task->entry);
|
|
|
|
task->proc(This, task);
|
|
task->destr(task);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void abort_dochost_tasks(DocHost *This, task_proc_t proc)
|
|
{
|
|
task_header_t *task, *cursor;
|
|
|
|
LIST_FOR_EACH_ENTRY_SAFE(task, cursor, &This->task_queue, task_header_t, entry) {
|
|
if(proc && proc != task->proc)
|
|
continue;
|
|
|
|
list_remove(&task->entry);
|
|
task->destr(task);
|
|
}
|
|
}
|
|
|
|
void on_commandstate_change(DocHost *doc_host, LONG command, BOOL enable)
|
|
{
|
|
DISPPARAMS dispparams;
|
|
VARIANTARG params[2];
|
|
|
|
TRACE("command=%d enable=%d\n", command, enable);
|
|
|
|
dispparams.cArgs = 2;
|
|
dispparams.cNamedArgs = 0;
|
|
dispparams.rgdispidNamedArgs = NULL;
|
|
dispparams.rgvarg = params;
|
|
|
|
V_VT(params) = VT_BOOL;
|
|
V_BOOL(params) = enable ? VARIANT_TRUE : VARIANT_FALSE;
|
|
|
|
V_VT(params+1) = VT_I4;
|
|
V_I4(params+1) = command;
|
|
|
|
call_sink(doc_host->cps.wbe2, DISPID_COMMANDSTATECHANGE, &dispparams);
|
|
|
|
doc_host->container_vtbl->on_command_state_change(doc_host, command, enable);
|
|
}
|
|
|
|
void update_navigation_commands(DocHost *dochost)
|
|
{
|
|
unsigned pos = dochost->travellog.loading_pos == -1 ? dochost->travellog.position : dochost->travellog.loading_pos;
|
|
|
|
on_commandstate_change(dochost, CSC_NAVIGATEBACK, pos > 0);
|
|
on_commandstate_change(dochost, CSC_NAVIGATEFORWARD, pos < dochost->travellog.length);
|
|
}
|
|
|
|
static void notif_complete(DocHost *This, DISPID dispid)
|
|
{
|
|
DISPPARAMS dispparams;
|
|
VARIANTARG params[2];
|
|
VARIANT url;
|
|
|
|
dispparams.cArgs = 2;
|
|
dispparams.cNamedArgs = 0;
|
|
dispparams.rgdispidNamedArgs = NULL;
|
|
dispparams.rgvarg = params;
|
|
|
|
V_VT(params) = (VT_BYREF|VT_VARIANT);
|
|
V_BYREF(params) = &url;
|
|
|
|
V_VT(params+1) = VT_DISPATCH;
|
|
V_DISPATCH(params+1) = (IDispatch*)This->wb;
|
|
|
|
V_VT(&url) = VT_BSTR;
|
|
V_BSTR(&url) = SysAllocString(This->url);
|
|
|
|
TRACE("%d >>>\n", dispid);
|
|
call_sink(This->cps.wbe2, dispid, &dispparams);
|
|
TRACE("%d <<<\n", dispid);
|
|
|
|
SysFreeString(V_BSTR(&url));
|
|
This->busy = VARIANT_FALSE;
|
|
}
|
|
|
|
static void object_available(DocHost *This)
|
|
{
|
|
IHlinkTarget *hlink;
|
|
HRESULT hres;
|
|
|
|
TRACE("(%p)\n", This);
|
|
|
|
if(!This->document) {
|
|
WARN("document == NULL\n");
|
|
return;
|
|
}
|
|
|
|
hres = IUnknown_QueryInterface(This->document, &IID_IHlinkTarget, (void**)&hlink);
|
|
if(SUCCEEDED(hres)) {
|
|
hres = IHlinkTarget_Navigate(hlink, 0, NULL);
|
|
IHlinkTarget_Release(hlink);
|
|
if(FAILED(hres))
|
|
FIXME("Navigate failed\n");
|
|
}else {
|
|
IOleObject *ole_object;
|
|
RECT rect;
|
|
|
|
TRACE("No IHlink iface\n");
|
|
|
|
hres = IUnknown_QueryInterface(This->document, &IID_IOleObject, (void**)&ole_object);
|
|
if(FAILED(hres)) {
|
|
FIXME("Could not get IOleObject iface: %08x\n", hres);
|
|
return;
|
|
}
|
|
|
|
GetClientRect(This->hwnd, &rect);
|
|
hres = IOleObject_DoVerb(ole_object, OLEIVERB_SHOW, NULL, &This->IOleClientSite_iface, -1, This->hwnd, &rect);
|
|
IOleObject_Release(ole_object);
|
|
if(FAILED(hres))
|
|
FIXME("DoVerb failed: %08x\n", hres);
|
|
}
|
|
}
|
|
|
|
static HRESULT get_doc_ready_state(DocHost *This, READYSTATE *ret)
|
|
{
|
|
DISPPARAMS dp = {NULL,NULL,0,0};
|
|
IDispatch *disp;
|
|
EXCEPINFO ei;
|
|
VARIANT var;
|
|
HRESULT hres;
|
|
|
|
hres = IUnknown_QueryInterface(This->document, &IID_IDispatch, (void**)&disp);
|
|
if(FAILED(hres))
|
|
return hres;
|
|
|
|
hres = IDispatch_Invoke(disp, DISPID_READYSTATE, &IID_NULL, LOCALE_SYSTEM_DEFAULT, DISPATCH_PROPERTYGET,
|
|
&dp, &var, &ei, NULL);
|
|
IDispatch_Release(disp);
|
|
if(FAILED(hres)) {
|
|
WARN("Invoke(DISPID_READYSTATE failed: %08x\n", hres);
|
|
return hres;
|
|
}
|
|
|
|
if(V_VT(&var) != VT_I4) {
|
|
WARN("V_VT(var) = %d\n", V_VT(&var));
|
|
VariantClear(&var);
|
|
return E_FAIL;
|
|
}
|
|
|
|
*ret = V_I4(&var);
|
|
return S_OK;
|
|
}
|
|
|
|
static void advise_prop_notif(DocHost *This, BOOL set)
|
|
{
|
|
IConnectionPointContainer *cp_container;
|
|
IConnectionPoint *cp;
|
|
HRESULT hres;
|
|
|
|
hres = IUnknown_QueryInterface(This->document, &IID_IConnectionPointContainer, (void**)&cp_container);
|
|
if(FAILED(hres))
|
|
return;
|
|
|
|
hres = IConnectionPointContainer_FindConnectionPoint(cp_container, &IID_IPropertyNotifySink, &cp);
|
|
IConnectionPointContainer_Release(cp_container);
|
|
if(FAILED(hres))
|
|
return;
|
|
|
|
if(set)
|
|
hres = IConnectionPoint_Advise(cp, (IUnknown*)&This->IPropertyNotifySink_iface, &This->prop_notif_cookie);
|
|
else
|
|
hres = IConnectionPoint_Unadvise(cp, This->prop_notif_cookie);
|
|
IConnectionPoint_Release(cp);
|
|
|
|
if(SUCCEEDED(hres))
|
|
This->is_prop_notif = set;
|
|
}
|
|
|
|
void set_doc_state(DocHost *This, READYSTATE doc_state)
|
|
{
|
|
This->doc_state = doc_state;
|
|
if(doc_state > This->ready_state)
|
|
This->ready_state = doc_state;
|
|
}
|
|
|
|
static void update_ready_state(DocHost *This, READYSTATE ready_state)
|
|
{
|
|
if(ready_state > READYSTATE_LOADING && This->travellog.loading_pos != -1) {
|
|
WARN("histupdate not notified\n");
|
|
This->travellog.position = This->travellog.loading_pos;
|
|
This->travellog.loading_pos = -1;
|
|
}
|
|
|
|
if(ready_state > READYSTATE_LOADING && This->doc_state <= READYSTATE_LOADING && !This->browser_service /* FIXME */)
|
|
notif_complete(This, DISPID_NAVIGATECOMPLETE2);
|
|
|
|
if(ready_state == READYSTATE_COMPLETE && This->doc_state < READYSTATE_COMPLETE) {
|
|
set_doc_state(This, READYSTATE_COMPLETE);
|
|
if(!This->browser_service) /* FIXME: Not fully correct */
|
|
notif_complete(This, DISPID_DOCUMENTCOMPLETE);
|
|
}else {
|
|
set_doc_state(This, ready_state);
|
|
}
|
|
}
|
|
|
|
typedef struct {
|
|
task_header_t header;
|
|
IUnknown *doc;
|
|
READYSTATE ready_state;
|
|
} ready_state_task_t;
|
|
|
|
static void ready_state_task_destr(task_header_t *_task)
|
|
{
|
|
ready_state_task_t *task = (ready_state_task_t*)_task;
|
|
|
|
IUnknown_Release(task->doc);
|
|
heap_free(task);
|
|
}
|
|
|
|
static void ready_state_proc(DocHost *This, task_header_t *_task)
|
|
{
|
|
ready_state_task_t *task = (ready_state_task_t*)_task;
|
|
|
|
if(task->doc == This->document)
|
|
update_ready_state(This, task->ready_state);
|
|
}
|
|
|
|
static void push_ready_state_task(DocHost *This, READYSTATE ready_state)
|
|
{
|
|
ready_state_task_t *task = heap_alloc(sizeof(ready_state_task_t));
|
|
|
|
IUnknown_AddRef(This->document);
|
|
task->doc = This->document;
|
|
task->ready_state = ready_state;
|
|
|
|
push_dochost_task(This, &task->header, ready_state_proc, ready_state_task_destr, FALSE);
|
|
}
|
|
|
|
static void object_available_task_destr(task_header_t *task)
|
|
{
|
|
heap_free(task);
|
|
}
|
|
|
|
static void object_available_proc(DocHost *This, task_header_t *task)
|
|
{
|
|
object_available(This);
|
|
}
|
|
|
|
HRESULT dochost_object_available(DocHost *This, IUnknown *doc)
|
|
{
|
|
READYSTATE ready_state;
|
|
task_header_t *task;
|
|
IOleObject *oleobj;
|
|
HRESULT hres;
|
|
|
|
IUnknown_AddRef(doc);
|
|
This->document = doc;
|
|
|
|
hres = IUnknown_QueryInterface(doc, &IID_IOleObject, (void**)&oleobj);
|
|
if(SUCCEEDED(hres)) {
|
|
CLSID clsid;
|
|
|
|
hres = IOleObject_GetUserClassID(oleobj, &clsid);
|
|
if(SUCCEEDED(hres))
|
|
TRACE("Got clsid %s\n",
|
|
IsEqualGUID(&clsid, &CLSID_HTMLDocument) ? "CLSID_HTMLDocument" : debugstr_guid(&clsid));
|
|
|
|
hres = IOleObject_SetClientSite(oleobj, &This->IOleClientSite_iface);
|
|
if(FAILED(hres))
|
|
FIXME("SetClientSite failed: %08x\n", hres);
|
|
|
|
IOleObject_Release(oleobj);
|
|
}else {
|
|
FIXME("Could not get IOleObject iface: %08x\n", hres);
|
|
}
|
|
|
|
/* FIXME: Call SetAdvise */
|
|
|
|
task = heap_alloc(sizeof(*task));
|
|
push_dochost_task(This, task, object_available_proc, object_available_task_destr, FALSE);
|
|
|
|
hres = get_doc_ready_state(This, &ready_state);
|
|
if(SUCCEEDED(hres)) {
|
|
if(ready_state == READYSTATE_COMPLETE)
|
|
push_ready_state_task(This, READYSTATE_COMPLETE);
|
|
if(ready_state != READYSTATE_COMPLETE || This->doc_navigate)
|
|
advise_prop_notif(This, TRUE);
|
|
}else if(!This->doc_navigate) {
|
|
/* If we can't get document's ready state, there is not much we can do.
|
|
* Assume that document is complete at this point. */
|
|
push_ready_state_task(This, READYSTATE_COMPLETE);
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static LRESULT resize_document(DocHost *This, LONG width, LONG height)
|
|
{
|
|
RECT rect = {0, 0, width, height};
|
|
|
|
TRACE("(%p)->(%d %d)\n", This, width, height);
|
|
|
|
if(This->view)
|
|
IOleDocumentView_SetRect(This->view, &rect);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static LRESULT WINAPI doc_view_proc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
DocHost *This;
|
|
|
|
static const WCHAR wszTHIS[] = {'T','H','I','S',0};
|
|
|
|
if(msg == WM_CREATE) {
|
|
This = *(DocHost**)lParam;
|
|
SetPropW(hwnd, wszTHIS, This);
|
|
}else {
|
|
This = GetPropW(hwnd, wszTHIS);
|
|
}
|
|
|
|
switch(msg) {
|
|
case WM_SIZE:
|
|
return resize_document(This, LOWORD(lParam), HIWORD(lParam));
|
|
}
|
|
|
|
return DefWindowProcW(hwnd, msg, wParam, lParam);
|
|
}
|
|
|
|
static void free_travellog_entry(travellog_entry_t *entry)
|
|
{
|
|
if(entry->stream)
|
|
IStream_Release(entry->stream);
|
|
heap_free(entry->url);
|
|
}
|
|
|
|
static IStream *get_travellog_stream(DocHost *This)
|
|
{
|
|
IPersistHistory *persist_history;
|
|
IStream *stream;
|
|
HRESULT hres;
|
|
|
|
hres = IUnknown_QueryInterface(This->document, &IID_IPersistHistory, (void**)&persist_history);
|
|
if(FAILED(hres))
|
|
return NULL;
|
|
|
|
hres = CreateStreamOnHGlobal(NULL, TRUE, &stream);
|
|
if(SUCCEEDED(hres))
|
|
hres = IPersistHistory_SaveHistory(persist_history, stream);
|
|
IPersistHistory_Release(persist_history);
|
|
if(FAILED(hres)) {
|
|
IStream_Release(stream);
|
|
return NULL;
|
|
}
|
|
|
|
return stream;
|
|
}
|
|
|
|
static void dump_travellog(DocHost *This)
|
|
{
|
|
unsigned i;
|
|
|
|
for(i=0; i < This->travellog.length; i++)
|
|
TRACE("%d: %s %s\n", i, i == This->travellog.position ? "=>" : " ", debugstr_w(This->travellog.log[i].url));
|
|
if(i == This->travellog.position)
|
|
TRACE("%d: =>\n", i);
|
|
}
|
|
|
|
static void update_travellog(DocHost *This)
|
|
{
|
|
travellog_entry_t *new_entry;
|
|
|
|
static const WCHAR about_schemeW[] = {'a','b','o','u','t',':'};
|
|
|
|
if(This->url && !strncmpiW(This->url, about_schemeW, sizeof(about_schemeW)/sizeof(*about_schemeW))) {
|
|
TRACE("Skipping about URL\n");
|
|
return;
|
|
}
|
|
|
|
if(!This->travellog.log) {
|
|
This->travellog.log = heap_alloc(4 * sizeof(*This->travellog.log));
|
|
if(!This->travellog.log)
|
|
return;
|
|
|
|
This->travellog.size = 4;
|
|
}else if(This->travellog.size < This->travellog.position+1) {
|
|
travellog_entry_t *new_travellog;
|
|
|
|
new_travellog = heap_realloc(This->travellog.log, This->travellog.size*2*sizeof(*This->travellog.log));
|
|
if(!new_travellog)
|
|
return;
|
|
|
|
This->travellog.log = new_travellog;
|
|
This->travellog.size *= 2;
|
|
}
|
|
|
|
if(This->travellog.loading_pos == -1) {
|
|
/* Clear forward history. */
|
|
while(This->travellog.length > This->travellog.position)
|
|
free_travellog_entry(This->travellog.log + --This->travellog.length);
|
|
}
|
|
|
|
new_entry = This->travellog.log + This->travellog.position;
|
|
|
|
new_entry->url = heap_strdupW(This->url);
|
|
TRACE("Adding %s at %d\n", debugstr_w(This->url), This->travellog.position);
|
|
if(!new_entry->url)
|
|
return;
|
|
|
|
new_entry->stream = get_travellog_stream(This);
|
|
|
|
if(This->travellog.loading_pos == -1) {
|
|
This->travellog.position++;
|
|
}else {
|
|
This->travellog.position = This->travellog.loading_pos;
|
|
This->travellog.loading_pos = -1;
|
|
}
|
|
if(This->travellog.position > This->travellog.length)
|
|
This->travellog.length = This->travellog.position;
|
|
|
|
dump_travellog(This);
|
|
}
|
|
|
|
void create_doc_view_hwnd(DocHost *This)
|
|
{
|
|
RECT rect;
|
|
|
|
static const WCHAR wszShell_DocObject_View[] =
|
|
{'S','h','e','l','l',' ','D','o','c','O','b','j','e','c','t',' ','V','i','e','w',0};
|
|
|
|
if(!doc_view_atom) {
|
|
static WNDCLASSEXW wndclass = {
|
|
sizeof(wndclass),
|
|
CS_PARENTDC,
|
|
doc_view_proc,
|
|
0, 0 /* native uses 4*/, NULL, NULL, NULL,
|
|
(HBRUSH)(COLOR_WINDOW + 1), NULL,
|
|
wszShell_DocObject_View,
|
|
NULL
|
|
};
|
|
|
|
wndclass.hInstance = ieframe_instance;
|
|
|
|
doc_view_atom = RegisterClassExW(&wndclass);
|
|
}
|
|
|
|
This->container_vtbl->get_docobj_rect(This, &rect);
|
|
This->hwnd = CreateWindowExW(0, wszShell_DocObject_View,
|
|
wszShell_DocObject_View,
|
|
WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN | WS_TABSTOP,
|
|
rect.left, rect.top, rect.right, rect.bottom, This->frame_hwnd,
|
|
NULL, ieframe_instance, This);
|
|
}
|
|
|
|
void deactivate_document(DocHost *This)
|
|
{
|
|
IOleInPlaceObjectWindowless *winobj;
|
|
IOleObject *oleobj = NULL;
|
|
IHlinkTarget *hlink = NULL;
|
|
HRESULT hres;
|
|
|
|
if(!This->document) return;
|
|
|
|
if(This->doc_navigate) {
|
|
IUnknown_Release(This->doc_navigate);
|
|
This->doc_navigate = NULL;
|
|
}
|
|
|
|
if(This->is_prop_notif)
|
|
advise_prop_notif(This, FALSE);
|
|
|
|
if(This->view)
|
|
IOleDocumentView_UIActivate(This->view, FALSE);
|
|
|
|
hres = IUnknown_QueryInterface(This->document, &IID_IOleInPlaceObjectWindowless,
|
|
(void**)&winobj);
|
|
if(SUCCEEDED(hres)) {
|
|
IOleInPlaceObjectWindowless_InPlaceDeactivate(winobj);
|
|
IOleInPlaceObjectWindowless_Release(winobj);
|
|
}
|
|
|
|
if(This->view) {
|
|
IOleDocumentView_Show(This->view, FALSE);
|
|
IOleDocumentView_CloseView(This->view, 0);
|
|
IOleDocumentView_SetInPlaceSite(This->view, NULL);
|
|
IOleDocumentView_Release(This->view);
|
|
This->view = NULL;
|
|
}
|
|
|
|
hres = IUnknown_QueryInterface(This->document, &IID_IOleObject, (void**)&oleobj);
|
|
if(SUCCEEDED(hres))
|
|
IOleObject_Close(oleobj, OLECLOSE_NOSAVE);
|
|
|
|
hres = IUnknown_QueryInterface(This->document, &IID_IHlinkTarget, (void**)&hlink);
|
|
if(SUCCEEDED(hres)) {
|
|
IHlinkTarget_SetBrowseContext(hlink, NULL);
|
|
IHlinkTarget_Release(hlink);
|
|
}
|
|
|
|
if(oleobj) {
|
|
IOleClientSite *client_site = NULL;
|
|
|
|
IOleObject_GetClientSite(oleobj, &client_site);
|
|
if(client_site) {
|
|
if(client_site == &This->IOleClientSite_iface)
|
|
IOleObject_SetClientSite(oleobj, NULL);
|
|
IOleClientSite_Release(client_site);
|
|
}
|
|
|
|
IOleObject_Release(oleobj);
|
|
}
|
|
|
|
IUnknown_Release(This->document);
|
|
This->document = NULL;
|
|
}
|
|
|
|
HRESULT refresh_document(DocHost *This, const VARIANT *level)
|
|
{
|
|
IOleCommandTarget *cmdtrg;
|
|
VARIANT vin, vout;
|
|
HRESULT hres;
|
|
|
|
if(level && (V_VT(level) != VT_I4 || V_I4(level) != REFRESH_NORMAL))
|
|
FIXME("Unsupported refresh level %s\n", debugstr_variant(level));
|
|
|
|
if(!This->document) {
|
|
FIXME("no document\n");
|
|
return E_FAIL;
|
|
}
|
|
|
|
hres = IUnknown_QueryInterface(This->document, &IID_IOleCommandTarget, (void**)&cmdtrg);
|
|
if(FAILED(hres))
|
|
return hres;
|
|
|
|
V_VT(&vin) = VT_EMPTY;
|
|
V_VT(&vout) = VT_EMPTY;
|
|
hres = IOleCommandTarget_Exec(cmdtrg, NULL, OLECMDID_REFRESH, OLECMDEXECOPT_PROMPTUSER, &vin, &vout);
|
|
IOleCommandTarget_Release(cmdtrg);
|
|
if(FAILED(hres))
|
|
return hres;
|
|
|
|
VariantClear(&vout);
|
|
return S_OK;
|
|
}
|
|
|
|
void release_dochost_client(DocHost *This)
|
|
{
|
|
if(This->hwnd) {
|
|
DestroyWindow(This->hwnd);
|
|
This->hwnd = NULL;
|
|
}
|
|
|
|
if(This->hostui) {
|
|
IDocHostUIHandler_Release(This->hostui);
|
|
This->hostui = NULL;
|
|
}
|
|
|
|
if(This->client_disp) {
|
|
IDispatch_Release(This->client_disp);
|
|
This->client_disp = NULL;
|
|
}
|
|
|
|
if(This->frame) {
|
|
IOleInPlaceFrame_Release(This->frame);
|
|
This->frame = NULL;
|
|
}
|
|
|
|
if(This->olecmd) {
|
|
IOleCommandTarget_Release(This->olecmd);
|
|
This->olecmd = NULL;
|
|
}
|
|
}
|
|
|
|
static inline DocHost *impl_from_IOleCommandTarget(IOleCommandTarget *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, DocHost, IOleCommandTarget_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI ClOleCommandTarget_QueryInterface(IOleCommandTarget *iface,
|
|
REFIID riid, void **ppv)
|
|
{
|
|
DocHost *This = impl_from_IOleCommandTarget(iface);
|
|
return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv);
|
|
}
|
|
|
|
static ULONG WINAPI ClOleCommandTarget_AddRef(IOleCommandTarget *iface)
|
|
{
|
|
DocHost *This = impl_from_IOleCommandTarget(iface);
|
|
return IOleClientSite_AddRef(&This->IOleClientSite_iface);
|
|
}
|
|
|
|
static ULONG WINAPI ClOleCommandTarget_Release(IOleCommandTarget *iface)
|
|
{
|
|
DocHost *This = impl_from_IOleCommandTarget(iface);
|
|
return IOleClientSite_Release(&This->IOleClientSite_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI ClOleCommandTarget_QueryStatus(IOleCommandTarget *iface,
|
|
const GUID *pguidCmdGroup, ULONG cCmds, OLECMD prgCmds[], OLECMDTEXT *pCmdText)
|
|
{
|
|
DocHost *This = impl_from_IOleCommandTarget(iface);
|
|
ULONG i;
|
|
|
|
TRACE("(%p)->(%s %u %p %p)\n", This, debugstr_guid(pguidCmdGroup), cCmds, prgCmds,
|
|
pCmdText);
|
|
for(i=0; prgCmds && i < cCmds; i++)
|
|
TRACE("unsupported command %u (%x)\n", prgCmds[i].cmdID, prgCmds[i].cmdf);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI ClOleCommandTarget_Exec(IOleCommandTarget *iface,
|
|
const GUID *pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt, VARIANT *pvaIn,
|
|
VARIANT *pvaOut)
|
|
{
|
|
DocHost *This = impl_from_IOleCommandTarget(iface);
|
|
|
|
TRACE("(%p)->(%s %d %d %s %s)\n", This, debugstr_guid(pguidCmdGroup), nCmdID, nCmdexecopt,
|
|
debugstr_variant(pvaIn), debugstr_variant(pvaOut));
|
|
|
|
if(!pguidCmdGroup) {
|
|
switch(nCmdID) {
|
|
case OLECMDID_UPDATECOMMANDS:
|
|
if(!This->olecmd)
|
|
return E_NOTIMPL;
|
|
return IOleCommandTarget_Exec(This->olecmd, pguidCmdGroup, nCmdID, nCmdexecopt, pvaIn, pvaOut);
|
|
case OLECMDID_SETDOWNLOADSTATE:
|
|
if(This->olecmd)
|
|
return IOleCommandTarget_Exec(This->olecmd, pguidCmdGroup, nCmdID, nCmdexecopt, pvaIn, pvaOut);
|
|
|
|
if(!pvaIn || V_VT(pvaIn) != VT_I4)
|
|
return E_INVALIDARG;
|
|
|
|
notify_download_state(This, V_I4(pvaIn));
|
|
return S_OK;
|
|
default:
|
|
TRACE("Unimplemented cmdid %d\n", nCmdID);
|
|
return E_NOTIMPL;
|
|
}
|
|
}
|
|
|
|
if(IsEqualGUID(pguidCmdGroup, &CGID_DocHostCmdPriv)) {
|
|
switch(nCmdID) {
|
|
case DOCHOST_DOCCANNAVIGATE:
|
|
if(!pvaIn || V_VT(pvaIn) != VT_UNKNOWN)
|
|
return E_INVALIDARG;
|
|
|
|
if(This->doc_navigate)
|
|
IUnknown_Release(This->doc_navigate);
|
|
IUnknown_AddRef(V_UNKNOWN(pvaIn));
|
|
This->doc_navigate = V_UNKNOWN(pvaIn);
|
|
return S_OK;
|
|
|
|
case 1: {
|
|
IHTMLWindow2 *win2;
|
|
SAFEARRAY *sa = V_ARRAY(pvaIn);
|
|
VARIANT status_code, url, htmlwindow;
|
|
LONG ind;
|
|
HRESULT hres;
|
|
|
|
if(V_VT(pvaIn) != VT_ARRAY || !sa || (SafeArrayGetDim(sa) != 1))
|
|
return E_INVALIDARG;
|
|
|
|
ind = 0;
|
|
hres = SafeArrayGetElement(sa, &ind, &status_code);
|
|
if(FAILED(hres) || V_VT(&status_code)!=VT_I4)
|
|
return E_INVALIDARG;
|
|
|
|
ind = 1;
|
|
hres = SafeArrayGetElement(sa, &ind, &url);
|
|
if(FAILED(hres) || V_VT(&url)!=VT_BSTR)
|
|
return E_INVALIDARG;
|
|
|
|
ind = 3;
|
|
hres = SafeArrayGetElement(sa, &ind, &htmlwindow);
|
|
if(FAILED(hres) || V_VT(&htmlwindow)!=VT_UNKNOWN || !V_UNKNOWN(&htmlwindow))
|
|
return E_INVALIDARG;
|
|
|
|
hres = IUnknown_QueryInterface(V_UNKNOWN(&htmlwindow), &IID_IHTMLWindow2, (void**)&win2);
|
|
if(FAILED(hres))
|
|
return E_INVALIDARG;
|
|
|
|
handle_navigation_error(This, V_I4(&status_code), V_BSTR(&url), win2);
|
|
IHTMLWindow2_Release(win2);
|
|
return S_OK;
|
|
}
|
|
|
|
default:
|
|
TRACE("unsupported command %d of CGID_DocHostCmdPriv\n", nCmdID);
|
|
return E_NOTIMPL;
|
|
}
|
|
}
|
|
|
|
if(IsEqualGUID(pguidCmdGroup, &CGID_Explorer)) {
|
|
switch(nCmdID) {
|
|
case CMDID_EXPLORER_UPDATEHISTORY:
|
|
update_travellog(This);
|
|
update_navigation_commands(This);
|
|
return S_OK;
|
|
|
|
default:
|
|
TRACE("Unimplemented cmdid %d of CGID_Explorer\n", nCmdID);
|
|
return E_NOTIMPL;
|
|
}
|
|
}
|
|
|
|
if(IsEqualGUID(pguidCmdGroup, &CGID_ShellDocView)) {
|
|
switch(nCmdID) {
|
|
default:
|
|
TRACE("Unimplemented cmdid %d of CGID_ShellDocView\n", nCmdID);
|
|
return E_NOTIMPL;
|
|
}
|
|
}
|
|
|
|
if(IsEqualGUID(&CGID_DocHostCommandHandler, pguidCmdGroup)) {
|
|
if(!This->olecmd)
|
|
return E_NOTIMPL;
|
|
return IOleCommandTarget_Exec(This->olecmd, pguidCmdGroup, nCmdID, nCmdexecopt, pvaIn, pvaOut);
|
|
}
|
|
|
|
TRACE("Unimplemented cmdid %d of group %s\n", nCmdID, debugstr_guid(pguidCmdGroup));
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IOleCommandTargetVtbl OleCommandTargetVtbl = {
|
|
ClOleCommandTarget_QueryInterface,
|
|
ClOleCommandTarget_AddRef,
|
|
ClOleCommandTarget_Release,
|
|
ClOleCommandTarget_QueryStatus,
|
|
ClOleCommandTarget_Exec
|
|
};
|
|
|
|
static inline DocHost *impl_from_IDocHostUIHandler2(IDocHostUIHandler2 *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, DocHost, IDocHostUIHandler2_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_QueryInterface(IDocHostUIHandler2 *iface,
|
|
REFIID riid, void **ppv)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv);
|
|
}
|
|
|
|
static ULONG WINAPI DocHostUIHandler_AddRef(IDocHostUIHandler2 *iface)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
return IOleClientSite_AddRef(&This->IOleClientSite_iface);
|
|
}
|
|
|
|
static ULONG WINAPI DocHostUIHandler_Release(IDocHostUIHandler2 *iface)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
return IOleClientSite_Release(&This->IOleClientSite_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_ShowContextMenu(IDocHostUIHandler2 *iface,
|
|
DWORD dwID, POINT *ppt, IUnknown *pcmdtReserved, IDispatch *pdispReserved)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
HRESULT hres;
|
|
|
|
TRACE("(%p)->(%d %p %p %p)\n", This, dwID, ppt, pcmdtReserved, pdispReserved);
|
|
|
|
if(This->hostui) {
|
|
hres = IDocHostUIHandler_ShowContextMenu(This->hostui, dwID, ppt, pcmdtReserved,
|
|
pdispReserved);
|
|
if(hres == S_OK)
|
|
return S_OK;
|
|
}
|
|
|
|
FIXME("default action not implemented\n");
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_GetHostInfo(IDocHostUIHandler2 *iface,
|
|
DOCHOSTUIINFO *pInfo)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
HRESULT hres;
|
|
|
|
TRACE("(%p)->(%p)\n", This, pInfo);
|
|
|
|
if(This->hostui) {
|
|
hres = IDocHostUIHandler_GetHostInfo(This->hostui, pInfo);
|
|
if(SUCCEEDED(hres))
|
|
return hres;
|
|
}
|
|
|
|
pInfo->dwFlags = DOCHOSTUIFLAG_DISABLE_HELP_MENU | DOCHOSTUIFLAG_OPENNEWWIN
|
|
| DOCHOSTUIFLAG_URL_ENCODING_ENABLE_UTF8 | DOCHOSTUIFLAG_ENABLE_INPLACE_NAVIGATION
|
|
| DOCHOSTUIFLAG_IME_ENABLE_RECONVERSION;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_ShowUI(IDocHostUIHandler2 *iface, DWORD dwID,
|
|
IOleInPlaceActiveObject *pActiveObject, IOleCommandTarget *pCommandTarget,
|
|
IOleInPlaceFrame *pFrame, IOleInPlaceUIWindow *pDoc)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
FIXME("(%p)->(%d %p %p %p %p)\n", This, dwID, pActiveObject, pCommandTarget,
|
|
pFrame, pDoc);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_HideUI(IDocHostUIHandler2 *iface)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
FIXME("(%p)\n", This);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_UpdateUI(IDocHostUIHandler2 *iface)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
|
|
TRACE("(%p)\n", This);
|
|
|
|
if(!This->hostui)
|
|
return S_FALSE;
|
|
|
|
return IDocHostUIHandler_UpdateUI(This->hostui);
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_EnableModeless(IDocHostUIHandler2 *iface,
|
|
BOOL fEnable)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
FIXME("(%p)->(%x)\n", This, fEnable);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_OnDocWindowActivate(IDocHostUIHandler2 *iface,
|
|
BOOL fActivate)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
FIXME("(%p)->(%x)\n", This, fActivate);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_OnFrameWindowActivate(IDocHostUIHandler2 *iface,
|
|
BOOL fActivate)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
FIXME("(%p)->(%x)\n", This, fActivate);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_ResizeBorder(IDocHostUIHandler2 *iface,
|
|
LPCRECT prcBorder, IOleInPlaceUIWindow *pUIWindow, BOOL fRameWindow)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
FIXME("(%p)->(%p %p %X)\n", This, prcBorder, pUIWindow, fRameWindow);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_TranslateAccelerator(IDocHostUIHandler2 *iface,
|
|
LPMSG lpMsg, const GUID *pguidCmdGroup, DWORD nCmdID)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
HRESULT hr = S_FALSE;
|
|
TRACE("(%p)->(%p %p %d)\n", This, lpMsg, pguidCmdGroup, nCmdID);
|
|
|
|
if(This->hostui)
|
|
hr = IDocHostUIHandler_TranslateAccelerator(This->hostui, lpMsg, pguidCmdGroup, nCmdID);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_GetOptionKeyPath(IDocHostUIHandler2 *iface,
|
|
LPOLESTR *pchKey, DWORD dw)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
|
|
TRACE("(%p)->(%p %d)\n", This, pchKey, dw);
|
|
|
|
if(This->hostui)
|
|
return IDocHostUIHandler_GetOptionKeyPath(This->hostui, pchKey, dw);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_GetDropTarget(IDocHostUIHandler2 *iface,
|
|
IDropTarget *pDropTarget, IDropTarget **ppDropTarget)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
FIXME("(%p)\n", This);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_GetExternal(IDocHostUIHandler2 *iface,
|
|
IDispatch **ppDispatch)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
|
|
TRACE("(%p)->(%p)\n", This, ppDispatch);
|
|
|
|
if(This->hostui)
|
|
return IDocHostUIHandler_GetExternal(This->hostui, ppDispatch);
|
|
|
|
if(!This->shell_ui_helper) {
|
|
HRESULT hres;
|
|
|
|
hres = create_shell_ui_helper(&This->shell_ui_helper);
|
|
if(FAILED(hres))
|
|
return hres;
|
|
}
|
|
|
|
*ppDispatch = (IDispatch*)This->shell_ui_helper;
|
|
IDispatch_AddRef(*ppDispatch);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_TranslateUrl(IDocHostUIHandler2 *iface,
|
|
DWORD dwTranslate, OLECHAR *pchURLIn, OLECHAR **ppchURLOut)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
|
|
TRACE("(%p)->(%d %s %p)\n", This, dwTranslate, debugstr_w(pchURLIn), ppchURLOut);
|
|
|
|
if(This->hostui)
|
|
return IDocHostUIHandler_TranslateUrl(This->hostui, dwTranslate,
|
|
pchURLIn, ppchURLOut);
|
|
|
|
return S_FALSE;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_FilterDataObject(IDocHostUIHandler2 *iface,
|
|
IDataObject *pDO, IDataObject **ppDORet)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
FIXME("(%p)->(%p %p)\n", This, pDO, ppDORet);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI DocHostUIHandler_GetOverrideKeyPath(IDocHostUIHandler2 *iface,
|
|
LPOLESTR *pchKey, DWORD dw)
|
|
{
|
|
DocHost *This = impl_from_IDocHostUIHandler2(iface);
|
|
IDocHostUIHandler2 *handler;
|
|
HRESULT hres;
|
|
|
|
TRACE("(%p)->(%p %d)\n", This, pchKey, dw);
|
|
|
|
if(!This->hostui)
|
|
return S_OK;
|
|
|
|
hres = IDocHostUIHandler_QueryInterface(This->hostui, &IID_IDocHostUIHandler2,
|
|
(void**)&handler);
|
|
if(SUCCEEDED(hres)) {
|
|
hres = IDocHostUIHandler2_GetOverrideKeyPath(handler, pchKey, dw);
|
|
IDocHostUIHandler2_Release(handler);
|
|
return hres;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static const IDocHostUIHandler2Vtbl DocHostUIHandler2Vtbl = {
|
|
DocHostUIHandler_QueryInterface,
|
|
DocHostUIHandler_AddRef,
|
|
DocHostUIHandler_Release,
|
|
DocHostUIHandler_ShowContextMenu,
|
|
DocHostUIHandler_GetHostInfo,
|
|
DocHostUIHandler_ShowUI,
|
|
DocHostUIHandler_HideUI,
|
|
DocHostUIHandler_UpdateUI,
|
|
DocHostUIHandler_EnableModeless,
|
|
DocHostUIHandler_OnDocWindowActivate,
|
|
DocHostUIHandler_OnFrameWindowActivate,
|
|
DocHostUIHandler_ResizeBorder,
|
|
DocHostUIHandler_TranslateAccelerator,
|
|
DocHostUIHandler_GetOptionKeyPath,
|
|
DocHostUIHandler_GetDropTarget,
|
|
DocHostUIHandler_GetExternal,
|
|
DocHostUIHandler_TranslateUrl,
|
|
DocHostUIHandler_FilterDataObject,
|
|
DocHostUIHandler_GetOverrideKeyPath
|
|
};
|
|
|
|
static inline DocHost *impl_from_IPropertyNotifySink(IPropertyNotifySink *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, DocHost, IPropertyNotifySink_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI PropertyNotifySink_QueryInterface(IPropertyNotifySink *iface,
|
|
REFIID riid, void **ppv)
|
|
{
|
|
DocHost *This = impl_from_IPropertyNotifySink(iface);
|
|
return IOleClientSite_QueryInterface(&This->IOleClientSite_iface, riid, ppv);
|
|
}
|
|
|
|
static ULONG WINAPI PropertyNotifySink_AddRef(IPropertyNotifySink *iface)
|
|
{
|
|
DocHost *This = impl_from_IPropertyNotifySink(iface);
|
|
return IOleClientSite_AddRef(&This->IOleClientSite_iface);
|
|
}
|
|
|
|
static ULONG WINAPI PropertyNotifySink_Release(IPropertyNotifySink *iface)
|
|
{
|
|
DocHost *This = impl_from_IPropertyNotifySink(iface);
|
|
return IOleClientSite_Release(&This->IOleClientSite_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI PropertyNotifySink_OnChanged(IPropertyNotifySink *iface, DISPID dispID)
|
|
{
|
|
DocHost *This = impl_from_IPropertyNotifySink(iface);
|
|
|
|
TRACE("(%p)->(%d)\n", This, dispID);
|
|
|
|
switch(dispID) {
|
|
case DISPID_READYSTATE: {
|
|
READYSTATE ready_state;
|
|
HRESULT hres;
|
|
|
|
hres = get_doc_ready_state(This, &ready_state);
|
|
if(FAILED(hres))
|
|
return hres;
|
|
|
|
if(ready_state == READYSTATE_COMPLETE && !This->doc_navigate)
|
|
advise_prop_notif(This, FALSE);
|
|
|
|
update_ready_state(This, ready_state);
|
|
break;
|
|
}
|
|
default:
|
|
FIXME("unimplemented dispid %d\n", dispID);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI PropertyNotifySink_OnRequestEdit(IPropertyNotifySink *iface, DISPID dispID)
|
|
{
|
|
DocHost *This = impl_from_IPropertyNotifySink(iface);
|
|
FIXME("(%p)->(%d)\n", This, dispID);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IPropertyNotifySinkVtbl PropertyNotifySinkVtbl = {
|
|
PropertyNotifySink_QueryInterface,
|
|
PropertyNotifySink_AddRef,
|
|
PropertyNotifySink_Release,
|
|
PropertyNotifySink_OnChanged,
|
|
PropertyNotifySink_OnRequestEdit
|
|
};
|
|
|
|
void DocHost_Init(DocHost *This, IWebBrowser2 *wb, const IDocHostContainerVtbl* container)
|
|
{
|
|
This->IDocHostUIHandler2_iface.lpVtbl = &DocHostUIHandler2Vtbl;
|
|
This->IOleCommandTarget_iface.lpVtbl = &OleCommandTargetVtbl;
|
|
This->IPropertyNotifySink_iface.lpVtbl = &PropertyNotifySinkVtbl;
|
|
|
|
This->wb = wb;
|
|
This->container_vtbl = container;
|
|
|
|
This->ready_state = READYSTATE_UNINITIALIZED;
|
|
list_init(&This->task_queue);
|
|
|
|
This->travellog.loading_pos = -1;
|
|
|
|
DocHost_ClientSite_Init(This);
|
|
DocHost_Frame_Init(This);
|
|
|
|
ConnectionPointContainer_Init(&This->cps, (IUnknown*)wb);
|
|
IEHTMLWindow_Init(This);
|
|
NewWindowManager_Init(This);
|
|
}
|
|
|
|
void DocHost_Release(DocHost *This)
|
|
{
|
|
if(This->shell_ui_helper)
|
|
IShellUIHelper2_Release(This->shell_ui_helper);
|
|
|
|
abort_dochost_tasks(This, NULL);
|
|
release_dochost_client(This);
|
|
DocHost_ClientSite_Release(This);
|
|
|
|
ConnectionPointContainer_Destroy(&This->cps);
|
|
|
|
while(This->travellog.length)
|
|
free_travellog_entry(This->travellog.log + --This->travellog.length);
|
|
heap_free(This->travellog.log);
|
|
|
|
heap_free(This->url);
|
|
}
|