4 * Implementation of OLE IPicture and related interfaces
6 * Copyright 2000 Huw D M Davies for CodeWeavers.
7 * Copyright 2001 Marcus Meissner
8 * Copyright 2008 Kirill K. Smirnov
10 * This library is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU Lesser General Public
12 * License as published by the Free Software Foundation; either
13 * version 2.1 of the License, or (at your option) any later version.
15 * This library is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18 * Lesser General Public License for more details.
20 * You should have received a copy of the GNU Lesser General Public
21 * License along with this library; if not, write to the Free Software
22 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
26 * Support PICTYPE_BITMAP and PICTYPE_ICON, although only bitmaps very well..
27 * Lots of methods are just stubs.
30 * NOTES (or things that msdn doesn't tell you)
32 * The width and height properties are returned in HIMETRIC units (0.01mm)
33 * IPicture::Render also uses these to select a region of the src picture.
34 * A bitmap's size is converted into these units by using the screen resolution
35 * thus an 8x8 bitmap on a 96dpi screen has a size of 212x212 (8/96 * 2540).
40 #include "wine/port.h"
50 #define NONAMELESSUNION
51 #define NONAMELESSSTRUCT
64 #include "wine/debug.h"
65 #include "wine/unicode.h"
66 #include "wine/library.h"
68 WINE_DEFAULT_DEBUG_CHANNEL(ole);
72 /* Header for Aldus Placable Metafiles - a standard metafile follows */
73 typedef struct _APM_HEADER
95 } CURSORICONFILEDIRENTRY;
102 CURSORICONFILEDIRENTRY idEntries[1];
107 /*************************************************************************
108 * Declaration of implementation class
111 typedef struct OLEPictureImpl {
114 * IPicture handles IUnknown
117 const IPictureVtbl *lpVtbl;
118 const IDispatchVtbl *lpvtblIDispatch;
119 const IPersistStreamVtbl *lpvtblIPersistStream;
120 const IConnectionPointContainerVtbl *lpvtblIConnectionPointContainer;
122 /* Object reference count */
125 /* We own the object and must destroy it ourselves */
128 /* Picture description */
131 /* These are the pixel size of a bitmap */
135 /* And these are the size of the picture converted into HIMETRIC units */
136 OLE_XSIZE_HIMETRIC himetricWidth;
137 OLE_YSIZE_HIMETRIC himetricHeight;
139 IConnectionPoint *pCP;
144 /* Bitmap transparency mask */
152 BOOL bIsDirty; /* Set to TRUE if picture has changed */
153 unsigned int loadtime_magic; /* If a length header was found, saves value */
154 unsigned int loadtime_format; /* for PICTYPE_BITMAP only, keeps track of image format (GIF/BMP/JPEG) */
158 * Macros to retrieve pointer to IUnknown (IPicture) from the other VTables.
161 static inline OLEPictureImpl *impl_from_IDispatch( IDispatch *iface )
163 return (OLEPictureImpl *)((char*)iface - FIELD_OFFSET(OLEPictureImpl, lpvtblIDispatch));
166 static inline OLEPictureImpl *impl_from_IPersistStream( IPersistStream *iface )
168 return (OLEPictureImpl *)((char*)iface - FIELD_OFFSET(OLEPictureImpl, lpvtblIPersistStream));
171 static inline OLEPictureImpl *impl_from_IConnectionPointContainer( IConnectionPointContainer *iface )
173 return (OLEPictureImpl *)((char*)iface - FIELD_OFFSET(OLEPictureImpl, lpvtblIConnectionPointContainer));
177 * Predeclare VTables. They get initialized at the end.
179 static const IPictureVtbl OLEPictureImpl_VTable;
180 static const IDispatchVtbl OLEPictureImpl_IDispatch_VTable;
181 static const IPersistStreamVtbl OLEPictureImpl_IPersistStream_VTable;
182 static const IConnectionPointContainerVtbl OLEPictureImpl_IConnectionPointContainer_VTable;
184 /* pixels to HIMETRIC units conversion */
185 static inline OLE_XSIZE_HIMETRIC xpixels_to_himetric(INT pixels, HDC hdc)
187 return MulDiv(pixels, 2540, GetDeviceCaps(hdc, LOGPIXELSX));
190 static inline OLE_YSIZE_HIMETRIC ypixels_to_himetric(INT pixels, HDC hdc)
192 return MulDiv(pixels, 2540, GetDeviceCaps(hdc, LOGPIXELSY));
195 /***********************************************************************
196 * Implementation of the OLEPictureImpl class.
199 static void OLEPictureImpl_SetBitmap(OLEPictureImpl *This)
204 TRACE("bitmap handle %p\n", This->desc.u.bmp.hbitmap);
205 if(GetObjectW(This->desc.u.bmp.hbitmap, sizeof(bm), &bm) != sizeof(bm)) {
206 ERR("GetObject fails\n");
209 This->origWidth = bm.bmWidth;
210 This->origHeight = bm.bmHeight;
212 /* The width and height are stored in HIMETRIC units (0.01 mm),
213 so we take our pixel width divide by pixels per inch and
214 multiply by 25.4 * 100 */
215 /* Should we use GetBitmapDimension if available? */
216 hdcRef = CreateCompatibleDC(0);
218 This->himetricWidth = xpixels_to_himetric(bm.bmWidth, hdcRef);
219 This->himetricHeight = ypixels_to_himetric(bm.bmHeight, hdcRef);
224 static void OLEPictureImpl_SetIcon(OLEPictureImpl * This)
228 TRACE("icon handle %p\n", This->desc.u.icon.hicon);
229 if (GetIconInfo(This->desc.u.icon.hicon, &infoIcon)) {
233 TRACE("bitmap handle for icon is %p\n", infoIcon.hbmColor);
234 if(GetObjectW(infoIcon.hbmColor ? infoIcon.hbmColor : infoIcon.hbmMask, sizeof(bm), &bm) != sizeof(bm)) {
235 ERR("GetObject fails on icon bitmap\n");
239 This->origWidth = bm.bmWidth;
240 This->origHeight = infoIcon.hbmColor ? bm.bmHeight : bm.bmHeight / 2;
241 /* see comment on HIMETRIC on OLEPictureImpl_SetBitmap() */
244 This->himetricWidth = xpixels_to_himetric(This->origWidth, hdcRef);
245 This->himetricHeight = ypixels_to_himetric(This->origHeight, hdcRef);
247 ReleaseDC(0, hdcRef);
249 DeleteObject(infoIcon.hbmMask);
250 if (infoIcon.hbmColor) DeleteObject(infoIcon.hbmColor);
252 ERR("GetIconInfo() fails on icon %p\n", This->desc.u.icon.hicon);
256 /************************************************************************
257 * OLEPictureImpl_Construct
259 * This method will construct a new instance of the OLEPictureImpl
262 * The caller of this method must release the object when it's
265 static OLEPictureImpl* OLEPictureImpl_Construct(LPPICTDESC pictDesc, BOOL fOwn)
267 OLEPictureImpl* newObject = 0;
270 TRACE("(%p) type = %d\n", pictDesc, pictDesc->picType);
273 * Allocate space for the object.
275 newObject = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(OLEPictureImpl));
281 * Initialize the virtual function table.
283 newObject->lpVtbl = &OLEPictureImpl_VTable;
284 newObject->lpvtblIDispatch = &OLEPictureImpl_IDispatch_VTable;
285 newObject->lpvtblIPersistStream = &OLEPictureImpl_IPersistStream_VTable;
286 newObject->lpvtblIConnectionPointContainer = &OLEPictureImpl_IConnectionPointContainer_VTable;
288 newObject->pCP = NULL;
289 CreateConnectionPoint((IUnknown*)newObject,&IID_IPropertyNotifySink,&newObject->pCP);
292 HeapFree(GetProcessHeap(), 0, newObject);
297 * Start with one reference count. The caller of this function
298 * must release the interface pointer when it is done.
301 newObject->hDCCur = 0;
303 newObject->fOwn = fOwn;
305 /* dunno about original value */
306 newObject->keepOrigFormat = TRUE;
308 newObject->hbmMask = NULL;
309 newObject->hbmXor = NULL;
310 newObject->loadtime_magic = 0xdeadbeef;
311 newObject->loadtime_format = 0;
312 newObject->bIsDirty = FALSE;
315 newObject->desc = *pictDesc;
317 switch(pictDesc->picType) {
319 OLEPictureImpl_SetBitmap(newObject);
322 case PICTYPE_METAFILE:
323 TRACE("metafile handle %p\n", pictDesc->u.wmf.hmeta);
324 newObject->himetricWidth = pictDesc->u.wmf.xExt;
325 newObject->himetricHeight = pictDesc->u.wmf.yExt;
329 /* not sure what to do here */
330 newObject->himetricWidth = newObject->himetricHeight = 0;
334 OLEPictureImpl_SetIcon(newObject);
336 case PICTYPE_ENHMETAFILE:
338 FIXME("Unsupported type %d\n", pictDesc->picType);
339 newObject->himetricWidth = newObject->himetricHeight = 0;
343 newObject->desc.picType = PICTYPE_UNINITIALIZED;
346 TRACE("returning %p\n", newObject);
350 /************************************************************************
351 * OLEPictureImpl_Destroy
353 * This method is called by the Release method when the reference
354 * count goes down to 0. It will free all resources used by
356 static void OLEPictureImpl_Destroy(OLEPictureImpl* Obj)
358 TRACE("(%p)\n", Obj);
361 IConnectionPoint_Release(Obj->pCP);
363 if(Obj->fOwn) { /* We need to destroy the picture */
364 switch(Obj->desc.picType) {
366 DeleteObject(Obj->desc.u.bmp.hbitmap);
367 if (Obj->hbmMask != NULL) DeleteObject(Obj->hbmMask);
368 if (Obj->hbmXor != NULL) DeleteObject(Obj->hbmXor);
370 case PICTYPE_METAFILE:
371 DeleteMetaFile(Obj->desc.u.wmf.hmeta);
374 DestroyIcon(Obj->desc.u.icon.hicon);
376 case PICTYPE_ENHMETAFILE:
377 DeleteEnhMetaFile(Obj->desc.u.emf.hemf);
380 case PICTYPE_UNINITIALIZED:
384 FIXME("Unsupported type %d - unable to delete\n", Obj->desc.picType);
388 HeapFree(GetProcessHeap(), 0, Obj->data);
389 HeapFree(GetProcessHeap(), 0, Obj);
393 /************************************************************************
394 * OLEPictureImpl_AddRef (IUnknown)
396 * See Windows documentation for more details on IUnknown methods.
398 static ULONG WINAPI OLEPictureImpl_AddRef(
401 OLEPictureImpl *This = (OLEPictureImpl *)iface;
402 ULONG refCount = InterlockedIncrement(&This->ref);
404 TRACE("(%p)->(ref before=%d)\n", This, refCount - 1);
409 /************************************************************************
410 * OLEPictureImpl_Release (IUnknown)
412 * See Windows documentation for more details on IUnknown methods.
414 static ULONG WINAPI OLEPictureImpl_Release(
417 OLEPictureImpl *This = (OLEPictureImpl *)iface;
418 ULONG refCount = InterlockedDecrement(&This->ref);
420 TRACE("(%p)->(ref before=%d)\n", This, refCount + 1);
423 * If the reference count goes down to 0, perform suicide.
425 if (!refCount) OLEPictureImpl_Destroy(This);
430 /************************************************************************
431 * OLEPictureImpl_QueryInterface (IUnknown)
433 * See Windows documentation for more details on IUnknown methods.
435 static HRESULT WINAPI OLEPictureImpl_QueryInterface(
440 OLEPictureImpl *This = (OLEPictureImpl *)iface;
442 TRACE("(%p)->(%s, %p)\n", This, debugstr_guid(riid), ppvObject);
449 if (IsEqualIID(&IID_IUnknown, riid) || IsEqualIID(&IID_IPicture, riid))
451 else if (IsEqualIID(&IID_IDispatch, riid))
452 *ppvObject = &This->lpvtblIDispatch;
453 else if (IsEqualIID(&IID_IPictureDisp, riid))
454 *ppvObject = &This->lpvtblIDispatch;
455 else if (IsEqualIID(&IID_IPersist, riid) || IsEqualIID(&IID_IPersistStream, riid))
456 *ppvObject = &This->lpvtblIPersistStream;
457 else if (IsEqualIID(&IID_IConnectionPointContainer, riid))
458 *ppvObject = &This->lpvtblIConnectionPointContainer;
462 FIXME("() : asking for un supported interface %s\n",debugstr_guid(riid));
463 return E_NOINTERFACE;
466 IPicture_AddRef(iface);
471 /***********************************************************************
472 * OLEPicture_SendNotify (internal)
474 * Sends notification messages of changed properties to any interested
477 static void OLEPicture_SendNotify(OLEPictureImpl* this, DISPID dispID)
479 IEnumConnections *pEnum;
482 if (IConnectionPoint_EnumConnections(this->pCP, &pEnum) != S_OK)
484 while(IEnumConnections_Next(pEnum, 1, &CD, NULL) == S_OK) {
485 IPropertyNotifySink *sink;
487 IUnknown_QueryInterface(CD.pUnk, &IID_IPropertyNotifySink, (LPVOID)&sink);
488 IPropertyNotifySink_OnChanged(sink, dispID);
489 IPropertyNotifySink_Release(sink);
490 IUnknown_Release(CD.pUnk);
492 IEnumConnections_Release(pEnum);
495 /************************************************************************
496 * OLEPictureImpl_get_Handle
498 static HRESULT WINAPI OLEPictureImpl_get_Handle(IPicture *iface,
501 OLEPictureImpl *This = (OLEPictureImpl *)iface;
502 TRACE("(%p)->(%p)\n", This, phandle);
507 switch(This->desc.picType) {
509 case PICTYPE_UNINITIALIZED:
513 *phandle = HandleToUlong(This->desc.u.bmp.hbitmap);
515 case PICTYPE_METAFILE:
516 *phandle = HandleToUlong(This->desc.u.wmf.hmeta);
519 *phandle = HandleToUlong(This->desc.u.icon.hicon);
521 case PICTYPE_ENHMETAFILE:
522 *phandle = HandleToUlong(This->desc.u.emf.hemf);
525 FIXME("Unimplemented type %d\n", This->desc.picType);
528 TRACE("returning handle %08x\n", *phandle);
532 /************************************************************************
533 * OLEPictureImpl_get_hPal
535 static HRESULT WINAPI OLEPictureImpl_get_hPal(IPicture *iface,
538 OLEPictureImpl *This = (OLEPictureImpl *)iface;
540 TRACE("(%p)->(%p)\n", This, phandle);
545 switch (This->desc.picType) {
546 case (UINT)PICTYPE_UNINITIALIZED:
552 *phandle = HandleToUlong(This->desc.u.bmp.hpal);
555 case PICTYPE_METAFILE:
559 case PICTYPE_ENHMETAFILE:
561 FIXME("unimplemented for type %d. Returning 0 palette.\n",
567 TRACE("returning 0x%08x, palette handle %08x\n", hres, *phandle);
571 /************************************************************************
572 * OLEPictureImpl_get_Type
574 static HRESULT WINAPI OLEPictureImpl_get_Type(IPicture *iface,
577 OLEPictureImpl *This = (OLEPictureImpl *)iface;
578 TRACE("(%p)->(%p): type is %d\n", This, ptype, This->desc.picType);
583 *ptype = This->desc.picType;
587 /************************************************************************
588 * OLEPictureImpl_get_Width
590 static HRESULT WINAPI OLEPictureImpl_get_Width(IPicture *iface,
591 OLE_XSIZE_HIMETRIC *pwidth)
593 OLEPictureImpl *This = (OLEPictureImpl *)iface;
594 TRACE("(%p)->(%p): width is %d\n", This, pwidth, This->himetricWidth);
595 *pwidth = This->himetricWidth;
599 /************************************************************************
600 * OLEPictureImpl_get_Height
602 static HRESULT WINAPI OLEPictureImpl_get_Height(IPicture *iface,
603 OLE_YSIZE_HIMETRIC *pheight)
605 OLEPictureImpl *This = (OLEPictureImpl *)iface;
606 TRACE("(%p)->(%p): height is %d\n", This, pheight, This->himetricHeight);
607 *pheight = This->himetricHeight;
611 /************************************************************************
612 * OLEPictureImpl_Render
614 static HRESULT WINAPI OLEPictureImpl_Render(IPicture *iface, HDC hdc,
615 LONG x, LONG y, LONG cx, LONG cy,
616 OLE_XPOS_HIMETRIC xSrc,
617 OLE_YPOS_HIMETRIC ySrc,
618 OLE_XSIZE_HIMETRIC cxSrc,
619 OLE_YSIZE_HIMETRIC cySrc,
622 OLEPictureImpl *This = (OLEPictureImpl *)iface;
623 TRACE("(%p)->(%p, (%d,%d), (%d,%d) <- (%d,%d), (%d,%d), %p)\n",
624 This, hdc, x, y, cx, cy, xSrc, ySrc, cxSrc, cySrc, prcWBounds);
626 TRACE("prcWBounds (%d,%d) - (%d,%d)\n", prcWBounds->left, prcWBounds->top,
627 prcWBounds->right, prcWBounds->bottom);
629 if(cx == 0 || cy == 0 || cxSrc == 0 || cySrc == 0){
630 return CTL_E_INVALIDPROPERTYVALUE;
634 * While the documentation suggests this to be here (or after rendering?)
635 * it does cause an endless recursion in my sample app. -MM 20010804
636 OLEPicture_SendNotify(This,DISPID_PICT_RENDER);
639 switch(This->desc.picType) {
640 case PICTYPE_UNINITIALIZED:
649 /* Set a mapping mode that maps bitmap pixels into HIMETRIC units.
650 NB y-axis gets flipped */
652 hdcBmp = CreateCompatibleDC(0);
653 SetMapMode(hdcBmp, MM_ANISOTROPIC);
654 SetWindowOrgEx(hdcBmp, 0, 0, NULL);
655 SetWindowExtEx(hdcBmp, This->himetricWidth, This->himetricHeight, NULL);
656 SetViewportOrgEx(hdcBmp, 0, This->origHeight, NULL);
657 SetViewportExtEx(hdcBmp, This->origWidth, -This->origHeight, NULL);
660 HDC hdcMask = CreateCompatibleDC(0);
661 HBITMAP hOldbm = SelectObject(hdcMask, This->hbmMask);
663 hbmpOld = SelectObject(hdcBmp, This->hbmXor);
665 SetMapMode(hdcMask, MM_ANISOTROPIC);
666 SetWindowOrgEx(hdcMask, 0, 0, NULL);
667 SetWindowExtEx(hdcMask, This->himetricWidth, This->himetricHeight, NULL);
668 SetViewportOrgEx(hdcMask, 0, This->origHeight, NULL);
669 SetViewportExtEx(hdcMask, This->origWidth, -This->origHeight, NULL);
671 SetBkColor(hdc, RGB(255, 255, 255));
672 SetTextColor(hdc, RGB(0, 0, 0));
673 StretchBlt(hdc, x, y, cx, cy, hdcMask, xSrc, ySrc, cxSrc, cySrc, SRCAND);
674 StretchBlt(hdc, x, y, cx, cy, hdcBmp, xSrc, ySrc, cxSrc, cySrc, SRCPAINT);
676 SelectObject(hdcMask, hOldbm);
679 hbmpOld = SelectObject(hdcBmp, This->desc.u.bmp.hbitmap);
680 StretchBlt(hdc, x, y, cx, cy, hdcBmp, xSrc, ySrc, cxSrc, cySrc, SRCCOPY);
683 SelectObject(hdcBmp, hbmpOld);
688 FIXME("Not quite correct implementation of rendering icons...\n");
689 DrawIconEx(hdc, x, y, This->desc.u.icon.hicon, cx, cy, 0, NULL, DI_NORMAL);
692 case PICTYPE_METAFILE:
694 POINT prevOrg, prevWndOrg;
695 SIZE prevExt, prevWndExt;
698 /* Render the WMF to the appropriate location by setting the
699 appropriate ratio between "device units" and "logical units" */
700 oldmode = SetMapMode(hdc, MM_ANISOTROPIC);
701 /* For the "source rectangle" the y-axis must be inverted */
702 SetWindowOrgEx(hdc, xSrc, This->himetricHeight-ySrc, &prevWndOrg);
703 SetWindowExtEx(hdc, cxSrc, -cySrc, &prevWndExt);
704 /* For the "destination rectangle" no inversion is necessary */
705 SetViewportOrgEx(hdc, x, y, &prevOrg);
706 SetViewportExtEx(hdc, cx, cy, &prevExt);
708 if (!PlayMetaFile(hdc, This->desc.u.wmf.hmeta))
709 ERR("PlayMetaFile failed!\n");
711 /* We're done, restore the DC to the previous settings for converting
712 logical units to device units */
713 SetWindowExtEx(hdc, prevWndExt.cx, prevWndExt.cy, NULL);
714 SetWindowOrgEx(hdc, prevWndOrg.x, prevWndOrg.y, NULL);
715 SetViewportExtEx(hdc, prevExt.cx, prevExt.cy, NULL);
716 SetViewportOrgEx(hdc, prevOrg.x, prevOrg.y, NULL);
717 SetMapMode(hdc, oldmode);
721 case PICTYPE_ENHMETAFILE:
723 RECT rc = { x, y, x + cx, y + cy };
724 PlayEnhMetaFile(hdc, This->desc.u.emf.hemf, &rc);
729 FIXME("type %d not implemented\n", This->desc.picType);
735 /************************************************************************
736 * OLEPictureImpl_set_hPal
738 static HRESULT WINAPI OLEPictureImpl_set_hPal(IPicture *iface,
741 OLEPictureImpl *This = (OLEPictureImpl *)iface;
742 FIXME("(%p)->(%08x): stub\n", This, hpal);
743 OLEPicture_SendNotify(This,DISPID_PICT_HPAL);
747 /************************************************************************
748 * OLEPictureImpl_get_CurDC
750 static HRESULT WINAPI OLEPictureImpl_get_CurDC(IPicture *iface,
753 OLEPictureImpl *This = (OLEPictureImpl *)iface;
754 TRACE("(%p), returning %p\n", This, This->hDCCur);
755 if (phdc) *phdc = This->hDCCur;
759 /************************************************************************
760 * OLEPictureImpl_SelectPicture
762 static HRESULT WINAPI OLEPictureImpl_SelectPicture(IPicture *iface,
765 OLE_HANDLE *phbmpOut)
767 OLEPictureImpl *This = (OLEPictureImpl *)iface;
768 TRACE("(%p)->(%p, %p, %p)\n", This, hdcIn, phdcOut, phbmpOut);
769 if (This->desc.picType == PICTYPE_BITMAP) {
770 SelectObject(hdcIn,This->desc.u.bmp.hbitmap);
773 *phdcOut = This->hDCCur;
774 This->hDCCur = hdcIn;
776 *phbmpOut = HandleToUlong(This->desc.u.bmp.hbitmap);
779 FIXME("Don't know how to select picture type %d\n",This->desc.picType);
784 /************************************************************************
785 * OLEPictureImpl_get_KeepOriginalFormat
787 static HRESULT WINAPI OLEPictureImpl_get_KeepOriginalFormat(IPicture *iface,
790 OLEPictureImpl *This = (OLEPictureImpl *)iface;
791 TRACE("(%p)->(%p)\n", This, pfKeep);
794 *pfKeep = This->keepOrigFormat;
798 /************************************************************************
799 * OLEPictureImpl_put_KeepOriginalFormat
801 static HRESULT WINAPI OLEPictureImpl_put_KeepOriginalFormat(IPicture *iface,
804 OLEPictureImpl *This = (OLEPictureImpl *)iface;
805 TRACE("(%p)->(%d)\n", This, keep);
806 This->keepOrigFormat = keep;
807 /* FIXME: what DISPID notification here? */
811 /************************************************************************
812 * OLEPictureImpl_PictureChanged
814 static HRESULT WINAPI OLEPictureImpl_PictureChanged(IPicture *iface)
816 OLEPictureImpl *This = (OLEPictureImpl *)iface;
817 TRACE("(%p)->()\n", This);
818 OLEPicture_SendNotify(This,DISPID_PICT_HANDLE);
819 This->bIsDirty = TRUE;
823 /************************************************************************
824 * OLEPictureImpl_SaveAsFile
826 static HRESULT WINAPI OLEPictureImpl_SaveAsFile(IPicture *iface,
831 OLEPictureImpl *This = (OLEPictureImpl *)iface;
832 FIXME("(%p)->(%p, %d, %p), hacked stub.\n", This, pstream, SaveMemCopy, pcbSize);
833 return IStream_Write(pstream,This->data,This->datalen,(ULONG*)pcbSize);
836 /************************************************************************
837 * OLEPictureImpl_get_Attributes
839 static HRESULT WINAPI OLEPictureImpl_get_Attributes(IPicture *iface,
842 OLEPictureImpl *This = (OLEPictureImpl *)iface;
843 TRACE("(%p)->(%p).\n", This, pdwAttr);
849 switch (This->desc.picType) {
850 case PICTYPE_UNINITIALIZED:
851 case PICTYPE_NONE: break;
852 case PICTYPE_BITMAP: if (This->hbmMask) *pdwAttr = PICTURE_TRANSPARENT; break; /* not 'truly' scalable, see MSDN. */
853 case PICTYPE_ICON: *pdwAttr = PICTURE_TRANSPARENT;break;
854 case PICTYPE_ENHMETAFILE: /* fall through */
855 case PICTYPE_METAFILE: *pdwAttr = PICTURE_TRANSPARENT|PICTURE_SCALABLE;break;
856 default:FIXME("Unknown pictype %d\n",This->desc.picType);break;
862 /************************************************************************
863 * IConnectionPointContainer
865 static HRESULT WINAPI OLEPictureImpl_IConnectionPointContainer_QueryInterface(
866 IConnectionPointContainer* iface,
870 OLEPictureImpl *This = impl_from_IConnectionPointContainer(iface);
872 return IPicture_QueryInterface((IPicture *)This,riid,ppvoid);
875 static ULONG WINAPI OLEPictureImpl_IConnectionPointContainer_AddRef(
876 IConnectionPointContainer* iface)
878 OLEPictureImpl *This = impl_from_IConnectionPointContainer(iface);
880 return IPicture_AddRef((IPicture *)This);
883 static ULONG WINAPI OLEPictureImpl_IConnectionPointContainer_Release(
884 IConnectionPointContainer* iface)
886 OLEPictureImpl *This = impl_from_IConnectionPointContainer(iface);
888 return IPicture_Release((IPicture *)This);
891 static HRESULT WINAPI OLEPictureImpl_EnumConnectionPoints(
892 IConnectionPointContainer* iface,
893 IEnumConnectionPoints** ppEnum)
895 OLEPictureImpl *This = impl_from_IConnectionPointContainer(iface);
897 FIXME("(%p,%p), stub!\n",This,ppEnum);
901 static HRESULT WINAPI OLEPictureImpl_FindConnectionPoint(
902 IConnectionPointContainer* iface,
904 IConnectionPoint **ppCP)
906 OLEPictureImpl *This = impl_from_IConnectionPointContainer(iface);
907 TRACE("(%p,%s,%p)\n",This,debugstr_guid(riid),ppCP);
911 if (IsEqualGUID(riid,&IID_IPropertyNotifySink))
912 return IConnectionPoint_QueryInterface(This->pCP,&IID_IConnectionPoint,(LPVOID)ppCP);
913 FIXME("no connection point for %s\n",debugstr_guid(riid));
914 return CONNECT_E_NOCONNECTION;
918 /************************************************************************
922 /************************************************************************
923 * OLEPictureImpl_IPersistStream_QueryInterface (IUnknown)
925 * See Windows documentation for more details on IUnknown methods.
927 static HRESULT WINAPI OLEPictureImpl_IPersistStream_QueryInterface(
928 IPersistStream* iface,
932 OLEPictureImpl *This = impl_from_IPersistStream(iface);
934 return IPicture_QueryInterface((IPicture *)This, riid, ppvoid);
937 /************************************************************************
938 * OLEPictureImpl_IPersistStream_AddRef (IUnknown)
940 * See Windows documentation for more details on IUnknown methods.
942 static ULONG WINAPI OLEPictureImpl_IPersistStream_AddRef(
943 IPersistStream* iface)
945 OLEPictureImpl *This = impl_from_IPersistStream(iface);
947 return IPicture_AddRef((IPicture *)This);
950 /************************************************************************
951 * OLEPictureImpl_IPersistStream_Release (IUnknown)
953 * See Windows documentation for more details on IUnknown methods.
955 static ULONG WINAPI OLEPictureImpl_IPersistStream_Release(
956 IPersistStream* iface)
958 OLEPictureImpl *This = impl_from_IPersistStream(iface);
960 return IPicture_Release((IPicture *)This);
963 /************************************************************************
964 * OLEPictureImpl_IPersistStream_GetClassID
966 static HRESULT WINAPI OLEPictureImpl_GetClassID(
967 IPersistStream* iface,CLSID* pClassID)
969 TRACE("(%p)\n", pClassID);
970 *pClassID = CLSID_StdPicture;
974 /************************************************************************
975 * OLEPictureImpl_IPersistStream_IsDirty
977 static HRESULT WINAPI OLEPictureImpl_IsDirty(
978 IPersistStream* iface)
980 OLEPictureImpl *This = impl_from_IPersistStream(iface);
981 FIXME("(%p),stub!\n",This);
985 static HRESULT OLEPictureImpl_LoadDIB(OLEPictureImpl *This, BYTE *xbuf, ULONG xread)
987 BITMAPFILEHEADER *bfh = (BITMAPFILEHEADER*)xbuf;
988 BITMAPINFO *bi = (BITMAPINFO*)(bfh+1);
991 /* Does not matter whether this is a coreheader or not, we only use
992 * components which are in both
995 This->desc.u.bmp.hbitmap = CreateDIBitmap(
1003 ReleaseDC(0, hdcref);
1004 if (This->desc.u.bmp.hbitmap == 0)
1006 This->desc.picType = PICTYPE_BITMAP;
1007 OLEPictureImpl_SetBitmap(This);
1011 static HRESULT OLEPictureImpl_LoadWICSource(OLEPictureImpl *This, IWICBitmapSource *src)
1014 BITMAPINFOHEADER bih;
1017 UINT stride, buffersize;
1020 IWICBitmapSource *real_source;
1022 COLORREF white = RGB(255, 255, 255), black = RGB(0, 0, 0);
1023 BOOL has_alpha=FALSE;
1025 hr = WICConvertBitmapSource(&GUID_WICPixelFormat32bppBGRA, src, &real_source);
1026 if (FAILED(hr)) return hr;
1028 hr = IWICBitmapSource_GetSize(real_source, &width, &height);
1029 if (FAILED(hr)) goto end;
1031 bih.biSize = sizeof(bih);
1032 bih.biWidth = width;
1033 bih.biHeight = -height;
1035 bih.biBitCount = 32;
1036 bih.biCompression = BI_RGB;
1037 bih.biSizeImage = 0;
1038 bih.biXPelsPerMeter = 4085; /* olepicture ignores the stored resolution */
1039 bih.biYPelsPerMeter = 4085;
1041 bih.biClrImportant = 0;
1044 buffersize = stride * height;
1046 bits = HeapAlloc(GetProcessHeap(), 0, buffersize);
1057 hr = IWICBitmapSource_CopyPixels(real_source, &rc, stride, buffersize, bits);
1062 This->desc.u.bmp.hbitmap = CreateDIBitmap(
1070 if (This->desc.u.bmp.hbitmap == 0)
1073 ReleaseDC(0, hdcref);
1077 This->desc.picType = PICTYPE_BITMAP;
1078 OLEPictureImpl_SetBitmap(This);
1080 /* set transparent pixels to black, all others to white */
1081 for(y = 0; y < height; y++){
1082 for(x = 0; x < width; x++){
1083 DWORD *pixel = (DWORD*)(bits + stride*y + 4*x);
1084 if((*pixel & 0x80000000) == 0)
1096 HDC hdcBmp, hdcXor, hdcMask;
1097 HBITMAP hbmoldBmp, hbmoldXor, hbmoldMask;
1099 This->hbmXor = CreateDIBitmap(
1108 This->hbmMask = CreateBitmap(width,-height,1,1,NULL);
1109 hdcBmp = CreateCompatibleDC(NULL);
1110 hdcXor = CreateCompatibleDC(NULL);
1111 hdcMask = CreateCompatibleDC(NULL);
1113 hbmoldBmp = SelectObject(hdcBmp,This->desc.u.bmp.hbitmap);
1114 hbmoldXor = SelectObject(hdcXor,This->hbmXor);
1115 hbmoldMask = SelectObject(hdcMask,This->hbmMask);
1117 SetBkColor(hdcXor,black);
1118 BitBlt(hdcMask,0,0,width,height,hdcXor,0,0,SRCCOPY);
1119 BitBlt(hdcXor,0,0,width,height,hdcBmp,0,0,SRCAND);
1121 SelectObject(hdcBmp,hbmoldBmp);
1122 SelectObject(hdcXor,hbmoldXor);
1123 SelectObject(hdcMask,hbmoldMask);
1130 ReleaseDC(0, hdcref);
1133 HeapFree(GetProcessHeap(), 0, bits);
1134 IWICBitmapSource_Release(real_source);
1138 static HRESULT OLEPictureImpl_LoadWICDecoder(OLEPictureImpl *This, REFCLSID decoder_clsid, BYTE *xbuf, ULONG xread)
1141 IWICImagingFactory *factory;
1142 IWICBitmapDecoder *decoder;
1143 IWICBitmapFrameDecode *framedecode;
1147 initresult = CoInitialize(NULL);
1149 hr = CoCreateInstance(&CLSID_WICImagingFactory, NULL, CLSCTX_INPROC_SERVER,
1150 &IID_IWICImagingFactory, (void**)&factory);
1151 if (SUCCEEDED(hr)) /* created factory */
1153 hr = IWICImagingFactory_CreateStream(factory, &stream);
1154 IWICImagingFactory_Release(factory);
1157 if (SUCCEEDED(hr)) /* created stream */
1159 hr = IWICStream_InitializeFromMemory(stream, xbuf, xread);
1161 if (SUCCEEDED(hr)) /* initialized stream */
1163 hr = CoCreateInstance(decoder_clsid, NULL, CLSCTX_INPROC_SERVER,
1164 &IID_IWICBitmapDecoder, (void**)&decoder);
1165 if (SUCCEEDED(hr)) /* created decoder */
1167 hr = IWICBitmapDecoder_Initialize(decoder, (IStream*)stream, WICDecodeMetadataCacheOnLoad);
1169 if (SUCCEEDED(hr)) /* initialized decoder */
1170 hr = IWICBitmapDecoder_GetFrame(decoder, 0, &framedecode);
1172 IWICBitmapDecoder_Release(decoder);
1176 IWICStream_Release(stream);
1179 if (SUCCEEDED(hr)) /* got framedecode */
1181 hr = OLEPictureImpl_LoadWICSource(This, (IWICBitmapSource*)framedecode);
1182 IWICBitmapFrameDecode_Release(framedecode);
1185 if (SUCCEEDED(initresult)) CoUninitialize();
1189 /*****************************************************
1190 * start of Icon-specific code
1193 static HRESULT OLEPictureImpl_LoadIcon(OLEPictureImpl *This, BYTE *xbuf, ULONG xread)
1196 CURSORICONFILEDIR *cifd = (CURSORICONFILEDIR*)xbuf;
1201 FIXME("icon.idReserved=%d\n",cifd->idReserved);
1202 FIXME("icon.idType=%d\n",cifd->idType);
1203 FIXME("icon.idCount=%d\n",cifd->idCount);
1205 for (i=0;i<cifd->idCount;i++) {
1206 FIXME("[%d] width %d\n",i,cifd->idEntries[i].bWidth);
1207 FIXME("[%d] height %d\n",i,cifd->idEntries[i].bHeight);
1208 FIXME("[%d] bColorCount %d\n",i,cifd->idEntries[i].bColorCount);
1209 FIXME("[%d] bReserved %d\n",i,cifd->idEntries[i].bReserved);
1210 FIXME("[%d] xHotspot %d\n",i,cifd->idEntries[i].xHotspot);
1211 FIXME("[%d] yHotspot %d\n",i,cifd->idEntries[i].yHotspot);
1212 FIXME("[%d] dwDIBSize %d\n",i,cifd->idEntries[i].dwDIBSize);
1213 FIXME("[%d] dwDIBOffset %d\n",i,cifd->idEntries[i].dwDIBOffset);
1217 /* If we have more than one icon, try to find the best.
1218 * this currently means '32 pixel wide'.
1220 if (cifd->idCount!=1) {
1221 for (i=0;i<cifd->idCount;i++) {
1222 if (cifd->idEntries[i].bWidth == 32)
1225 if (i==cifd->idCount) i=0;
1228 hicon = CreateIconFromResourceEx(
1229 xbuf+cifd->idEntries[i].dwDIBOffset,
1230 cifd->idEntries[i].dwDIBSize,
1233 cifd->idEntries[i].bWidth,
1234 cifd->idEntries[i].bHeight,
1238 ERR("CreateIcon failed.\n");
1241 This->desc.picType = PICTYPE_ICON;
1242 This->desc.u.icon.hicon = hicon;
1243 This->origWidth = cifd->idEntries[i].bWidth;
1244 This->origHeight = cifd->idEntries[i].bHeight;
1245 hdcRef = CreateCompatibleDC(0);
1246 This->himetricWidth = xpixels_to_himetric(cifd->idEntries[i].bWidth, hdcRef);
1247 This->himetricHeight= ypixels_to_himetric(cifd->idEntries[i].bHeight, hdcRef);
1253 static HRESULT OLEPictureImpl_LoadEnhMetafile(OLEPictureImpl *This,
1254 const BYTE *data, ULONG size)
1259 hemf = SetEnhMetaFileBits(size, data);
1260 if (!hemf) return E_FAIL;
1262 GetEnhMetaFileHeader(hemf, sizeof(hdr), &hdr);
1264 This->desc.picType = PICTYPE_ENHMETAFILE;
1265 This->desc.u.emf.hemf = hemf;
1267 This->origWidth = 0;
1268 This->origHeight = 0;
1269 This->himetricWidth = hdr.rclFrame.right - hdr.rclFrame.left;
1270 This->himetricHeight = hdr.rclFrame.bottom - hdr.rclFrame.top;
1275 static HRESULT OLEPictureImpl_LoadAPM(OLEPictureImpl *This,
1276 const BYTE *data, ULONG size)
1278 const APM_HEADER *header = (const APM_HEADER *)data;
1281 if (size < sizeof(APM_HEADER))
1283 if (header->key != 0x9ac6cdd7)
1286 /* SetMetaFileBitsEx performs data check on its own */
1287 hmf = SetMetaFileBitsEx(size - sizeof(*header), data + sizeof(*header));
1288 if (!hmf) return E_FAIL;
1290 This->desc.picType = PICTYPE_METAFILE;
1291 This->desc.u.wmf.hmeta = hmf;
1292 This->desc.u.wmf.xExt = 0;
1293 This->desc.u.wmf.yExt = 0;
1295 This->origWidth = 0;
1296 This->origHeight = 0;
1297 This->himetricWidth = MulDiv((INT)header->right - header->left, 2540, header->inch);
1298 This->himetricHeight = MulDiv((INT)header->bottom - header->top, 2540, header->inch);
1302 /************************************************************************
1303 * BITMAP FORMAT FLAGS -
1304 * Flags that differentiate between different types of bitmaps.
1307 #define BITMAP_FORMAT_BMP 0x4d42 /* "BM" */
1308 #define BITMAP_FORMAT_JPEG 0xd8ff
1309 #define BITMAP_FORMAT_GIF 0x4947
1310 #define BITMAP_FORMAT_PNG 0x5089
1311 #define BITMAP_FORMAT_APM 0xcdd7
1313 /************************************************************************
1314 * OLEPictureImpl_IPersistStream_Load (IUnknown)
1316 * Loads the binary data from the IStream. Starts at current position.
1317 * There appears to be an 2 DWORD header:
1321 * Currently implemented: BITMAP, ICON, JPEG, GIF, WMF, EMF
1323 static HRESULT WINAPI OLEPictureImpl_Load(IPersistStream* iface, IStream *pStm) {
1326 BOOL statfailed = FALSE;
1327 ULONG xread, toread;
1333 OLEPictureImpl *This = impl_from_IPersistStream(iface);
1335 TRACE("(%p,%p)\n",This,pStm);
1337 /****************************************************************************************
1338 * Part 1: Load the data
1340 /* Sometimes we have a header, sometimes we don't. Apply some guesses to find
1341 * out whether we do.
1343 * UPDATE: the IStream can be mapped to a plain file instead of a stream in a
1344 * compound file. This may explain most, if not all, of the cases of "no
1345 * header", and the header validation should take this into account.
1346 * At least in Visual Basic 6, resource streams, valid headers are
1347 * header[0] == "lt\0\0",
1348 * header[1] == length_of_stream.
1350 * Also handle streams where we do not have a working "Stat" method by
1351 * reading all data until the end of the stream.
1353 hr = IStream_Stat(pStm,&statstg,STATFLAG_NONAME);
1355 TRACE("stat failed with hres %x, proceeding to read all data.\n",hr);
1357 /* we will read at least 8 byte ... just right below */
1358 statstg.cbSize.QuadPart = 8;
1363 headerisdata = FALSE;
1365 hr = IStream_Read(pStm, header, 8, &xread);
1366 if (hr != S_OK || xread!=8) {
1367 ERR("Failure while reading picture header (hr is %x, nread is %d).\n",hr,xread);
1368 return (hr?hr:E_FAIL);
1370 headerread += xread;
1373 if (!memcmp(&(header[0]),"lt\0\0", 4) && (statfailed || (header[1] + headerread <= statstg.cbSize.QuadPart))) {
1374 if (toread != 0 && toread != header[1])
1375 FIXME("varying lengths of image data (prev=%u curr=%u), only last one will be used\n",
1378 if (toread == 0) break;
1380 if (!memcmp(&(header[0]), "GIF8", 4) || /* GIF header */
1381 !memcmp(&(header[0]), "BM", 2) || /* BMP header */
1382 !memcmp(&(header[0]), "\xff\xd8", 2) || /* JPEG header */
1383 (header[0] == EMR_HEADER) || /* EMF header */
1384 (header[1] > statstg.cbSize.QuadPart)|| /* invalid size */
1386 ) {/* Found start of bitmap data */
1387 headerisdata = TRUE;
1389 toread = statstg.cbSize.QuadPart-8;
1393 FIXME("Unknown stream header magic: %08x\n", header[0]);
1397 } while (!headerisdata);
1399 if (statfailed) { /* we don't know the size ... read all we get */
1401 int origsize = sizeinc;
1404 TRACE("Reading all data from stream.\n");
1405 xbuf = HeapAlloc (GetProcessHeap(), HEAP_ZERO_MEMORY, origsize);
1407 memcpy (xbuf, header, 8);
1409 while (xread < origsize) {
1410 hr = IStream_Read(pStm,xbuf+xread,origsize-xread,&nread);
1412 if (hr != S_OK || !nread)
1415 if (!nread || hr != S_OK) /* done, or error */
1417 if (xread == origsize) {
1418 origsize += sizeinc;
1419 sizeinc = 2*sizeinc; /* exponential increase */
1420 xbuf = HeapReAlloc (GetProcessHeap(), HEAP_ZERO_MEMORY, xbuf, origsize);
1424 TRACE("hr in no-stat loader case is %08x\n", hr);
1425 TRACE("loaded %d bytes.\n", xread);
1426 This->datalen = xread;
1429 This->datalen = toread+(headerisdata?8:0);
1430 xbuf = This->data = HeapAlloc (GetProcessHeap(), HEAP_ZERO_MEMORY, This->datalen);
1432 return E_OUTOFMEMORY;
1435 memcpy (xbuf, header, 8);
1437 while (xread < This->datalen) {
1439 hr = IStream_Read(pStm,xbuf+xread,This->datalen-xread,&nread);
1441 if (hr != S_OK || !nread)
1444 if (xread != This->datalen)
1445 ERR("Could only read %d of %d bytes out of stream?\n",xread,This->datalen);
1447 if (This->datalen == 0) { /* Marks the "NONE" picture */
1448 This->desc.picType = PICTYPE_NONE;
1453 /****************************************************************************************
1454 * Part 2: Process the loaded data
1457 magic = xbuf[0] + (xbuf[1]<<8);
1458 This->loadtime_format = magic;
1461 case BITMAP_FORMAT_GIF: /* GIF */
1462 hr = OLEPictureImpl_LoadWICDecoder(This, &CLSID_WICGifDecoder, xbuf, xread);
1464 case BITMAP_FORMAT_JPEG: /* JPEG */
1465 hr = OLEPictureImpl_LoadWICDecoder(This, &CLSID_WICJpegDecoder, xbuf, xread);
1467 case BITMAP_FORMAT_BMP: /* Bitmap */
1468 hr = OLEPictureImpl_LoadDIB(This, xbuf, xread);
1470 case BITMAP_FORMAT_PNG: /* PNG */
1471 hr = OLEPictureImpl_LoadWICDecoder(This, &CLSID_WICPngDecoder, xbuf, xread);
1473 case BITMAP_FORMAT_APM: /* APM */
1474 hr = OLEPictureImpl_LoadAPM(This, xbuf, xread);
1476 case 0x0000: { /* ICON , first word is dwReserved */
1477 hr = OLEPictureImpl_LoadIcon(This, xbuf, xread);
1484 /* let's see if it's a EMF */
1485 hr = OLEPictureImpl_LoadEnhMetafile(This, xbuf, xread);
1486 if (hr == S_OK) break;
1488 FIXME("Unknown magic %04x, %d read bytes:\n",magic,xread);
1490 for (i=0;i<xread+8;i++) {
1491 if (i<8) MESSAGE("%02x ",((unsigned char*)header)[i]);
1492 else MESSAGE("%02x ",xbuf[i-8]);
1493 if (i % 10 == 9) MESSAGE("\n");
1499 This->bIsDirty = FALSE;
1501 /* FIXME: this notify is not really documented */
1503 OLEPicture_SendNotify(This,DISPID_PICT_TYPE);
1507 static int serializeBMP(HBITMAP hBitmap, void ** ppBuffer, unsigned int * pLength)
1511 BITMAPINFO * pInfoBitmap;
1512 int iNumPaletteEntries;
1513 unsigned char * pPixelData;
1514 BITMAPFILEHEADER * pFileHeader;
1515 BITMAPINFO * pInfoHeader;
1517 pInfoBitmap = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY,
1518 sizeof(BITMAPINFOHEADER) + 256 * sizeof(RGBQUAD));
1520 /* Find out bitmap size and padded length */
1522 pInfoBitmap->bmiHeader.biSize = sizeof(pInfoBitmap->bmiHeader);
1523 GetDIBits(hDC, hBitmap, 0, 0, NULL, pInfoBitmap, DIB_RGB_COLORS);
1525 /* Fetch bitmap palette & pixel data */
1527 pPixelData = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, pInfoBitmap->bmiHeader.biSizeImage);
1528 GetDIBits(hDC, hBitmap, 0, pInfoBitmap->bmiHeader.biHeight, pPixelData, pInfoBitmap, DIB_RGB_COLORS);
1530 /* Calculate the total length required for the BMP data */
1531 if (pInfoBitmap->bmiHeader.biClrUsed != 0) {
1532 iNumPaletteEntries = pInfoBitmap->bmiHeader.biClrUsed;
1533 if (iNumPaletteEntries > 256) iNumPaletteEntries = 256;
1535 if (pInfoBitmap->bmiHeader.biBitCount <= 8)
1536 iNumPaletteEntries = 1 << pInfoBitmap->bmiHeader.biBitCount;
1538 iNumPaletteEntries = 0;
1541 sizeof(BITMAPFILEHEADER) +
1542 sizeof(BITMAPINFOHEADER) +
1543 iNumPaletteEntries * sizeof(RGBQUAD) +
1544 pInfoBitmap->bmiHeader.biSizeImage;
1545 *ppBuffer = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, *pLength);
1547 /* Fill the BITMAPFILEHEADER */
1548 pFileHeader = *ppBuffer;
1549 pFileHeader->bfType = BITMAP_FORMAT_BMP;
1550 pFileHeader->bfSize = *pLength;
1551 pFileHeader->bfOffBits =
1552 sizeof(BITMAPFILEHEADER) +
1553 sizeof(BITMAPINFOHEADER) +
1554 iNumPaletteEntries * sizeof(RGBQUAD);
1556 /* Fill the BITMAPINFOHEADER and the palette data */
1557 pInfoHeader = (BITMAPINFO *)((unsigned char *)(*ppBuffer) + sizeof(BITMAPFILEHEADER));
1558 memcpy(pInfoHeader, pInfoBitmap, sizeof(BITMAPINFOHEADER) + iNumPaletteEntries * sizeof(RGBQUAD));
1560 (unsigned char *)(*ppBuffer) +
1561 sizeof(BITMAPFILEHEADER) +
1562 sizeof(BITMAPINFOHEADER) +
1563 iNumPaletteEntries * sizeof(RGBQUAD),
1564 pPixelData, pInfoBitmap->bmiHeader.biSizeImage);
1567 HeapFree(GetProcessHeap(), 0, pPixelData);
1568 HeapFree(GetProcessHeap(), 0, pInfoBitmap);
1572 static int serializeIcon(HICON hIcon, void ** ppBuffer, unsigned int * pLength)
1577 *ppBuffer = NULL; *pLength = 0;
1578 if (GetIconInfo(hIcon, &infoIcon)) {
1580 BITMAPINFO * pInfoBitmap;
1581 unsigned char * pIconData = NULL;
1582 unsigned int iDataSize = 0;
1584 pInfoBitmap = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(BITMAPINFOHEADER) + 256 * sizeof(RGBQUAD));
1586 /* Find out icon size */
1588 pInfoBitmap->bmiHeader.biSize = sizeof(pInfoBitmap->bmiHeader);
1589 GetDIBits(hDC, infoIcon.hbmColor, 0, 0, NULL, pInfoBitmap, DIB_RGB_COLORS);
1591 /* Auxiliary pointers */
1592 CURSORICONFILEDIR * pIconDir;
1593 CURSORICONFILEDIRENTRY * pIconEntry;
1594 BITMAPINFOHEADER * pIconBitmapHeader;
1595 unsigned int iOffsetPalette;
1596 unsigned int iOffsetColorData;
1597 unsigned int iOffsetMaskData;
1599 unsigned int iLengthScanLineMask;
1600 unsigned int iNumEntriesPalette;
1602 iLengthScanLineMask = ((pInfoBitmap->bmiHeader.biWidth + 31) >> 5) << 2;
1604 FIXME("DEBUG: bitmap size is %d x %d\n",
1605 pInfoBitmap->bmiHeader.biWidth,
1606 pInfoBitmap->bmiHeader.biHeight);
1607 FIXME("DEBUG: bitmap bpp is %d\n",
1608 pInfoBitmap->bmiHeader.biBitCount);
1609 FIXME("DEBUG: bitmap nplanes is %d\n",
1610 pInfoBitmap->bmiHeader.biPlanes);
1611 FIXME("DEBUG: bitmap biSizeImage is %u\n",
1612 pInfoBitmap->bmiHeader.biSizeImage);
1614 /* Let's start with one CURSORICONFILEDIR and one CURSORICONFILEDIRENTRY */
1615 iDataSize += 3 * sizeof(WORD) + sizeof(CURSORICONFILEDIRENTRY) + sizeof(BITMAPINFOHEADER);
1616 pIconData = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, iDataSize);
1618 /* Fill out the CURSORICONFILEDIR */
1619 pIconDir = (CURSORICONFILEDIR *)pIconData;
1620 pIconDir->idType = 1;
1621 pIconDir->idCount = 1;
1623 /* Fill out the CURSORICONFILEDIRENTRY */
1624 pIconEntry = (CURSORICONFILEDIRENTRY *)(pIconData + 3 * sizeof(WORD));
1625 pIconEntry->bWidth = (unsigned char)pInfoBitmap->bmiHeader.biWidth;
1626 pIconEntry->bHeight = (unsigned char)pInfoBitmap->bmiHeader.biHeight;
1627 pIconEntry->bColorCount =
1628 (pInfoBitmap->bmiHeader.biBitCount < 8)
1629 ? 1 << pInfoBitmap->bmiHeader.biBitCount
1631 pIconEntry->xHotspot = pInfoBitmap->bmiHeader.biPlanes;
1632 pIconEntry->yHotspot = pInfoBitmap->bmiHeader.biBitCount;
1633 pIconEntry->dwDIBSize = 0;
1634 pIconEntry->dwDIBOffset = 3 * sizeof(WORD) + sizeof(CURSORICONFILEDIRENTRY);
1636 /* Fill out the BITMAPINFOHEADER */
1637 pIconBitmapHeader = (BITMAPINFOHEADER *)(pIconData + 3 * sizeof(WORD) + sizeof(CURSORICONFILEDIRENTRY));
1638 *pIconBitmapHeader = pInfoBitmap->bmiHeader;
1640 /* Find out whether a palette exists for the bitmap */
1641 if ( (pInfoBitmap->bmiHeader.biBitCount == 16 && pInfoBitmap->bmiHeader.biCompression == BI_RGB)
1642 || (pInfoBitmap->bmiHeader.biBitCount == 24)
1643 || (pInfoBitmap->bmiHeader.biBitCount == 32 && pInfoBitmap->bmiHeader.biCompression == BI_RGB)) {
1644 iNumEntriesPalette = pInfoBitmap->bmiHeader.biClrUsed;
1645 if (iNumEntriesPalette > 256) iNumEntriesPalette = 256;
1646 } else if ((pInfoBitmap->bmiHeader.biBitCount == 16 || pInfoBitmap->bmiHeader.biBitCount == 32)
1647 && pInfoBitmap->bmiHeader.biCompression == BI_BITFIELDS) {
1648 iNumEntriesPalette = 3;
1649 } else if (pInfoBitmap->bmiHeader.biBitCount <= 8) {
1650 iNumEntriesPalette = 1 << pInfoBitmap->bmiHeader.biBitCount;
1652 iNumEntriesPalette = 0;
1655 /* Add bitmap size and header size to icon data size. */
1656 iOffsetPalette = iDataSize;
1657 iDataSize += iNumEntriesPalette * sizeof(DWORD);
1658 iOffsetColorData = iDataSize;
1659 iDataSize += pIconBitmapHeader->biSizeImage;
1660 iOffsetMaskData = iDataSize;
1661 iDataSize += pIconBitmapHeader->biHeight * iLengthScanLineMask;
1662 pIconBitmapHeader->biSizeImage += pIconBitmapHeader->biHeight * iLengthScanLineMask;
1663 pIconBitmapHeader->biHeight *= 2;
1664 pIconData = HeapReAlloc(GetProcessHeap(), 0, pIconData, iDataSize);
1665 pIconEntry = (CURSORICONFILEDIRENTRY *)(pIconData + 3 * sizeof(WORD));
1666 pIconBitmapHeader = (BITMAPINFOHEADER *)(pIconData + 3 * sizeof(WORD) + sizeof(CURSORICONFILEDIRENTRY));
1667 pIconEntry->dwDIBSize = iDataSize - (3 * sizeof(WORD) + sizeof(CURSORICONFILEDIRENTRY));
1669 /* Get the actual bitmap data from the icon bitmap */
1670 GetDIBits(hDC, infoIcon.hbmColor, 0, pInfoBitmap->bmiHeader.biHeight,
1671 pIconData + iOffsetColorData, pInfoBitmap, DIB_RGB_COLORS);
1672 if (iNumEntriesPalette > 0) {
1673 memcpy(pIconData + iOffsetPalette, pInfoBitmap->bmiColors,
1674 iNumEntriesPalette * sizeof(RGBQUAD));
1677 /* Reset all values so that GetDIBits call succeeds */
1678 memset(pIconData + iOffsetMaskData, 0, iDataSize - iOffsetMaskData);
1679 memset(pInfoBitmap, 0, sizeof(BITMAPINFOHEADER) + 256 * sizeof(RGBQUAD));
1680 pInfoBitmap->bmiHeader.biSize = sizeof(pInfoBitmap->bmiHeader);
1682 if (!(GetDIBits(hDC, infoIcon.hbmMask, 0, 0, NULL, pInfoBitmap, DIB_RGB_COLORS)
1683 && GetDIBits(hDC, infoIcon.hbmMask, 0, pIconEntry->bHeight,
1684 pIconData + iOffsetMaskData, pInfoBitmap, DIB_RGB_COLORS))) {
1686 printf("ERROR: unable to get bitmap mask (error %u)\n",
1691 GetDIBits(hDC, infoIcon.hbmMask, 0, 0, NULL, pInfoBitmap, DIB_RGB_COLORS);
1692 GetDIBits(hDC, infoIcon.hbmMask, 0, pIconEntry->bHeight, pIconData + iOffsetMaskData, pInfoBitmap, DIB_RGB_COLORS);
1694 /* Write out everything produced so far to the stream */
1695 *ppBuffer = pIconData; *pLength = iDataSize;
1699 printf("ERROR: unable to get bitmap information via GetDIBits() (error %u)\n",
1704 Remarks (from MSDN entry on GetIconInfo):
1706 GetIconInfo creates bitmaps for the hbmMask and hbmColor
1707 members of ICONINFO. The calling application must manage
1708 these bitmaps and delete them when they are no longer
1711 if (hDC) ReleaseDC(0, hDC);
1712 DeleteObject(infoIcon.hbmMask);
1713 if (infoIcon.hbmColor) DeleteObject(infoIcon.hbmColor);
1714 HeapFree(GetProcessHeap(), 0, pInfoBitmap);
1716 printf("ERROR: Unable to get icon information (error %u)\n",
1722 static HRESULT WINAPI OLEPictureImpl_Save(
1723 IPersistStream* iface,IStream*pStm,BOOL fClearDirty)
1725 HRESULT hResult = E_NOTIMPL;
1727 unsigned int iDataSize;
1729 int iSerializeResult = 0;
1730 OLEPictureImpl *This = impl_from_IPersistStream(iface);
1732 TRACE("%p %p %d\n", This, pStm, fClearDirty);
1734 switch (This->desc.picType) {
1736 if (This->bIsDirty || !This->data) {
1737 if (!serializeIcon(This->desc.u.icon.hicon, &pIconData, &iDataSize)) {
1738 ERR("(%p,%p,%d), serializeIcon() failed\n", This, pStm, fClearDirty);
1742 HeapFree(GetProcessHeap(), 0, This->data);
1743 This->data = pIconData;
1744 This->datalen = iDataSize;
1746 if (This->loadtime_magic != 0xdeadbeef) {
1749 header[0] = This->loadtime_magic;
1750 header[1] = This->datalen;
1751 IStream_Write(pStm, header, 2 * sizeof(DWORD), &dummy);
1753 IStream_Write(pStm, This->data, This->datalen, &dummy);
1757 case PICTYPE_BITMAP:
1758 if (This->bIsDirty) {
1759 switch (This->keepOrigFormat ? This->loadtime_format : BITMAP_FORMAT_BMP) {
1760 case BITMAP_FORMAT_BMP:
1761 iSerializeResult = serializeBMP(This->desc.u.bmp.hbitmap, &pIconData, &iDataSize);
1763 case BITMAP_FORMAT_JPEG:
1764 FIXME("(%p,%p,%d), PICTYPE_BITMAP (format JPEG) not implemented!\n",This,pStm,fClearDirty);
1766 case BITMAP_FORMAT_GIF:
1767 FIXME("(%p,%p,%d), PICTYPE_BITMAP (format GIF) not implemented!\n",This,pStm,fClearDirty);
1769 case BITMAP_FORMAT_PNG:
1770 FIXME("(%p,%p,%d), PICTYPE_BITMAP (format PNG) not implemented!\n",This,pStm,fClearDirty);
1773 FIXME("(%p,%p,%d), PICTYPE_BITMAP (format UNKNOWN, using BMP?) not implemented!\n",This,pStm,fClearDirty);
1776 if (iSerializeResult) {
1778 if (This->loadtime_magic != 0xdeadbeef) {
1783 header[0] = (This->loadtime_magic != 0xdeadbeef) ? This->loadtime_magic : 0x0000746c;
1784 header[1] = iDataSize;
1785 IStream_Write(pStm, header, 2 * sizeof(DWORD), &dummy);
1787 IStream_Write(pStm, pIconData, iDataSize, &dummy);
1789 HeapFree(GetProcessHeap(), 0, This->data);
1790 This->data = pIconData;
1791 This->datalen = iDataSize;
1796 if (This->loadtime_magic != 0xdeadbeef) {
1801 header[0] = (This->loadtime_magic != 0xdeadbeef) ? This->loadtime_magic : 0x0000746c;
1802 header[1] = This->datalen;
1803 IStream_Write(pStm, header, 2 * sizeof(DWORD), &dummy);
1805 IStream_Write(pStm, This->data, This->datalen, &dummy);
1809 case PICTYPE_METAFILE:
1810 FIXME("(%p,%p,%d), PICTYPE_METAFILE not implemented!\n",This,pStm,fClearDirty);
1812 case PICTYPE_ENHMETAFILE:
1813 FIXME("(%p,%p,%d),PICTYPE_ENHMETAFILE not implemented!\n",This,pStm,fClearDirty);
1816 FIXME("(%p,%p,%d), [unknown type] not implemented!\n",This,pStm,fClearDirty);
1819 if (hResult == S_OK && fClearDirty) This->bIsDirty = FALSE;
1823 static HRESULT WINAPI OLEPictureImpl_GetSizeMax(
1824 IPersistStream* iface,ULARGE_INTEGER*pcbSize)
1826 OLEPictureImpl *This = impl_from_IPersistStream(iface);
1827 FIXME("(%p,%p),stub!\n",This,pcbSize);
1832 /************************************************************************
1836 /************************************************************************
1837 * OLEPictureImpl_IDispatch_QueryInterface (IUnknown)
1839 * See Windows documentation for more details on IUnknown methods.
1841 static HRESULT WINAPI OLEPictureImpl_IDispatch_QueryInterface(
1846 OLEPictureImpl *This = impl_from_IDispatch(iface);
1848 return IPicture_QueryInterface((IPicture *)This, riid, ppvoid);
1851 /************************************************************************
1852 * OLEPictureImpl_IDispatch_AddRef (IUnknown)
1854 * See Windows documentation for more details on IUnknown methods.
1856 static ULONG WINAPI OLEPictureImpl_IDispatch_AddRef(
1859 OLEPictureImpl *This = impl_from_IDispatch(iface);
1861 return IPicture_AddRef((IPicture *)This);
1864 /************************************************************************
1865 * OLEPictureImpl_IDispatch_Release (IUnknown)
1867 * See Windows documentation for more details on IUnknown methods.
1869 static ULONG WINAPI OLEPictureImpl_IDispatch_Release(
1872 OLEPictureImpl *This = impl_from_IDispatch(iface);
1874 return IPicture_Release((IPicture *)This);
1877 /************************************************************************
1878 * OLEPictureImpl_GetTypeInfoCount (IDispatch)
1880 * See Windows documentation for more details on IDispatch methods.
1882 static HRESULT WINAPI OLEPictureImpl_GetTypeInfoCount(
1884 unsigned int* pctinfo)
1886 TRACE("(%p)\n", pctinfo);
1893 /************************************************************************
1894 * OLEPictureImpl_GetTypeInfo (IDispatch)
1896 * See Windows documentation for more details on IDispatch methods.
1898 static HRESULT WINAPI OLEPictureImpl_GetTypeInfo(
1902 ITypeInfo** ppTInfo)
1904 static const WCHAR stdole2tlb[] = {'s','t','d','o','l','e','2','.','t','l','b',0};
1908 TRACE("(iTInfo=%d, lcid=%04x, %p)\n", iTInfo, (int)lcid, ppTInfo);
1913 hres = LoadTypeLib(stdole2tlb, &tl);
1916 ERR("Could not load stdole2.tlb\n");
1920 hres = ITypeLib_GetTypeInfoOfGuid(tl, &IID_IPictureDisp, ppTInfo);
1922 ERR("Did not get IPictureDisp typeinfo from typelib, hres %x\n", hres);
1927 /************************************************************************
1928 * OLEPictureImpl_GetIDsOfNames (IDispatch)
1930 * See Windows documentation for more details on IDispatch methods.
1932 static HRESULT WINAPI OLEPictureImpl_GetIDsOfNames(
1935 LPOLESTR* rgszNames,
1943 TRACE("(%p,%s,%p,cNames=%d,lcid=%04x,%p)\n", iface, debugstr_guid(riid),
1944 rgszNames, cNames, (int)lcid, rgDispId);
1948 return E_INVALIDARG;
1952 /* retrieve type information */
1953 hres = OLEPictureImpl_GetTypeInfo(iface, 0, lcid, &pTInfo);
1957 ERR("GetTypeInfo failed.\n");
1961 /* convert names to DISPIDs */
1962 hres = DispGetIDsOfNames (pTInfo, rgszNames, cNames, rgDispId);
1963 ITypeInfo_Release(pTInfo);
1969 /************************************************************************
1970 * OLEPictureImpl_Invoke (IDispatch)
1972 * See Windows documentation for more details on IDispatch methods.
1974 static HRESULT WINAPI OLEPictureImpl_Invoke(
1976 DISPID dispIdMember,
1980 DISPPARAMS* pDispParams,
1981 VARIANT* pVarResult,
1982 EXCEPINFO* pExepInfo,
1985 OLEPictureImpl *This = impl_from_IDispatch(iface);
1987 /* validate parameters */
1989 if (!IsEqualIID(riid, &IID_NULL))
1991 ERR("riid was %s instead of IID_NULL\n", debugstr_guid(riid));
1992 return DISP_E_UNKNOWNNAME;
1997 ERR("null pDispParams not allowed\n");
1998 return DISP_E_PARAMNOTOPTIONAL;
2001 if (wFlags & DISPATCH_PROPERTYGET)
2003 if (pDispParams->cArgs != 0)
2005 ERR("param count for DISPATCH_PROPERTYGET was %d instead of 0\n", pDispParams->cArgs);
2006 return DISP_E_BADPARAMCOUNT;
2010 ERR("null pVarResult not allowed when DISPATCH_PROPERTYGET specified\n");
2011 return DISP_E_PARAMNOTOPTIONAL;
2014 else if (wFlags & DISPATCH_PROPERTYPUT)
2016 if (pDispParams->cArgs != 1)
2018 ERR("param count for DISPATCH_PROPERTYPUT was %d instead of 1\n", pDispParams->cArgs);
2019 return DISP_E_BADPARAMCOUNT;
2023 switch (dispIdMember)
2025 case DISPID_PICT_HANDLE:
2026 if (wFlags & DISPATCH_PROPERTYGET)
2028 TRACE("DISPID_PICT_HANDLE\n");
2029 V_VT(pVarResult) = VT_I4;
2030 return IPicture_get_Handle((IPicture *)&This->lpVtbl, &V_UINT(pVarResult));
2033 case DISPID_PICT_HPAL:
2034 if (wFlags & DISPATCH_PROPERTYGET)
2036 TRACE("DISPID_PICT_HPAL\n");
2037 V_VT(pVarResult) = VT_I4;
2038 return IPicture_get_hPal((IPicture *)&This->lpVtbl, &V_UINT(pVarResult));
2040 else if (wFlags & DISPATCH_PROPERTYPUT)
2044 TRACE("DISPID_PICT_HPAL\n");
2046 VariantInit(&vararg);
2047 hr = VariantChangeTypeEx(&vararg, &pDispParams->rgvarg[0], lcid, 0, VT_I4);
2051 hr = IPicture_set_hPal((IPicture *)&This->lpVtbl, V_I4(&vararg));
2053 VariantClear(&vararg);
2057 case DISPID_PICT_TYPE:
2058 if (wFlags & DISPATCH_PROPERTYGET)
2060 TRACE("DISPID_PICT_TYPE\n");
2061 V_VT(pVarResult) = VT_I2;
2062 return OLEPictureImpl_get_Type((IPicture *)&This->lpVtbl, &V_I2(pVarResult));
2065 case DISPID_PICT_WIDTH:
2066 if (wFlags & DISPATCH_PROPERTYGET)
2068 TRACE("DISPID_PICT_WIDTH\n");
2069 V_VT(pVarResult) = VT_I4;
2070 return IPicture_get_Width((IPicture *)&This->lpVtbl, &V_I4(pVarResult));
2073 case DISPID_PICT_HEIGHT:
2074 if (wFlags & DISPATCH_PROPERTYGET)
2076 TRACE("DISPID_PICT_HEIGHT\n");
2077 V_VT(pVarResult) = VT_I4;
2078 return IPicture_get_Height((IPicture *)&This->lpVtbl, &V_I4(pVarResult));
2083 ERR("invalid dispid 0x%x or wFlags 0x%x\n", dispIdMember, wFlags);
2084 return DISP_E_MEMBERNOTFOUND;
2088 static const IPictureVtbl OLEPictureImpl_VTable =
2090 OLEPictureImpl_QueryInterface,
2091 OLEPictureImpl_AddRef,
2092 OLEPictureImpl_Release,
2093 OLEPictureImpl_get_Handle,
2094 OLEPictureImpl_get_hPal,
2095 OLEPictureImpl_get_Type,
2096 OLEPictureImpl_get_Width,
2097 OLEPictureImpl_get_Height,
2098 OLEPictureImpl_Render,
2099 OLEPictureImpl_set_hPal,
2100 OLEPictureImpl_get_CurDC,
2101 OLEPictureImpl_SelectPicture,
2102 OLEPictureImpl_get_KeepOriginalFormat,
2103 OLEPictureImpl_put_KeepOriginalFormat,
2104 OLEPictureImpl_PictureChanged,
2105 OLEPictureImpl_SaveAsFile,
2106 OLEPictureImpl_get_Attributes
2109 static const IDispatchVtbl OLEPictureImpl_IDispatch_VTable =
2111 OLEPictureImpl_IDispatch_QueryInterface,
2112 OLEPictureImpl_IDispatch_AddRef,
2113 OLEPictureImpl_IDispatch_Release,
2114 OLEPictureImpl_GetTypeInfoCount,
2115 OLEPictureImpl_GetTypeInfo,
2116 OLEPictureImpl_GetIDsOfNames,
2117 OLEPictureImpl_Invoke
2120 static const IPersistStreamVtbl OLEPictureImpl_IPersistStream_VTable =
2122 OLEPictureImpl_IPersistStream_QueryInterface,
2123 OLEPictureImpl_IPersistStream_AddRef,
2124 OLEPictureImpl_IPersistStream_Release,
2125 OLEPictureImpl_GetClassID,
2126 OLEPictureImpl_IsDirty,
2127 OLEPictureImpl_Load,
2128 OLEPictureImpl_Save,
2129 OLEPictureImpl_GetSizeMax
2132 static const IConnectionPointContainerVtbl OLEPictureImpl_IConnectionPointContainer_VTable =
2134 OLEPictureImpl_IConnectionPointContainer_QueryInterface,
2135 OLEPictureImpl_IConnectionPointContainer_AddRef,
2136 OLEPictureImpl_IConnectionPointContainer_Release,
2137 OLEPictureImpl_EnumConnectionPoints,
2138 OLEPictureImpl_FindConnectionPoint
2141 /***********************************************************************
2142 * OleCreatePictureIndirect (OLEAUT32.419)
2144 HRESULT WINAPI OleCreatePictureIndirect(LPPICTDESC lpPictDesc, REFIID riid,
2145 BOOL Own, void **ppvObj )
2147 OLEPictureImpl* newPict;
2150 TRACE("(%p,%s,%d,%p)\n", lpPictDesc, debugstr_guid(riid), Own, ppvObj);
2154 newPict = OLEPictureImpl_Construct(lpPictDesc, Own);
2156 if (newPict == NULL)
2157 return E_OUTOFMEMORY;
2160 * Make sure it supports the interface required by the caller.
2162 hr = IPicture_QueryInterface((IPicture*)newPict, riid, ppvObj);
2165 * Release the reference obtained in the constructor. If
2166 * the QueryInterface was unsuccessful, it will free the class.
2168 IPicture_Release((IPicture*)newPict);
2174 /***********************************************************************
2175 * OleLoadPicture (OLEAUT32.418)
2177 HRESULT WINAPI OleLoadPicture( LPSTREAM lpstream, LONG lSize, BOOL fRunmode,
2178 REFIID riid, LPVOID *ppvObj )
2184 TRACE("(%p,%d,%d,%s,%p), partially implemented.\n",
2185 lpstream, lSize, fRunmode, debugstr_guid(riid), ppvObj);
2187 hr = OleCreatePictureIndirect(NULL,riid,!fRunmode,(LPVOID*)&newpic);
2190 hr = IPicture_QueryInterface(newpic,&IID_IPersistStream, (LPVOID*)&ps);
2192 ERR("Could not get IPersistStream iface from Ole Picture?\n");
2193 IPicture_Release(newpic);
2197 hr = IPersistStream_Load(ps,lpstream);
2198 IPersistStream_Release(ps);
2201 ERR("IPersistStream_Load failed\n");
2202 IPicture_Release(newpic);
2206 hr = IPicture_QueryInterface(newpic,riid,ppvObj);
2208 ERR("Failed to get interface %s from IPicture.\n",debugstr_guid(riid));
2209 IPicture_Release(newpic);
2213 /***********************************************************************
2214 * OleLoadPictureEx (OLEAUT32.401)
2216 HRESULT WINAPI OleLoadPictureEx( LPSTREAM lpstream, LONG lSize, BOOL fRunmode,
2217 REFIID riid, DWORD xsiz, DWORD ysiz, DWORD flags, LPVOID *ppvObj )
2223 FIXME("(%p,%d,%d,%s,x=%d,y=%d,f=%x,%p), partially implemented.\n",
2224 lpstream, lSize, fRunmode, debugstr_guid(riid), xsiz, ysiz, flags, ppvObj);
2226 hr = OleCreatePictureIndirect(NULL,riid,!fRunmode,(LPVOID*)&newpic);
2229 hr = IPicture_QueryInterface(newpic,&IID_IPersistStream, (LPVOID*)&ps);
2231 ERR("Could not get IPersistStream iface from Ole Picture?\n");
2232 IPicture_Release(newpic);
2236 hr = IPersistStream_Load(ps,lpstream);
2237 IPersistStream_Release(ps);
2240 ERR("IPersistStream_Load failed\n");
2241 IPicture_Release(newpic);
2245 hr = IPicture_QueryInterface(newpic,riid,ppvObj);
2247 ERR("Failed to get interface %s from IPicture.\n",debugstr_guid(riid));
2248 IPicture_Release(newpic);
2252 /***********************************************************************
2253 * OleLoadPicturePath (OLEAUT32.424)
2255 HRESULT WINAPI OleLoadPicturePath( LPOLESTR szURLorPath, LPUNKNOWN punkCaller,
2256 DWORD dwReserved, OLE_COLOR clrReserved, REFIID riid,
2259 static const WCHAR file[] = { 'f','i','l','e',':',0 };
2263 HGLOBAL hGlobal = NULL;
2264 DWORD dwBytesRead = 0;
2267 IPersistStream *pStream;
2270 WCHAR *file_candidate;
2271 WCHAR path_buf[MAX_PATH];
2273 TRACE("(%s,%p,%d,%08x,%s,%p): stub\n",
2274 debugstr_w(szURLorPath), punkCaller, dwReserved, clrReserved,
2275 debugstr_guid(riid), ppvRet);
2277 if (!szURLorPath || !ppvRet)
2278 return E_INVALIDARG;
2282 /* Convert file URLs to DOS paths. */
2283 if (strncmpW(szURLorPath, file, 5) == 0) {
2285 hRes = CoInternetParseUrl(szURLorPath, PARSE_PATH_FROM_URL, 0, path_buf,
2286 sizeof(path_buf)/sizeof(WCHAR), &size, 0);
2290 file_candidate = path_buf;
2293 file_candidate = szURLorPath;
2295 /* Handle candidate DOS paths separately. */
2296 if (file_candidate[1] == ':') {
2297 hFile = CreateFileW(file_candidate, GENERIC_READ, 0, NULL, OPEN_EXISTING,
2299 if (hFile == INVALID_HANDLE_VALUE)
2300 return E_UNEXPECTED;
2302 dwFileSize = GetFileSize(hFile, NULL);
2303 if (dwFileSize != INVALID_FILE_SIZE )
2305 hGlobal = GlobalAlloc(GMEM_FIXED,dwFileSize);
2308 bRead = ReadFile(hFile, hGlobal, dwFileSize, &dwBytesRead, NULL);
2311 GlobalFree(hGlobal);
2319 return E_UNEXPECTED;
2321 hRes = CreateStreamOnHGlobal(hGlobal, TRUE, &stream);
2324 GlobalFree(hGlobal);
2331 hRes = CreateBindCtx(0, &pbc);
2332 if (SUCCEEDED(hRes))
2334 hRes = CreateURLMoniker(NULL, szURLorPath, &pmnk);
2335 if (SUCCEEDED(hRes))
2337 hRes = IMoniker_BindToStorage(pmnk, pbc, NULL, &IID_IStream, (LPVOID*)&stream);
2338 IMoniker_Release(pmnk);
2340 IBindCtx_Release(pbc);
2346 init_res = CoInitialize(NULL);
2348 hRes = CoCreateInstance(&CLSID_StdPicture, punkCaller, CLSCTX_INPROC_SERVER,
2349 &IID_IPicture, (LPVOID*)&ipicture);
2350 if (SUCCEEDED(hRes)) {
2351 hRes = IPicture_QueryInterface(ipicture, &IID_IPersistStream, (LPVOID*)&pStream);
2353 if (SUCCEEDED(hRes)) {
2354 hRes = IPersistStream_Load(pStream, stream);
2356 if (SUCCEEDED(hRes)) {
2357 hRes = IPicture_QueryInterface(ipicture, riid, ppvRet);
2360 ERR("Failed to get interface %s from IPicture.\n", debugstr_guid(riid));
2362 IPersistStream_Release(pStream);
2364 IPicture_Release(ipicture);
2367 IStream_Release(stream);
2369 if (SUCCEEDED(init_res))
2375 /*******************************************************************************
2376 * StdPic ClassFactory
2380 /* IUnknown fields */
2381 const IClassFactoryVtbl *lpVtbl;
2383 } IClassFactoryImpl;
2385 static HRESULT WINAPI
2386 SPCF_QueryInterface(LPCLASSFACTORY iface,REFIID riid,LPVOID *ppobj) {
2387 IClassFactoryImpl *This = (IClassFactoryImpl *)iface;
2389 FIXME("(%p)->(%s,%p),stub!\n",This,debugstr_guid(riid),ppobj);
2390 return E_NOINTERFACE;
2394 SPCF_AddRef(LPCLASSFACTORY iface) {
2395 IClassFactoryImpl *This = (IClassFactoryImpl *)iface;
2396 return InterlockedIncrement(&This->ref);
2399 static ULONG WINAPI SPCF_Release(LPCLASSFACTORY iface) {
2400 IClassFactoryImpl *This = (IClassFactoryImpl *)iface;
2401 /* static class, won't be freed */
2402 return InterlockedDecrement(&This->ref);
2405 static HRESULT WINAPI SPCF_CreateInstance(
2406 LPCLASSFACTORY iface,LPUNKNOWN pOuter,REFIID riid,LPVOID *ppobj
2408 /* Creates an uninitialized picture */
2409 return OleCreatePictureIndirect(NULL,riid,TRUE,ppobj);
2413 static HRESULT WINAPI SPCF_LockServer(LPCLASSFACTORY iface,BOOL dolock) {
2414 IClassFactoryImpl *This = (IClassFactoryImpl *)iface;
2415 FIXME("(%p)->(%d),stub!\n",This,dolock);
2419 static const IClassFactoryVtbl SPCF_Vtbl = {
2420 SPCF_QueryInterface,
2423 SPCF_CreateInstance,
2426 static IClassFactoryImpl STDPIC_CF = {&SPCF_Vtbl, 1 };
2428 void _get_STDPIC_CF(LPVOID *ppv) { *ppv = &STDPIC_CF; }