371 lines
12 KiB
C
371 lines
12 KiB
C
/*
|
|
* Copyright 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 <stdarg.h>
|
|
|
|
#define COBJMACROS
|
|
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "winuser.h"
|
|
#include "ole2.h"
|
|
|
|
#include "wine/debug.h"
|
|
|
|
#include "mshtml_private.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(mshtml);
|
|
|
|
typedef struct {
|
|
DispatchEx dispex;
|
|
IHTMLSelectionObject IHTMLSelectionObject_iface;
|
|
IHTMLSelectionObject2 IHTMLSelectionObject2_iface;
|
|
|
|
LONG ref;
|
|
|
|
nsISelection *nsselection;
|
|
HTMLDocumentNode *doc;
|
|
|
|
struct list entry;
|
|
} HTMLSelectionObject;
|
|
|
|
static inline HTMLSelectionObject *impl_from_IHTMLSelectionObject(IHTMLSelectionObject *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, HTMLSelectionObject, IHTMLSelectionObject_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject_QueryInterface(IHTMLSelectionObject *iface,
|
|
REFIID riid, void **ppv)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject(iface);
|
|
|
|
TRACE("(%p)->(%s %p)\n", This, debugstr_mshtml_guid(riid), ppv);
|
|
|
|
if(IsEqualGUID(&IID_IUnknown, riid)) {
|
|
*ppv = &This->IHTMLSelectionObject_iface;
|
|
}else if(IsEqualGUID(&IID_IDispatch, riid)) {
|
|
*ppv = &This->IHTMLSelectionObject_iface;
|
|
}else if(IsEqualGUID(&IID_IHTMLSelectionObject, riid)) {
|
|
*ppv = &This->IHTMLSelectionObject_iface;
|
|
}else if(IsEqualGUID(&IID_IHTMLSelectionObject2, riid)) {
|
|
*ppv = &This->IHTMLSelectionObject2_iface;
|
|
}else if(dispex_query_interface(&This->dispex, riid, ppv)) {
|
|
return *ppv ? S_OK : E_NOINTERFACE;
|
|
}else {
|
|
*ppv = NULL;
|
|
WARN("(%p)->(%s %p)\n", This, debugstr_guid(riid), ppv);
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
IUnknown_AddRef((IUnknown*)*ppv);
|
|
return S_OK;
|
|
}
|
|
|
|
static ULONG WINAPI HTMLSelectionObject_AddRef(IHTMLSelectionObject *iface)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject(iface);
|
|
LONG ref = InterlockedIncrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%ld\n", This, ref);
|
|
|
|
return ref;
|
|
}
|
|
|
|
static ULONG WINAPI HTMLSelectionObject_Release(IHTMLSelectionObject *iface)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject(iface);
|
|
LONG ref = InterlockedDecrement(&This->ref);
|
|
|
|
TRACE("(%p) ref=%ld\n", This, ref);
|
|
|
|
if(!ref) {
|
|
if(This->nsselection)
|
|
nsISelection_Release(This->nsselection);
|
|
if(This->doc)
|
|
list_remove(&This->entry);
|
|
release_dispex(&This->dispex);
|
|
heap_free(This);
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject_GetTypeInfoCount(IHTMLSelectionObject *iface, UINT *pctinfo)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject(iface);
|
|
|
|
return IDispatchEx_GetTypeInfoCount(&This->dispex.IDispatchEx_iface, pctinfo);
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject_GetTypeInfo(IHTMLSelectionObject *iface, UINT iTInfo,
|
|
LCID lcid, ITypeInfo **ppTInfo)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject(iface);
|
|
|
|
return IDispatchEx_GetTypeInfo(&This->dispex.IDispatchEx_iface, iTInfo, lcid, ppTInfo);
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject_GetIDsOfNames(IHTMLSelectionObject *iface, REFIID riid,
|
|
LPOLESTR *rgszNames, UINT cNames,
|
|
LCID lcid, DISPID *rgDispId)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject(iface);
|
|
|
|
return IDispatchEx_GetIDsOfNames(&This->dispex.IDispatchEx_iface, riid, rgszNames,
|
|
cNames, lcid, rgDispId);
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject_Invoke(IHTMLSelectionObject *iface, DISPID dispIdMember,
|
|
REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams,
|
|
VARIANT *pVarResult, EXCEPINFO *pExcepInfo, UINT *puArgErr)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject(iface);
|
|
|
|
|
|
return IDispatchEx_Invoke(&This->dispex.IDispatchEx_iface, dispIdMember, riid,
|
|
lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject_createRange(IHTMLSelectionObject *iface, IDispatch **range)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject(iface);
|
|
IHTMLTxtRange *range_obj = NULL;
|
|
nsIDOMRange *nsrange = NULL;
|
|
HRESULT hres;
|
|
|
|
TRACE("(%p)->(%p)\n", This, range);
|
|
|
|
if(This->nsselection) {
|
|
LONG nsrange_cnt = 0;
|
|
nsresult nsres;
|
|
|
|
nsISelection_GetRangeCount(This->nsselection, &nsrange_cnt);
|
|
if(!nsrange_cnt) {
|
|
nsIDOMHTMLElement *nsbody = NULL;
|
|
|
|
TRACE("nsrange_cnt = 0\n");
|
|
|
|
if(!This->doc->nsdoc) {
|
|
WARN("nsdoc is NULL\n");
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
nsres = nsIDOMHTMLDocument_GetBody(This->doc->nsdoc, &nsbody);
|
|
if(NS_FAILED(nsres) || !nsbody) {
|
|
ERR("Could not get body: %08lx\n", nsres);
|
|
return E_FAIL;
|
|
}
|
|
|
|
nsres = nsISelection_Collapse(This->nsselection, (nsIDOMNode*)nsbody, 0);
|
|
nsIDOMHTMLElement_Release(nsbody);
|
|
if(NS_FAILED(nsres))
|
|
ERR("Collapse failed: %08lx\n", nsres);
|
|
}else if(nsrange_cnt > 1) {
|
|
FIXME("range_cnt = %ld\n", nsrange_cnt);
|
|
}
|
|
|
|
nsres = nsISelection_GetRangeAt(This->nsselection, 0, &nsrange);
|
|
if(NS_FAILED(nsres))
|
|
ERR("GetRangeAt failed: %08lx\n", nsres);
|
|
}
|
|
|
|
hres = HTMLTxtRange_Create(This->doc, nsrange, &range_obj);
|
|
|
|
if (nsrange) nsIDOMRange_Release(nsrange);
|
|
*range = (IDispatch*)range_obj;
|
|
return hres;
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject_empty(IHTMLSelectionObject *iface)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject(iface);
|
|
FIXME("(%p)\n", This);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject_clear(IHTMLSelectionObject *iface)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject(iface);
|
|
FIXME("(%p)\n", This);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject_get_type(IHTMLSelectionObject *iface, BSTR *p)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject(iface);
|
|
cpp_bool collapsed = TRUE;
|
|
|
|
TRACE("(%p)->(%p)\n", This, p);
|
|
|
|
if(This->nsselection)
|
|
nsISelection_GetIsCollapsed(This->nsselection, &collapsed);
|
|
|
|
*p = SysAllocString(collapsed ? L"None" : L"Text"); /* FIXME: control */
|
|
TRACE("ret %s\n", debugstr_w(*p));
|
|
return S_OK;
|
|
}
|
|
|
|
static const IHTMLSelectionObjectVtbl HTMLSelectionObjectVtbl = {
|
|
HTMLSelectionObject_QueryInterface,
|
|
HTMLSelectionObject_AddRef,
|
|
HTMLSelectionObject_Release,
|
|
HTMLSelectionObject_GetTypeInfoCount,
|
|
HTMLSelectionObject_GetTypeInfo,
|
|
HTMLSelectionObject_GetIDsOfNames,
|
|
HTMLSelectionObject_Invoke,
|
|
HTMLSelectionObject_createRange,
|
|
HTMLSelectionObject_empty,
|
|
HTMLSelectionObject_clear,
|
|
HTMLSelectionObject_get_type
|
|
};
|
|
|
|
static inline HTMLSelectionObject *impl_from_IHTMLSelectionObject2(IHTMLSelectionObject2 *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, HTMLSelectionObject, IHTMLSelectionObject2_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject2_QueryInterface(IHTMLSelectionObject2 *iface, REFIID riid, void **ppv)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject2(iface);
|
|
|
|
return IHTMLSelectionObject_QueryInterface(&This->IHTMLSelectionObject_iface, riid, ppv);
|
|
}
|
|
|
|
static ULONG WINAPI HTMLSelectionObject2_AddRef(IHTMLSelectionObject2 *iface)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject2(iface);
|
|
|
|
return IHTMLSelectionObject_AddRef(&This->IHTMLSelectionObject_iface);
|
|
}
|
|
|
|
static ULONG WINAPI HTMLSelectionObject2_Release(IHTMLSelectionObject2 *iface)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject2(iface);
|
|
|
|
return IHTMLSelectionObject_Release(&This->IHTMLSelectionObject_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject2_GetTypeInfoCount(IHTMLSelectionObject2 *iface, UINT *pctinfo)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject2(iface);
|
|
|
|
return IDispatchEx_GetTypeInfoCount(&This->dispex.IDispatchEx_iface, pctinfo);
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject2_GetTypeInfo(IHTMLSelectionObject2 *iface, UINT iTInfo,
|
|
LCID lcid, ITypeInfo **ppTInfo)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject2(iface);
|
|
|
|
return IDispatchEx_GetTypeInfo(&This->dispex.IDispatchEx_iface, iTInfo, lcid, ppTInfo);
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject2_GetIDsOfNames(IHTMLSelectionObject2 *iface, REFIID riid,
|
|
LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject2(iface);
|
|
|
|
return IDispatchEx_GetIDsOfNames(&This->dispex.IDispatchEx_iface, riid, rgszNames,
|
|
cNames, lcid, rgDispId);
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject2_Invoke(IHTMLSelectionObject2 *iface, DISPID dispIdMember,
|
|
REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS *pDispParams, VARIANT *pVarResult,
|
|
EXCEPINFO *pExcepInfo, UINT *puArgErr)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject2(iface);
|
|
|
|
return IDispatchEx_Invoke(&This->dispex.IDispatchEx_iface, dispIdMember, riid,
|
|
lcid, wFlags, pDispParams, pVarResult, pExcepInfo, puArgErr);
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject2_createRangeCollection(IHTMLSelectionObject2 *iface, IDispatch **rangeCollection)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject2(iface);
|
|
FIXME("(%p)->(%p)\n", This, rangeCollection);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI HTMLSelectionObject2_get_typeDetail(IHTMLSelectionObject2 *iface, BSTR *p)
|
|
{
|
|
HTMLSelectionObject *This = impl_from_IHTMLSelectionObject2(iface);
|
|
|
|
FIXME("(%p)->(%p) semi-stub\n", This, p);
|
|
|
|
/* FIXME: We should try to use ISelectionServicesListener::GetTypeDetail here. */
|
|
*p = SysAllocString(L"undefined");
|
|
return *p ? S_OK : E_OUTOFMEMORY;
|
|
}
|
|
|
|
static const IHTMLSelectionObject2Vtbl HTMLSelectionObject2Vtbl = {
|
|
HTMLSelectionObject2_QueryInterface,
|
|
HTMLSelectionObject2_AddRef,
|
|
HTMLSelectionObject2_Release,
|
|
HTMLSelectionObject2_GetTypeInfoCount,
|
|
HTMLSelectionObject2_GetTypeInfo,
|
|
HTMLSelectionObject2_GetIDsOfNames,
|
|
HTMLSelectionObject2_Invoke,
|
|
HTMLSelectionObject2_createRangeCollection,
|
|
HTMLSelectionObject2_get_typeDetail
|
|
};
|
|
|
|
static const tid_t HTMLSelectionObject_iface_tids[] = {
|
|
IHTMLSelectionObject_tid,
|
|
IHTMLSelectionObject2_tid,
|
|
0
|
|
};
|
|
static dispex_static_data_t HTMLSelectionObject_dispex = {
|
|
L"MSSelection",
|
|
NULL,
|
|
IHTMLSelectionObject_tid, /* FIXME: We have a test for that, but it doesn't expose IHTMLSelectionObject2 iface. */
|
|
HTMLSelectionObject_iface_tids
|
|
};
|
|
|
|
HRESULT HTMLSelectionObject_Create(HTMLDocumentNode *doc, nsISelection *nsselection, IHTMLSelectionObject **ret)
|
|
{
|
|
HTMLSelectionObject *selection;
|
|
|
|
selection = heap_alloc(sizeof(HTMLSelectionObject));
|
|
if(!selection)
|
|
return E_OUTOFMEMORY;
|
|
|
|
init_dispatch(&selection->dispex, (IUnknown*)&selection->IHTMLSelectionObject_iface,
|
|
&HTMLSelectionObject_dispex, dispex_compat_mode(&doc->node.event_target.dispex));
|
|
|
|
selection->IHTMLSelectionObject_iface.lpVtbl = &HTMLSelectionObjectVtbl;
|
|
selection->IHTMLSelectionObject2_iface.lpVtbl = &HTMLSelectionObject2Vtbl;
|
|
selection->ref = 1;
|
|
selection->nsselection = nsselection; /* We shouldn't call AddRef here */
|
|
|
|
selection->doc = doc;
|
|
list_add_head(&doc->selection_list, &selection->entry);
|
|
|
|
*ret = &selection->IHTMLSelectionObject_iface;
|
|
return S_OK;
|
|
}
|
|
|
|
void detach_selection(HTMLDocumentNode *This)
|
|
{
|
|
HTMLSelectionObject *iter, *next;
|
|
|
|
LIST_FOR_EACH_ENTRY_SAFE(iter, next, &This->selection_list, HTMLSelectionObject, entry) {
|
|
iter->doc = NULL;
|
|
list_remove(&iter->entry);
|
|
}
|
|
}
|