2 * Copyright 2006 Jacek Caban for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
29 #include "wine/debug.h"
31 #include "mshtml_private.h"
32 #include "htmlevent.h"
34 WINE_DEFAULT_DEBUG_CHANNEL(mshtml
);
39 IHTMLSelectElement IHTMLSelectElement_iface
;
41 nsIDOMHTMLSelectElement
*nsselect
;
44 static inline HTMLSelectElement
*impl_from_IHTMLSelectElement(IHTMLSelectElement
*iface
)
46 return CONTAINING_RECORD(iface
, HTMLSelectElement
, IHTMLSelectElement_iface
);
49 static HRESULT
htmlselect_item(HTMLSelectElement
*This
, int i
, IDispatch
**ret
)
51 nsIDOMHTMLOptionsCollection
*nscol
;
56 nsres
= nsIDOMHTMLSelectElement_GetOptions(This
->nsselect
, &nscol
);
57 if(NS_FAILED(nsres
)) {
58 ERR("GetOptions failed: %08x\n", nsres
);
62 nsres
= nsIDOMHTMLOptionsCollection_Item(nscol
, i
, &nsnode
);
63 nsIDOMHTMLOptionsCollection_Release(nscol
);
64 if(NS_FAILED(nsres
)) {
65 ERR("Item failed: %08x\n", nsres
);
72 hres
= get_node(This
->element
.node
.doc
, nsnode
, TRUE
, &node
);
73 nsIDOMNode_Release(nsnode
);
77 *ret
= (IDispatch
*)&node
->IHTMLDOMNode_iface
;
84 static HRESULT WINAPI
HTMLSelectElement_QueryInterface(IHTMLSelectElement
*iface
,
85 REFIID riid
, void **ppv
)
87 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
89 return IHTMLDOMNode_QueryInterface(&This
->element
.node
.IHTMLDOMNode_iface
, riid
, ppv
);
92 static ULONG WINAPI
HTMLSelectElement_AddRef(IHTMLSelectElement
*iface
)
94 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
96 return IHTMLDOMNode_AddRef(&This
->element
.node
.IHTMLDOMNode_iface
);
99 static ULONG WINAPI
HTMLSelectElement_Release(IHTMLSelectElement
*iface
)
101 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
103 return IHTMLDOMNode_Release(&This
->element
.node
.IHTMLDOMNode_iface
);
106 static HRESULT WINAPI
HTMLSelectElement_GetTypeInfoCount(IHTMLSelectElement
*iface
, UINT
*pctinfo
)
108 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
110 return IDispatchEx_GetTypeInfoCount(&This
->element
.node
.dispex
.IDispatchEx_iface
, pctinfo
);
113 static HRESULT WINAPI
HTMLSelectElement_GetTypeInfo(IHTMLSelectElement
*iface
, UINT iTInfo
,
114 LCID lcid
, ITypeInfo
**ppTInfo
)
116 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
118 return IDispatchEx_GetTypeInfo(&This
->element
.node
.dispex
.IDispatchEx_iface
, iTInfo
, lcid
,
122 static HRESULT WINAPI
HTMLSelectElement_GetIDsOfNames(IHTMLSelectElement
*iface
, REFIID riid
,
123 LPOLESTR
*rgszNames
, UINT cNames
,
124 LCID lcid
, DISPID
*rgDispId
)
126 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
128 return IDispatchEx_GetIDsOfNames(&This
->element
.node
.dispex
.IDispatchEx_iface
, riid
, rgszNames
,
129 cNames
, lcid
, rgDispId
);
132 static HRESULT WINAPI
HTMLSelectElement_Invoke(IHTMLSelectElement
*iface
, DISPID dispIdMember
,
133 REFIID riid
, LCID lcid
, WORD wFlags
, DISPPARAMS
*pDispParams
,
134 VARIANT
*pVarResult
, EXCEPINFO
*pExcepInfo
, UINT
*puArgErr
)
136 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
138 return IDispatchEx_Invoke(&This
->element
.node
.dispex
.IDispatchEx_iface
, dispIdMember
, riid
,
139 lcid
, wFlags
, pDispParams
, pVarResult
, pExcepInfo
, puArgErr
);
142 static HRESULT WINAPI
HTMLSelectElement_put_size(IHTMLSelectElement
*iface
, LONG v
)
144 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
145 FIXME("(%p)->(%d)\n", This
, v
);
149 static HRESULT WINAPI
HTMLSelectElement_get_size(IHTMLSelectElement
*iface
, LONG
*p
)
151 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
152 FIXME("(%p)->(%p)\n", This
, p
);
156 static HRESULT WINAPI
HTMLSelectElement_put_multiple(IHTMLSelectElement
*iface
, VARIANT_BOOL v
)
158 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
161 TRACE("(%p)->(%x)\n", This
, v
);
163 nsres
= nsIDOMHTMLSelectElement_SetMultiple(This
->nsselect
, !!v
);
164 assert(nsres
== NS_OK
);
168 static HRESULT WINAPI
HTMLSelectElement_get_multiple(IHTMLSelectElement
*iface
, VARIANT_BOOL
*p
)
170 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
174 TRACE("(%p)->(%p)\n", This
, p
);
176 nsres
= nsIDOMHTMLSelectElement_GetMultiple(This
->nsselect
, &val
);
177 assert(nsres
== NS_OK
);
179 *p
= val
? VARIANT_TRUE
: VARIANT_FALSE
;
183 static HRESULT WINAPI
HTMLSelectElement_put_name(IHTMLSelectElement
*iface
, BSTR v
)
185 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
186 FIXME("(%p)->(%s)\n", This
, debugstr_w(v
));
190 static HRESULT WINAPI
HTMLSelectElement_get_name(IHTMLSelectElement
*iface
, BSTR
*p
)
192 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
194 const PRUnichar
*name
= NULL
;
197 TRACE("(%p)->(%p)\n", This
, p
);
199 nsAString_Init(&name_str
, NULL
);
201 nsres
= nsIDOMHTMLSelectElement_GetName(This
->nsselect
, &name_str
);
202 if(NS_SUCCEEDED(nsres
)) {
203 static const WCHAR wszGarbage
[] = {'g','a','r','b','a','g','e',0};
205 nsAString_GetData(&name_str
, &name
);
208 * Native never returns empty string here. If an element has no name,
209 * name of previous element or ramdom data is returned.
211 *p
= SysAllocString(*name
? name
: wszGarbage
);
213 ERR("GetName failed: %08x\n", nsres
);
216 nsAString_Finish(&name_str
);
218 TRACE("name=%s\n", debugstr_w(*p
));
222 static HRESULT WINAPI
HTMLSelectElement_get_options(IHTMLSelectElement
*iface
, IDispatch
**p
)
224 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
226 TRACE("(%p)->(%p)\n", This
, p
);
228 *p
= (IDispatch
*)&This
->IHTMLSelectElement_iface
;
229 IDispatch_AddRef(*p
);
233 static HRESULT WINAPI
HTMLSelectElement_put_onchange(IHTMLSelectElement
*iface
, VARIANT v
)
235 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
237 TRACE("(%p)->()\n", This
);
239 return set_node_event(&This
->element
.node
, EVENTID_CHANGE
, &v
);
242 static HRESULT WINAPI
HTMLSelectElement_get_onchange(IHTMLSelectElement
*iface
, VARIANT
*p
)
244 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
245 FIXME("(%p)->(%p)\n", This
, p
);
249 static HRESULT WINAPI
HTMLSelectElement_put_selectedIndex(IHTMLSelectElement
*iface
, LONG v
)
251 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
254 TRACE("(%p)->(%d)\n", This
, v
);
256 nsres
= nsIDOMHTMLSelectElement_SetSelectedIndex(This
->nsselect
, v
);
258 ERR("SetSelectedIndex failed: %08x\n", nsres
);
263 static HRESULT WINAPI
HTMLSelectElement_get_selectedIndex(IHTMLSelectElement
*iface
, LONG
*p
)
265 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
268 TRACE("(%p)->(%p)\n", This
, p
);
270 nsres
= nsIDOMHTMLSelectElement_GetSelectedIndex(This
->nsselect
, p
);
271 if(NS_FAILED(nsres
)) {
272 ERR("GetSelectedIndex failed: %08x\n", nsres
);
279 static HRESULT WINAPI
HTMLSelectElement_get_type(IHTMLSelectElement
*iface
, BSTR
*p
)
281 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
285 TRACE("(%p)->(%p)\n", This
, p
);
287 nsAString_Init(&type_str
, NULL
);
288 nsres
= nsIDOMHTMLSelectElement_GetType(This
->nsselect
, &type_str
);
289 return return_nsstr(nsres
, &type_str
, p
);
292 static HRESULT WINAPI
HTMLSelectElement_put_value(IHTMLSelectElement
*iface
, BSTR v
)
294 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
298 TRACE("(%p)->(%s)\n", This
, debugstr_w(v
));
300 nsAString_InitDepend(&value_str
, v
);
301 nsres
= nsIDOMHTMLSelectElement_SetValue(This
->nsselect
, &value_str
);
302 nsAString_Finish(&value_str
);
304 ERR("SetValue failed: %08x\n", nsres
);
309 static HRESULT WINAPI
HTMLSelectElement_get_value(IHTMLSelectElement
*iface
, BSTR
*p
)
311 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
315 TRACE("(%p)->(%p)\n", This
, p
);
317 nsAString_Init(&value_str
, NULL
);
318 nsres
= nsIDOMHTMLSelectElement_GetValue(This
->nsselect
, &value_str
);
319 return return_nsstr(nsres
, &value_str
, p
);
322 static HRESULT WINAPI
HTMLSelectElement_put_disabled(IHTMLSelectElement
*iface
, VARIANT_BOOL v
)
324 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
327 TRACE("(%p)->(%x)\n", This
, v
);
329 nsres
= nsIDOMHTMLSelectElement_SetDisabled(This
->nsselect
, v
!= VARIANT_FALSE
);
330 if(NS_FAILED(nsres
)) {
331 ERR("SetDisabled failed: %08x\n", nsres
);
338 static HRESULT WINAPI
HTMLSelectElement_get_disabled(IHTMLSelectElement
*iface
, VARIANT_BOOL
*p
)
340 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
341 cpp_bool disabled
= FALSE
;
344 TRACE("(%p)->(%p)\n", This
, p
);
346 nsres
= nsIDOMHTMLSelectElement_GetDisabled(This
->nsselect
, &disabled
);
347 if(NS_FAILED(nsres
)) {
348 ERR("GetDisabled failed: %08x\n", nsres
);
352 *p
= disabled
? VARIANT_TRUE
: VARIANT_FALSE
;
356 static HRESULT WINAPI
HTMLSelectElement_get_form(IHTMLSelectElement
*iface
, IHTMLFormElement
**p
)
358 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
359 FIXME("(%p)->(%p)\n", This
, p
);
363 static HRESULT WINAPI
HTMLSelectElement_add(IHTMLSelectElement
*iface
, IHTMLElement
*element
,
366 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
367 nsIWritableVariant
*nsvariant
;
368 HTMLElement
*element_obj
;
371 TRACE("(%p)->(%p %s)\n", This
, element
, debugstr_variant(&before
));
373 element_obj
= unsafe_impl_from_IHTMLElement(element
);
375 FIXME("External IHTMLElement implementation?\n");
379 nsvariant
= create_nsvariant();
383 switch(V_VT(&before
)) {
385 nsres
= nsIWritableVariant_SetAsEmpty(nsvariant
);
388 nsres
= nsIWritableVariant_SetAsInt16(nsvariant
, V_I2(&before
));
391 FIXME("unhandled before %s\n", debugstr_variant(&before
));
392 nsIWritableVariant_Release(nsvariant
);
396 if(NS_SUCCEEDED(nsres
))
397 nsres
= nsIDOMHTMLSelectElement_Add(This
->nsselect
, element_obj
->nselem
, (nsIVariant
*)nsvariant
);
398 nsIWritableVariant_Release(nsvariant
);
399 if(NS_FAILED(nsres
)) {
400 ERR("Add failed: %08x\n", nsres
);
407 static HRESULT WINAPI
HTMLSelectElement_remove(IHTMLSelectElement
*iface
, LONG index
)
409 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
411 TRACE("(%p)->(%d)\n", This
, index
);
412 nsres
= nsIDOMHTMLSelectElement_select_Remove(This
->nsselect
, index
);
413 if(NS_FAILED(nsres
)) {
414 ERR("Remove failed: %08x\n", nsres
);
420 static HRESULT WINAPI
HTMLSelectElement_put_length(IHTMLSelectElement
*iface
, LONG v
)
422 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
425 TRACE("(%p)->(%d)\n", This
, v
);
427 nsres
= nsIDOMHTMLSelectElement_SetLength(This
->nsselect
, v
);
429 ERR("SetLength failed: %08x\n", nsres
);
434 static HRESULT WINAPI
HTMLSelectElement_get_length(IHTMLSelectElement
*iface
, LONG
*p
)
436 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
440 TRACE("(%p)->(%p)\n", This
, p
);
442 nsres
= nsIDOMHTMLSelectElement_GetLength(This
->nsselect
, &length
);
444 ERR("GetLength failed: %08x\n", nsres
);
448 TRACE("ret %d\n", *p
);
452 static HRESULT WINAPI
HTMLSelectElement_get__newEnum(IHTMLSelectElement
*iface
, IUnknown
**p
)
454 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
455 FIXME("(%p)->(%p)\n", This
, p
);
459 static HRESULT WINAPI
HTMLSelectElement_item(IHTMLSelectElement
*iface
, VARIANT name
,
460 VARIANT index
, IDispatch
**pdisp
)
462 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
464 TRACE("(%p)->(%s %s %p)\n", This
, debugstr_variant(&name
), debugstr_variant(&index
), pdisp
);
470 if(V_VT(&name
) == VT_I4
) {
473 return htmlselect_item(This
, V_I4(&name
), pdisp
);
476 FIXME("Unsupported args\n");
480 static HRESULT WINAPI
HTMLSelectElement_tags(IHTMLSelectElement
*iface
, VARIANT tagName
,
483 HTMLSelectElement
*This
= impl_from_IHTMLSelectElement(iface
);
484 FIXME("(%p)->(v %p)\n", This
, pdisp
);
488 static const IHTMLSelectElementVtbl HTMLSelectElementVtbl
= {
489 HTMLSelectElement_QueryInterface
,
490 HTMLSelectElement_AddRef
,
491 HTMLSelectElement_Release
,
492 HTMLSelectElement_GetTypeInfoCount
,
493 HTMLSelectElement_GetTypeInfo
,
494 HTMLSelectElement_GetIDsOfNames
,
495 HTMLSelectElement_Invoke
,
496 HTMLSelectElement_put_size
,
497 HTMLSelectElement_get_size
,
498 HTMLSelectElement_put_multiple
,
499 HTMLSelectElement_get_multiple
,
500 HTMLSelectElement_put_name
,
501 HTMLSelectElement_get_name
,
502 HTMLSelectElement_get_options
,
503 HTMLSelectElement_put_onchange
,
504 HTMLSelectElement_get_onchange
,
505 HTMLSelectElement_put_selectedIndex
,
506 HTMLSelectElement_get_selectedIndex
,
507 HTMLSelectElement_get_type
,
508 HTMLSelectElement_put_value
,
509 HTMLSelectElement_get_value
,
510 HTMLSelectElement_put_disabled
,
511 HTMLSelectElement_get_disabled
,
512 HTMLSelectElement_get_form
,
513 HTMLSelectElement_add
,
514 HTMLSelectElement_remove
,
515 HTMLSelectElement_put_length
,
516 HTMLSelectElement_get_length
,
517 HTMLSelectElement_get__newEnum
,
518 HTMLSelectElement_item
,
519 HTMLSelectElement_tags
522 static inline HTMLSelectElement
*impl_from_HTMLDOMNode(HTMLDOMNode
*iface
)
524 return CONTAINING_RECORD(iface
, HTMLSelectElement
, element
.node
);
527 static HRESULT
HTMLSelectElement_QI(HTMLDOMNode
*iface
, REFIID riid
, void **ppv
)
529 HTMLSelectElement
*This
= impl_from_HTMLDOMNode(iface
);
533 if(IsEqualGUID(&IID_IUnknown
, riid
)) {
534 TRACE("(%p)->(IID_IUnknown %p)\n", This
, ppv
);
535 *ppv
= &This
->IHTMLSelectElement_iface
;
536 }else if(IsEqualGUID(&IID_IDispatch
, riid
)) {
537 TRACE("(%p)->(IID_IDispatch %p)\n", This
, ppv
);
538 *ppv
= &This
->IHTMLSelectElement_iface
;
539 }else if(IsEqualGUID(&IID_IHTMLSelectElement
, riid
)) {
540 TRACE("(%p)->(IID_IHTMLSelectElement %p)\n", This
, ppv
);
541 *ppv
= &This
->IHTMLSelectElement_iface
;
545 IUnknown_AddRef((IUnknown
*)*ppv
);
549 return HTMLElement_QI(&This
->element
.node
, riid
, ppv
);
552 static HRESULT
HTMLSelectElementImpl_put_disabled(HTMLDOMNode
*iface
, VARIANT_BOOL v
)
554 HTMLSelectElement
*This
= impl_from_HTMLDOMNode(iface
);
555 return IHTMLSelectElement_put_disabled(&This
->IHTMLSelectElement_iface
, v
);
558 static HRESULT
HTMLSelectElementImpl_get_disabled(HTMLDOMNode
*iface
, VARIANT_BOOL
*p
)
560 HTMLSelectElement
*This
= impl_from_HTMLDOMNode(iface
);
561 return IHTMLSelectElement_get_disabled(&This
->IHTMLSelectElement_iface
, p
);
564 #define DISPID_OPTIONCOL_0 MSHTML_DISPID_CUSTOM_MIN
566 static HRESULT
HTMLSelectElement_get_dispid(HTMLDOMNode
*iface
, BSTR name
, DWORD flags
, DISPID
*dispid
)
571 for(ptr
= name
; *ptr
&& isdigitW(*ptr
); ptr
++) {
572 idx
= idx
*10 + (*ptr
-'0');
573 if(idx
> MSHTML_CUSTOM_DISPID_CNT
) {
574 WARN("too big idx\n");
575 return DISP_E_UNKNOWNNAME
;
579 return DISP_E_UNKNOWNNAME
;
581 *dispid
= DISPID_OPTIONCOL_0
+ idx
;
585 static HRESULT
HTMLSelectElement_invoke(HTMLDOMNode
*iface
, DISPID id
, LCID lcid
, WORD flags
, DISPPARAMS
*params
,
586 VARIANT
*res
, EXCEPINFO
*ei
, IServiceProvider
*caller
)
588 HTMLSelectElement
*This
= impl_from_HTMLDOMNode(iface
);
590 TRACE("(%p)->(%x %x %x %p %p %p %p)\n", This
, id
, lcid
, flags
, params
, res
, ei
, caller
);
593 case DISPATCH_PROPERTYGET
: {
597 hres
= htmlselect_item(This
, id
-DISPID_OPTIONCOL_0
, &ret
);
602 V_VT(res
) = VT_DISPATCH
;
603 V_DISPATCH(res
) = ret
;
611 FIXME("unimplemented flags %x\n", flags
);
618 static const NodeImplVtbl HTMLSelectElementImplVtbl
= {
619 HTMLSelectElement_QI
,
620 HTMLElement_destructor
,
623 HTMLElement_handle_event
,
624 HTMLElement_get_attr_col
,
627 HTMLSelectElementImpl_put_disabled
,
628 HTMLSelectElementImpl_get_disabled
,
631 HTMLSelectElement_get_dispid
,
632 HTMLSelectElement_invoke
635 static const tid_t HTMLSelectElement_tids
[] = {
637 IHTMLSelectElement_tid
,
641 static dispex_static_data_t HTMLSelectElement_dispex
= {
643 DispHTMLSelectElement_tid
,
645 HTMLSelectElement_tids
648 HRESULT
HTMLSelectElement_Create(HTMLDocumentNode
*doc
, nsIDOMHTMLElement
*nselem
, HTMLElement
**elem
)
650 HTMLSelectElement
*ret
;
653 ret
= heap_alloc_zero(sizeof(HTMLSelectElement
));
655 return E_OUTOFMEMORY
;
657 ret
->IHTMLSelectElement_iface
.lpVtbl
= &HTMLSelectElementVtbl
;
658 ret
->element
.node
.vtbl
= &HTMLSelectElementImplVtbl
;
660 HTMLElement_Init(&ret
->element
, doc
, nselem
, &HTMLSelectElement_dispex
);
662 nsres
= nsIDOMHTMLElement_QueryInterface(nselem
, &IID_nsIDOMHTMLSelectElement
,
663 (void**)&ret
->nsselect
);
665 /* Share nsselect reference with nsnode */
666 assert(nsres
== NS_OK
&& (nsIDOMNode
*)ret
->nsselect
== ret
->element
.node
.nsnode
);
667 nsIDOMNode_Release(ret
->element
.node
.nsnode
);
669 *elem
= &ret
->element
;