wine/dlls/mshtml/selection.c

278 lines
8.8 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;
nsISelection *nsselection;
HTMLDocumentNode *doc;
struct list entry;
} HTMLSelectionObject;
static inline HTMLSelectionObject *impl_from_IHTMLSelectionObject(IHTMLSelectionObject *iface)
{
return CONTAINING_RECORD(iface, HTMLSelectionObject, IHTMLSelectionObject_iface);
}
DISPEX_IDISPATCH_IMPL(HTMLSelectionObject, IHTMLSelectionObject,
impl_from_IHTMLSelectionObject(iface)->dispex)
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->dom_document) {
WARN("dom_document is NULL\n");
return E_UNEXPECTED;
}
if(!This->doc->html_document) {
FIXME("Not implemented for XML document\n");
return E_NOTIMPL;
}
nsres = nsIDOMHTMLDocument_GetBody(This->doc->html_document, &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);
}
DISPEX_IDISPATCH_IMPL(HTMLSelectionObject2, IHTMLSelectionObject2,
impl_from_IHTMLSelectionObject2(iface)->dispex)
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 inline HTMLSelectionObject *impl_from_DispatchEx(DispatchEx *iface)
{
return CONTAINING_RECORD(iface, HTMLSelectionObject, dispex);
}
static void *HTMLSelectionObject_query_interface(DispatchEx *dispex, REFIID riid)
{
HTMLSelectionObject *This = impl_from_DispatchEx(dispex);
if(IsEqualGUID(&IID_IHTMLSelectionObject, riid))
return &This->IHTMLSelectionObject_iface;
if(IsEqualGUID(&IID_IHTMLSelectionObject2, riid))
return &This->IHTMLSelectionObject2_iface;
return NULL;
}
static void HTMLSelectionObject_traverse(DispatchEx *dispex, nsCycleCollectionTraversalCallback *cb)
{
HTMLSelectionObject *This = impl_from_DispatchEx(dispex);
if(This->nsselection)
note_cc_edge((nsISupports*)This->nsselection, "nsselection", cb);
}
static void HTMLSelectionObject_unlink(DispatchEx *dispex)
{
HTMLSelectionObject *This = impl_from_DispatchEx(dispex);
unlink_ref(&This->nsselection);
if(This->doc) {
This->doc = NULL;
list_remove(&This->entry);
}
}
static void HTMLSelectionObject_destructor(DispatchEx *dispex)
{
HTMLSelectionObject *This = impl_from_DispatchEx(dispex);
free(This);
}
static const dispex_static_data_vtbl_t HTMLSelectionObject_dispex_vtbl = {
.query_interface = HTMLSelectionObject_query_interface,
.destructor = HTMLSelectionObject_destructor,
.traverse = HTMLSelectionObject_traverse,
.unlink = HTMLSelectionObject_unlink
};
static const tid_t HTMLSelectionObject_iface_tids[] = {
IHTMLSelectionObject_tid,
IHTMLSelectionObject2_tid,
0
};
static dispex_static_data_t HTMLSelectionObject_dispex = {
"MSSelection",
&HTMLSelectionObject_dispex_vtbl,
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 = malloc(sizeof(HTMLSelectionObject));
if(!selection)
return E_OUTOFMEMORY;
init_dispatch(&selection->dispex, &HTMLSelectionObject_dispex, dispex_compat_mode(&doc->node.event_target.dispex));
selection->IHTMLSelectionObject_iface.lpVtbl = &HTMLSelectionObjectVtbl;
selection->IHTMLSelectionObject2_iface.lpVtbl = &HTMLSelectionObject2Vtbl;
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);
}
}