| /* |
| * 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=%d\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=%d\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: %08x\n", nsres); |
| return E_FAIL; |
| } |
| |
| nsres = nsISelection_Collapse(This->nsselection, (nsIDOMNode*)nsbody, 0); |
| nsIDOMHTMLElement_Release(nsbody); |
| if(NS_FAILED(nsres)) |
| ERR("Collapse failed: %08x\n", nsres); |
| }else if(nsrange_cnt > 1) { |
| FIXME("range_cnt = %d\n", nsrange_cnt); |
| } |
| |
| nsres = nsISelection_GetRangeAt(This->nsselection, 0, &nsrange); |
| if(NS_FAILED(nsres)) |
| ERR("GetRangeAt failed: %08x\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; |
| |
| static const WCHAR wszNone[] = {'N','o','n','e',0}; |
| static const WCHAR wszText[] = {'T','e','x','t',0}; |
| |
| TRACE("(%p)->(%p)\n", This, p); |
| |
| if(This->nsselection) |
| nsISelection_GetIsCollapsed(This->nsselection, &collapsed); |
| |
| *p = SysAllocString(collapsed ? wszNone : wszText); /* 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); |
| |
| static const WCHAR undefinedW[] = {'u','n','d','e','f','i','n','e','d',0}; |
| |
| FIXME("(%p)->(%p) semi-stub\n", This, p); |
| |
| /* FIXME: We should try to use ISelectionServicesListener::GetTypeDetail here. */ |
| *p = SysAllocString(undefinedW); |
| 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 = { |
| 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_dispex(&selection->dispex, (IUnknown*)&selection->IHTMLSelectionObject_iface, &HTMLSelectionObject_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; |
| |
| LIST_FOR_EACH_ENTRY(iter, &This->selection_list, HTMLSelectionObject, entry) { |
| iter->doc = NULL; |
| } |
| } |