2 * Ole 2 Create functions implementation
4 * Copyright (C) 1999-2000 Abey George
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
25 #define NONAMELESSUNION
31 #include "wine/debug.h"
34 #include "compobj_private.h"
36 WINE_DEFAULT_DEBUG_CHANNEL(ole
);
38 /******************************************************************************
39 * OleQueryCreateFromData [OLE32.@]
41 * Checks whether an object can become an embedded object.
42 * the clipboard or OLE drag and drop.
43 * Returns : S_OK - Format that supports Embedded object creation are present.
44 * OLE_E_STATIC - Format that supports static object creation are present.
45 * S_FALSE - No acceptable format is available.
48 HRESULT WINAPI
OleQueryCreateFromData(IDataObject
*data
)
50 IEnumFORMATETC
*enum_fmt
;
52 BOOL found_static
= FALSE
;
55 hr
= IDataObject_EnumFormatEtc(data
, DATADIR_GET
, &enum_fmt
);
57 if(FAILED(hr
)) return hr
;
61 hr
= IEnumFORMATETC_Next(enum_fmt
, 1, &fmt
, NULL
);
64 if(fmt
.cfFormat
== embedded_object_clipboard_format
||
65 fmt
.cfFormat
== embed_source_clipboard_format
||
66 fmt
.cfFormat
== filename_clipboard_format
)
68 IEnumFORMATETC_Release(enum_fmt
);
72 if(fmt
.cfFormat
== CF_METAFILEPICT
||
73 fmt
.cfFormat
== CF_BITMAP
||
74 fmt
.cfFormat
== CF_DIB
)
79 IEnumFORMATETC_Release(enum_fmt
);
81 return found_static
? OLE_S_STATIC
: S_FALSE
;
84 static inline void init_fmtetc(FORMATETC
*fmt
, CLIPFORMAT cf
, TYMED tymed
)
88 fmt
->dwAspect
= DVASPECT_CONTENT
;
93 /***************************************************************************
96 * Retrieve an object's storage from a variety of sources.
100 static HRESULT
get_storage(IDataObject
*data
, IStorage
*stg
, UINT
*src_cf
, BOOL other_fmts
)
102 static const UINT fmt_id
[] = { CF_METAFILEPICT
, CF_BITMAP
, CF_DIB
};
107 IPersistStorage
*persist
;
110 if (src_cf
) *src_cf
= 0;
112 /* CF_EMBEDEDOBJECT */
113 init_fmtetc(&fmt
, embedded_object_clipboard_format
, TYMED_ISTORAGE
);
114 med
.tymed
= TYMED_ISTORAGE
;
116 med
.pUnkForRelease
= NULL
;
117 hr
= IDataObject_GetDataHere(data
, &fmt
, &med
);
120 if (src_cf
) *src_cf
= embedded_object_clipboard_format
;
125 init_fmtetc(&fmt
, embed_source_clipboard_format
, TYMED_ISTORAGE
);
126 med
.tymed
= TYMED_ISTORAGE
;
128 med
.pUnkForRelease
= NULL
;
129 hr
= IDataObject_GetDataHere(data
, &fmt
, &med
);
132 if (src_cf
) *src_cf
= embed_source_clipboard_format
;
138 for (i
= 0; i
< sizeof(fmt_id
)/sizeof(fmt_id
[0]); i
++)
140 init_fmtetc(&fmt
, fmt_id
[i
], TYMED_ISTORAGE
);
141 hr
= IDataObject_QueryGetData(data
, &fmt
);
144 if (src_cf
) *src_cf
= fmt_id
[i
];
150 /* IPersistStorage */
151 hr
= IDataObject_QueryInterface(data
, &IID_IPersistStorage
, (void**)&persist
);
152 if(FAILED(hr
)) return hr
;
154 hr
= IPersistStorage_GetClassID(persist
, &clsid
);
155 if(FAILED(hr
)) goto end
;
157 hr
= IStorage_SetClass(stg
, &clsid
);
158 if(FAILED(hr
)) goto end
;
160 hr
= IPersistStorage_Save(persist
, stg
, FALSE
);
161 if(FAILED(hr
)) goto end
;
163 hr
= IPersistStorage_SaveCompleted(persist
, NULL
);
166 IPersistStorage_Release(persist
);
171 /******************************************************************************
172 * OleCreateFromDataEx [OLE32.@]
174 * Creates an embedded object from data transfer object retrieved from
175 * the clipboard or OLE drag and drop.
177 HRESULT WINAPI
OleCreateFromDataEx(IDataObject
*data
, REFIID iid
, DWORD flags
,
178 DWORD renderopt
, ULONG num_cache_fmts
, DWORD
*adv_flags
, FORMATETC
*cache_fmts
,
179 IAdviseSink
*sink
, DWORD
*conns
,
180 IOleClientSite
*client_site
, IStorage
*stg
, void **obj
)
185 FIXME("(%p, %s, %08x, %08x, %d, %p, %p, %p, %p, %p, %p, %p): stub\n",
186 data
, debugstr_guid(iid
), flags
, renderopt
, num_cache_fmts
, adv_flags
, cache_fmts
,
187 sink
, conns
, client_site
, stg
, obj
);
189 hr
= get_storage(data
, stg
, &src_cf
, TRUE
);
190 if(FAILED(hr
)) return hr
;
192 hr
= OleLoad(stg
, iid
, client_site
, obj
);
193 if(FAILED(hr
)) return hr
;
195 /* FIXME: Init cache */
200 /******************************************************************************
201 * OleCreateFromData [OLE32.@]
203 HRESULT WINAPI
OleCreateFromData(LPDATAOBJECT data
, REFIID iid
,
204 DWORD renderopt
, LPFORMATETC fmt
,
205 LPOLECLIENTSITE client_site
, LPSTORAGE stg
,
208 DWORD advf
= ADVF_PRIMEFIRST
;
210 return OleCreateFromDataEx(data
, iid
, 0, renderopt
, fmt
? 1 : 0, fmt
? &advf
: NULL
,
211 fmt
, NULL
, NULL
, client_site
, stg
, obj
);
214 /******************************************************************************
215 * OleCreateLinkFromData [OLE32.@]
217 HRESULT WINAPI
OleCreateLinkFromData(IDataObject
*data
, REFIID iid
,
218 DWORD renderopt
, FORMATETC
*fmt
,
219 IOleClientSite
*client_site
, IStorage
*stg
,
222 FIXME("%p,%s,%08x,%p,%p,%p,%p: semi-stub\n",
223 data
, debugstr_guid(iid
), renderopt
, fmt
, client_site
, stg
, obj
);
224 return OleCreateFromData(data
, iid
, renderopt
, fmt
, client_site
, stg
, obj
);
227 /******************************************************************************
228 * OleCreateStaticFromData [OLE32.@]
230 HRESULT WINAPI
OleCreateStaticFromData(IDataObject
*data
, REFIID iid
,
231 DWORD renderopt
, FORMATETC
*fmt
,
232 IOleClientSite
*client_site
, IStorage
*stg
,
235 FIXME("%p,%s,%08x,%p,%p,%p,%p: semi-stub\n",
236 data
, debugstr_guid(iid
), renderopt
, fmt
, client_site
, stg
, obj
);
237 return OleCreateFromData(data
, iid
, renderopt
, fmt
, client_site
, stg
, obj
);
240 /******************************************************************************
241 * OleCreateFromFileEx [OLE32.@]
243 HRESULT WINAPI
OleCreateFromFileEx(REFCLSID clsid
, const OLECHAR
*filename
, REFIID iid
, DWORD flags
,
244 DWORD renderopt
, ULONG num_fmts
, DWORD
*adv_flags
, FORMATETC
*fmts
, IAdviseSink
*sink
,
245 DWORD
*conns
, IOleClientSite
*client_site
, IStorage
*stg
, void **obj
)
250 IUnknown
*unk
= NULL
;
251 IOleCache
*cache
= NULL
;
254 TRACE("cls %s, %s, iid %s, flags %d, render opts %d, num fmts %d, adv flags %p, fmts %p\n", debugstr_guid(clsid
),
255 debugstr_w(filename
), debugstr_guid(iid
), flags
, renderopt
, num_fmts
, adv_flags
, fmts
);
256 TRACE("sink %p, conns %p, client site %p, storage %p, obj %p\n", sink
, conns
, client_site
, stg
, obj
);
257 for (i
= 0; i
< num_fmts
; i
++)
258 TRACE("\t%d: fmt %s adv flags %d\n", i
, debugstr_formatetc(fmts
+ i
), adv_flags
[i
]);
260 hr
= CreateFileMoniker( filename
, &mon
);
261 if (FAILED(hr
)) return hr
;
263 hr
= BindMoniker( mon
, 0, &IID_IDataObject
, (void**)&data
);
264 IMoniker_Release( mon
);
265 if (FAILED(hr
)) return hr
;
267 hr
= get_storage( data
, stg
, NULL
, FALSE
);
268 if (FAILED(hr
)) goto end
;
270 hr
= OleLoad( stg
, &IID_IUnknown
, client_site
, (void**)&unk
);
271 if (FAILED(hr
)) goto end
;
273 if (renderopt
== OLERENDER_FORMAT
)
275 hr
= IUnknown_QueryInterface( unk
, &IID_IOleCache
, (void**)&cache
);
276 if (FAILED(hr
)) goto end
;
278 for (i
= 0; i
< num_fmts
; i
++)
283 memset( &med
, 0, sizeof(med
) );
284 hr
= IDataObject_GetData( data
, fmts
+ i
, &med
);
285 if (FAILED(hr
)) goto end
;
286 hr
= IOleCache_Cache( cache
, fmts
+ i
, adv_flags
[i
], &dummy_conn
);
288 hr
= IOleCache_SetData( cache
, fmts
+ i
, &med
, TRUE
);
291 ReleaseStgMedium( &med
);
297 hr
= IUnknown_QueryInterface( unk
, iid
, obj
);
300 if (cache
) IOleCache_Release( cache
);
301 if (unk
) IUnknown_Release( unk
);
302 IDataObject_Release( data
);
306 /******************************************************************************
307 * OleCreateFromFile [OLE32.@]
309 HRESULT WINAPI
OleCreateFromFile(REFCLSID clsid
, const OLECHAR
*filename
, REFIID iid
, DWORD renderopt
,
310 FORMATETC
*fmt
, IOleClientSite
*client_site
, IStorage
*storage
, void **obj
)
312 DWORD advf
= ADVF_PRIMEFIRST
;
314 return OleCreateFromFileEx(clsid
, filename
, iid
, 0, renderopt
, fmt
? 1 : 0, fmt
? &advf
: NULL
, fmt
,
315 NULL
, NULL
, client_site
, storage
, obj
);
318 /******************************************************************************
319 * OleDuplicateData [OLE32.@]
321 * Duplicates clipboard data.
324 * hSrc [I] Handle of the source clipboard data.
325 * cfFormat [I] The clipboard format of hSrc.
326 * uiFlags [I] Flags to pass to GlobalAlloc.
329 * Success: handle to the duplicated data.
332 HANDLE WINAPI
OleDuplicateData(HANDLE hSrc
, CLIPFORMAT cfFormat
,
337 TRACE("(%p,%x,%x)\n", hSrc
, cfFormat
, uiFlags
);
339 if (!uiFlags
) uiFlags
= GMEM_MOVEABLE
;
344 hDst
= CopyEnhMetaFileW(hSrc
, NULL
);
346 case CF_METAFILEPICT
:
347 hDst
= CopyMetaFileW(hSrc
, NULL
);
351 LOGPALETTE
* logpalette
;
352 UINT nEntries
= GetPaletteEntries(hSrc
, 0, 0, NULL
);
353 if (!nEntries
) return NULL
;
354 logpalette
= HeapAlloc(GetProcessHeap(), 0,
355 FIELD_OFFSET(LOGPALETTE
, palPalEntry
[nEntries
]));
356 if (!logpalette
) return NULL
;
357 if (!GetPaletteEntries(hSrc
, 0, nEntries
, logpalette
->palPalEntry
))
359 HeapFree(GetProcessHeap(), 0, logpalette
);
362 logpalette
->palVersion
= 0x300;
363 logpalette
->palNumEntries
= (WORD
)nEntries
;
365 hDst
= CreatePalette(logpalette
);
367 HeapFree(GetProcessHeap(), 0, logpalette
);
374 if (!GetObjectW(hSrc
, sizeof(bm
), &bm
))
376 size
= GetBitmapBits(hSrc
, 0, NULL
);
377 if (!size
) return NULL
;
378 bm
.bmBits
= HeapAlloc(GetProcessHeap(), 0, size
);
379 if (!bm
.bmBits
) return NULL
;
380 if (GetBitmapBits(hSrc
, size
, bm
.bmBits
))
381 hDst
= CreateBitmapIndirect(&bm
);
382 HeapFree(GetProcessHeap(), 0, bm
.bmBits
);
387 SIZE_T size
= GlobalSize(hSrc
);
391 /* allocate space for object */
392 if (!size
) return NULL
;
393 hDst
= GlobalAlloc(uiFlags
, size
);
394 if (!hDst
) return NULL
;
397 pvSrc
= GlobalLock(hSrc
);
403 pvDst
= GlobalLock(hDst
);
411 memcpy(pvDst
, pvSrc
, size
);
419 TRACE("returning %p\n", hDst
);