1 /*
2 * OLE Font encapsulation implementation
3 *
4 * This file contains an implementation of the IFont
5 * interface and the OleCreateFontIndirect API call.
6 *
7 * Copyright 1999 Francis Beaudet
8 * Copyright 2006 (Google) Benjamin Arai
9 *
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.
14 *
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.
19 *
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
23 */
24 #include <assert.h>
25 #include <stdarg.h>
26 #include <string.h>
27
28 #define COBJMACROS
29 #define NONAMELESSUNION
30 #define NONAMELESSSTRUCT
31
32 #include "winerror.h"
33 #include "windef.h"
34 #include "winbase.h"
35 #include "wingdi.h"
36 #include "winuser.h"
37 #include "wine/list.h"
38 #include "objbase.h"
39 #include "oleauto.h" /* for SysAllocString(....) */
40 #include "ole2.h"
41 #include "olectl.h"
42 #include "wine/debug.h"
43 #include "connpt.h" /* for CreateConnectionPoint */
44 #include "oaidl.h"
45
46 WINE_DEFAULT_DEBUG_CHANNEL(ole);
47
48 /***********************************************************************
49 * Declaration of constants used when serializing the font object.
50 */
51 #define FONTPERSIST_ITALIC 0x02
52 #define FONTPERSIST_UNDERLINE 0x04
53 #define FONTPERSIST_STRIKETHROUGH 0x08
54
55 static HDC olefont_hdc;
56
57 /***********************************************************************
58 * List of the HFONTs it has given out, with each one having a separate
59 * ref count.
60 */
61 typedef struct _HFONTItem
62 {
63 struct list entry;
64
65 /* Reference count of any IFont objects that own this hfont */
66 LONG int_refs;
67
68 /* Total reference count of any refs held by the application obtained by AddRefHfont plus any internal refs */
69 LONG total_refs;
70
71 /* The font associated with this object. */
72 HFONT gdiFont;
73
74 } HFONTItem, *PHFONTItem;
75
76 static struct list OLEFontImpl_hFontList = LIST_INIT(OLEFontImpl_hFontList);
77
78 /* Counts how many fonts contain at least one lock */
79 static LONG ifont_cnt = 0;
80
81 /***********************************************************************
82 * Critical section for OLEFontImpl_hFontList
83 */
84 static CRITICAL_SECTION OLEFontImpl_csHFONTLIST;
85 static CRITICAL_SECTION_DEBUG OLEFontImpl_csHFONTLIST_debug =
86 {
87 0, 0, &OLEFontImpl_csHFONTLIST,
88 { &OLEFontImpl_csHFONTLIST_debug.ProcessLocksList,
89 &OLEFontImpl_csHFONTLIST_debug.ProcessLocksList },
90 0, 0, { (DWORD_PTR)(__FILE__ ": OLEFontImpl_csHFONTLIST") }
91 };
92 static CRITICAL_SECTION OLEFontImpl_csHFONTLIST = { &OLEFontImpl_csHFONTLIST_debug, -1, 0, 0, 0, 0 };
93
get_dc(void)94 static HDC get_dc(void)
95 {
96 HDC hdc;
97 EnterCriticalSection(&OLEFontImpl_csHFONTLIST);
98 if(!olefont_hdc)
99 olefont_hdc = CreateCompatibleDC(NULL);
100 hdc = olefont_hdc;
101 LeaveCriticalSection(&OLEFontImpl_csHFONTLIST);
102 return hdc;
103 }
104
delete_dc(void)105 static void delete_dc(void)
106 {
107 EnterCriticalSection(&OLEFontImpl_csHFONTLIST);
108 if(olefont_hdc)
109 {
110 DeleteDC(olefont_hdc);
111 olefont_hdc = NULL;
112 }
113 LeaveCriticalSection(&OLEFontImpl_csHFONTLIST);
114 }
115
HFONTItem_Delete(PHFONTItem item)116 static void HFONTItem_Delete(PHFONTItem item)
117 {
118 DeleteObject(item->gdiFont);
119 list_remove(&item->entry);
120 HeapFree(GetProcessHeap(), 0, item);
121 }
122
123 /* Find hfont item entry in the list. Should be called while holding the crit sect */
find_hfontitem(HFONT hfont)124 static HFONTItem *find_hfontitem(HFONT hfont)
125 {
126 HFONTItem *item;
127
128 LIST_FOR_EACH_ENTRY(item, &OLEFontImpl_hFontList, HFONTItem, entry)
129 {
130 if (item->gdiFont == hfont)
131 return item;
132 }
133 return NULL;
134 }
135
136 /* Add an item to the list with one internal reference */
add_hfontitem(HFONT hfont)137 static HRESULT add_hfontitem(HFONT hfont)
138 {
139 HFONTItem *new_item = HeapAlloc(GetProcessHeap(), 0, sizeof(*new_item));
140
141 if(!new_item) return E_OUTOFMEMORY;
142
143 new_item->int_refs = 1;
144 new_item->total_refs = 1;
145 new_item->gdiFont = hfont;
146 EnterCriticalSection(&OLEFontImpl_csHFONTLIST);
147 list_add_tail(&OLEFontImpl_hFontList,&new_item->entry);
148 LeaveCriticalSection(&OLEFontImpl_csHFONTLIST);
149 return S_OK;
150 }
151
inc_int_ref(HFONT hfont)152 static HRESULT inc_int_ref(HFONT hfont)
153 {
154 HFONTItem *item;
155 HRESULT hr = S_FALSE;
156
157 EnterCriticalSection(&OLEFontImpl_csHFONTLIST);
158 item = find_hfontitem(hfont);
159
160 if(item)
161 {
162 item->int_refs++;
163 item->total_refs++;
164 hr = S_OK;
165 }
166 LeaveCriticalSection(&OLEFontImpl_csHFONTLIST);
167
168 return hr;
169 }
170
171 /* decrements the internal ref of a hfont item. If both refs are zero it'll
172 remove the item from the list and delete the hfont */
dec_int_ref(HFONT hfont)173 static HRESULT dec_int_ref(HFONT hfont)
174 {
175 HFONTItem *item;
176 HRESULT hr = S_FALSE;
177
178 EnterCriticalSection(&OLEFontImpl_csHFONTLIST);
179 item = find_hfontitem(hfont);
180
181 if(item)
182 {
183 item->int_refs--;
184 item->total_refs--;
185 if(item->int_refs == 0 && item->total_refs == 0)
186 HFONTItem_Delete(item);
187 hr = S_OK;
188 }
189 LeaveCriticalSection(&OLEFontImpl_csHFONTLIST);
190
191 return hr;
192 }
193
inc_ext_ref(HFONT hfont)194 static HRESULT inc_ext_ref(HFONT hfont)
195 {
196 HFONTItem *item;
197 HRESULT hr = S_FALSE;
198
199 EnterCriticalSection(&OLEFontImpl_csHFONTLIST);
200
201 item = find_hfontitem(hfont);
202 if(item)
203 {
204 item->total_refs++;
205 hr = S_OK;
206 }
207 LeaveCriticalSection(&OLEFontImpl_csHFONTLIST);
208
209 return hr;
210 }
211
dec_ext_ref(HFONT hfont)212 static HRESULT dec_ext_ref(HFONT hfont)
213 {
214 HFONTItem *item;
215 HRESULT hr = S_FALSE;
216
217 EnterCriticalSection(&OLEFontImpl_csHFONTLIST);
218
219 item = find_hfontitem(hfont);
220 if(item)
221 {
222 if(--item->total_refs >= 0) hr = S_OK;
223 }
224 LeaveCriticalSection(&OLEFontImpl_csHFONTLIST);
225
226 return hr;
227 }
228
strdupW(const WCHAR * str)229 static WCHAR *strdupW(const WCHAR* str)
230 {
231 WCHAR *ret;
232 DWORD size = (lstrlenW(str) + 1) * sizeof(WCHAR);
233
234 ret = HeapAlloc(GetProcessHeap(), 0, size);
235 if(ret)
236 memcpy(ret, str, size);
237 return ret;
238 }
239
240 /***********************************************************************
241 * Declaration of the implementation class for the IFont interface
242 */
243 typedef struct OLEFontImpl OLEFontImpl;
244
245 struct OLEFontImpl
246 {
247 /*
248 * This class supports many interfaces. IUnknown, IFont,
249 * IDispatch, IDispFont IPersistStream and IConnectionPointContainer.
250 * The first two are supported by the first vtable, the next two are
251 * supported by the second table and the last two have their own.
252 */
253 IFont IFont_iface;
254 IDispatch IDispatch_iface;
255 IPersistStream IPersistStream_iface;
256 IConnectionPointContainer IConnectionPointContainer_iface;
257 IPersistPropertyBag IPersistPropertyBag_iface;
258 IPersistStreamInit IPersistStreamInit_iface;
259 /*
260 * Reference count for that instance of the class.
261 */
262 LONG ref;
263
264 /*
265 * This structure contains the description of the class.
266 */
267 FONTDESC description;
268
269 /*
270 * Contain the font associated with this object.
271 */
272 HFONT gdiFont;
273 BOOL dirty;
274 /*
275 * Size ratio
276 */
277 LONG cyLogical;
278 LONG cyHimetric;
279
280 /*
281 * Stash realized height (pixels) from TEXTMETRIC - used in get_Size()
282 */
283 LONG nRealHeight;
284
285 IConnectionPoint *pPropertyNotifyCP;
286 IConnectionPoint *pFontEventsCP;
287 };
288
impl_from_IFont(IFont * iface)289 static inline OLEFontImpl *impl_from_IFont(IFont *iface)
290 {
291 return CONTAINING_RECORD(iface, OLEFontImpl, IFont_iface);
292 }
293
impl_from_IDispatch(IDispatch * iface)294 static inline OLEFontImpl *impl_from_IDispatch( IDispatch *iface )
295 {
296 return CONTAINING_RECORD(iface, OLEFontImpl, IDispatch_iface);
297 }
298
impl_from_IPersistStream(IPersistStream * iface)299 static inline OLEFontImpl *impl_from_IPersistStream( IPersistStream *iface )
300 {
301 return CONTAINING_RECORD(iface, OLEFontImpl, IPersistStream_iface);
302 }
303
impl_from_IConnectionPointContainer(IConnectionPointContainer * iface)304 static inline OLEFontImpl *impl_from_IConnectionPointContainer( IConnectionPointContainer *iface )
305 {
306 return CONTAINING_RECORD(iface, OLEFontImpl, IConnectionPointContainer_iface);
307 }
308
impl_from_IPersistPropertyBag(IPersistPropertyBag * iface)309 static inline OLEFontImpl *impl_from_IPersistPropertyBag( IPersistPropertyBag *iface )
310 {
311 return CONTAINING_RECORD(iface, OLEFontImpl, IPersistPropertyBag_iface);
312 }
313
impl_from_IPersistStreamInit(IPersistStreamInit * iface)314 static inline OLEFontImpl *impl_from_IPersistStreamInit( IPersistStreamInit *iface )
315 {
316 return CONTAINING_RECORD(iface, OLEFontImpl, IPersistStreamInit_iface);
317 }
318
319
320 /***********************************************************************
321 * Prototypes for the implementation functions for the IFont
322 * interface
323 */
324 static OLEFontImpl* OLEFontImpl_Construct(const FONTDESC *fontDesc);
325 static void OLEFontImpl_Destroy(OLEFontImpl* fontDesc);
326 static ULONG WINAPI OLEFontImpl_AddRef(IFont* iface);
327
328 /******************************************************************************
329 * OleCreateFontIndirect [OLEAUT32.420]
330 */
OleCreateFontIndirect(LPFONTDESC lpFontDesc,REFIID riid,LPVOID * ppvObj)331 HRESULT WINAPI OleCreateFontIndirect(
332 LPFONTDESC lpFontDesc,
333 REFIID riid,
334 LPVOID* ppvObj)
335 {
336 OLEFontImpl* newFont;
337 HRESULT hr;
338 FONTDESC fd;
339
340 TRACE("(%p, %s, %p)\n", lpFontDesc, debugstr_guid(riid), ppvObj);
341
342 if (!ppvObj) return E_POINTER;
343
344 *ppvObj = 0;
345
346 if (!lpFontDesc) {
347 static WCHAR fname[] = { 'S','y','s','t','e','m',0 };
348
349 fd.cbSizeofstruct = sizeof(fd);
350 fd.lpstrName = fname;
351 fd.cySize.s.Lo = 80000;
352 fd.cySize.s.Hi = 0;
353 fd.sWeight = 0;
354 fd.sCharset = 0;
355 fd.fItalic = FALSE;
356 fd.fUnderline = FALSE;
357 fd.fStrikethrough = FALSE;
358 lpFontDesc = &fd;
359 }
360
361 newFont = OLEFontImpl_Construct(lpFontDesc);
362 if (!newFont) return E_OUTOFMEMORY;
363
364 hr = IFont_QueryInterface(&newFont->IFont_iface, riid, ppvObj);
365 IFont_Release(&newFont->IFont_iface);
366
367 return hr;
368 }
369
370
371 /***********************************************************************
372 * Implementation of the OLEFontImpl class.
373 */
374
375 /***********************************************************************
376 * OLEFont_SendNotify (internal)
377 *
378 * Sends notification messages of changed properties to any interested
379 * connections.
380 */
OLEFont_SendNotify(OLEFontImpl * this,DISPID dispID)381 static void OLEFont_SendNotify(OLEFontImpl* this, DISPID dispID)
382 {
383 static const WCHAR wszName[] = {'N','a','m','e',0};
384 static const WCHAR wszSize[] = {'S','i','z','e',0};
385 static const WCHAR wszBold[] = {'B','o','l','d',0};
386 static const WCHAR wszItalic[] = {'I','t','a','l','i','c',0};
387 static const WCHAR wszUnder[] = {'U','n','d','e','r','l','i','n','e',0};
388 static const WCHAR wszStrike[] = {'S','t','r','i','k','e','t','h','r','o','u','g','h',0};
389 static const WCHAR wszWeight[] = {'W','e','i','g','h','t',0};
390 static const WCHAR wszCharset[] = {'C','h','a','r','s','e','t',0};
391 static const LPCWSTR dispid_mapping[] =
392 {
393 wszName,
394 NULL,
395 wszSize,
396 wszBold,
397 wszItalic,
398 wszUnder,
399 wszStrike,
400 wszWeight,
401 wszCharset
402 };
403
404 IEnumConnections *pEnum;
405 CONNECTDATA CD;
406 HRESULT hres;
407
408 this->dirty = TRUE;
409
410 hres = IConnectionPoint_EnumConnections(this->pPropertyNotifyCP, &pEnum);
411 if (SUCCEEDED(hres))
412 {
413 while(IEnumConnections_Next(pEnum, 1, &CD, NULL) == S_OK) {
414 IPropertyNotifySink *sink;
415
416 IUnknown_QueryInterface(CD.pUnk, &IID_IPropertyNotifySink, (void**)&sink);
417 IPropertyNotifySink_OnChanged(sink, dispID);
418 IPropertyNotifySink_Release(sink);
419 IUnknown_Release(CD.pUnk);
420 }
421 IEnumConnections_Release(pEnum);
422 }
423
424 hres = IConnectionPoint_EnumConnections(this->pFontEventsCP, &pEnum);
425 if (SUCCEEDED(hres))
426 {
427 DISPPARAMS dispparams;
428 VARIANTARG vararg;
429
430 VariantInit(&vararg);
431 V_VT(&vararg) = VT_BSTR;
432 V_BSTR(&vararg) = SysAllocString(dispid_mapping[dispID]);
433
434 dispparams.cArgs = 1;
435 dispparams.cNamedArgs = 0;
436 dispparams.rgdispidNamedArgs = NULL;
437 dispparams.rgvarg = &vararg;
438
439 while(IEnumConnections_Next(pEnum, 1, &CD, NULL) == S_OK) {
440 IFontEventsDisp *disp;
441
442 IUnknown_QueryInterface(CD.pUnk, &IID_IFontEventsDisp, (void**)&disp);
443 IFontEventsDisp_Invoke(disp, DISPID_FONT_CHANGED, &IID_NULL,
444 LOCALE_NEUTRAL, INVOKE_FUNC, &dispparams, NULL,
445 NULL, NULL);
446
447 IFontEventsDisp_Release(disp);
448 IUnknown_Release(CD.pUnk);
449 }
450 VariantClear(&vararg);
451 IEnumConnections_Release(pEnum);
452 }
453 }
454
455 /************************************************************************
456 * OLEFontImpl_QueryInterface (IUnknown)
457 *
458 * See Windows documentation for more details on IUnknown methods.
459 */
OLEFontImpl_QueryInterface(IFont * iface,REFIID riid,void ** ppvObject)460 static HRESULT WINAPI OLEFontImpl_QueryInterface(
461 IFont* iface,
462 REFIID riid,
463 void** ppvObject)
464 {
465 OLEFontImpl *this = impl_from_IFont(iface);
466
467 TRACE("(%p)->(%s, %p)\n", this, debugstr_guid(riid), ppvObject);
468
469 *ppvObject = 0;
470
471 if (IsEqualGUID(&IID_IUnknown, riid) ||
472 IsEqualGUID(&IID_IFont, riid))
473 {
474 *ppvObject = this;
475 }
476 else if (IsEqualGUID(&IID_IDispatch, riid) ||
477 IsEqualGUID(&IID_IFontDisp, riid))
478 {
479 *ppvObject = &this->IDispatch_iface;
480 }
481 else if (IsEqualGUID(&IID_IPersist, riid) ||
482 IsEqualGUID(&IID_IPersistStream, riid))
483 {
484 *ppvObject = &this->IPersistStream_iface;
485 }
486 else if (IsEqualGUID(&IID_IConnectionPointContainer, riid))
487 {
488 *ppvObject = &this->IConnectionPointContainer_iface;
489 }
490 else if (IsEqualGUID(&IID_IPersistPropertyBag, riid))
491 {
492 *ppvObject = &this->IPersistPropertyBag_iface;
493 }
494 else if (IsEqualGUID(&IID_IPersistStreamInit, riid))
495 {
496 *ppvObject = &this->IPersistStreamInit_iface;
497 }
498
499 if (!*ppvObject)
500 {
501 FIXME("() : asking for unsupported interface %s\n", debugstr_guid(riid));
502 return E_NOINTERFACE;
503 }
504
505 IFont_AddRef(iface);
506
507 return S_OK;
508 }
509
510 /************************************************************************
511 * OLEFontImpl_AddRef (IUnknown)
512 */
OLEFontImpl_AddRef(IFont * iface)513 static ULONG WINAPI OLEFontImpl_AddRef(
514 IFont* iface)
515 {
516 OLEFontImpl *this = impl_from_IFont(iface);
517 TRACE("(%p)->(ref=%d)\n", this, this->ref);
518 return InterlockedIncrement(&this->ref);
519 }
520
521 /************************************************************************
522 * OLEFontImpl_Release (IUnknown)
523 */
OLEFontImpl_Release(IFont * iface)524 static ULONG WINAPI OLEFontImpl_Release(IFont* iface)
525 {
526 OLEFontImpl *this = impl_from_IFont(iface);
527 ULONG ref;
528
529 TRACE("(%p)->(ref=%d)\n", this, this->ref);
530
531 ref = InterlockedDecrement(&this->ref);
532
533 if (ref == 0)
534 {
535 ULONG fontlist_refs = InterlockedDecrement(&ifont_cnt);
536
537 /* Final IFont object so destroy font cache */
538 if (fontlist_refs == 0)
539 {
540 HFONTItem *item, *cursor2;
541
542 EnterCriticalSection(&OLEFontImpl_csHFONTLIST);
543 LIST_FOR_EACH_ENTRY_SAFE(item, cursor2, &OLEFontImpl_hFontList, HFONTItem, entry)
544 HFONTItem_Delete(item);
545 LeaveCriticalSection(&OLEFontImpl_csHFONTLIST);
546 delete_dc();
547 }
548 else
549 {
550 dec_int_ref(this->gdiFont);
551 }
552 OLEFontImpl_Destroy(this);
553 }
554
555 return ref;
556 }
557
558 typedef struct
559 {
560 short orig_cs;
561 short avail_cs;
562 } enum_data;
563
font_enum_proc(const LOGFONTW * elf,const TEXTMETRICW * ntm,DWORD type,LPARAM lp)564 static int CALLBACK font_enum_proc(const LOGFONTW *elf, const TEXTMETRICW *ntm, DWORD type, LPARAM lp)
565 {
566 enum_data *data = (enum_data*)lp;
567
568 if(elf->lfCharSet == data->orig_cs)
569 {
570 data->avail_cs = data->orig_cs;
571 return 0;
572 }
573 if(data->avail_cs == -1) data->avail_cs = elf->lfCharSet;
574 return 1;
575 }
576
realize_font(OLEFontImpl * This)577 static void realize_font(OLEFontImpl *This)
578 {
579 LOGFONTW logFont;
580 INT fontHeight;
581 WCHAR text_face[LF_FACESIZE];
582 HDC hdc = get_dc();
583 HFONT old_font;
584 TEXTMETRICW tm;
585
586 if (!This->dirty) return;
587
588 text_face[0] = 0;
589
590 if(This->gdiFont)
591 {
592 old_font = SelectObject(hdc, This->gdiFont);
593 GetTextFaceW(hdc, ARRAY_SIZE(text_face), text_face);
594 SelectObject(hdc, old_font);
595 dec_int_ref(This->gdiFont);
596 This->gdiFont = 0;
597 }
598
599 memset(&logFont, 0, sizeof(LOGFONTW));
600
601 lstrcpynW(logFont.lfFaceName, This->description.lpstrName, LF_FACESIZE);
602 logFont.lfCharSet = This->description.sCharset;
603
604 /* If the font name has been changed then enumerate all charsets
605 and pick one that'll result in the font specified being selected */
606 if(text_face[0] && lstrcmpiW(text_face, This->description.lpstrName))
607 {
608 enum_data data;
609 data.orig_cs = This->description.sCharset;
610 data.avail_cs = -1;
611 logFont.lfCharSet = DEFAULT_CHARSET;
612 EnumFontFamiliesExW(get_dc(), &logFont, font_enum_proc, (LPARAM)&data, 0);
613 if(data.avail_cs != -1) logFont.lfCharSet = data.avail_cs;
614 }
615
616 /*
617 * The height of the font returned by the get_Size property is the
618 * height of the font in points multiplied by 10000... Using some
619 * simple conversions and the ratio given by the application, it can
620 * be converted to a height in pixels.
621 *
622 * Standard ratio is 72 / 2540, or 18 / 635 in lowest terms.
623 * Ratio is applied here relative to the standard.
624 */
625
626 fontHeight = MulDiv( This->description.cySize.s.Lo, This->cyLogical*635, This->cyHimetric*18 );
627
628 logFont.lfHeight = ((fontHeight%10000L)>5000L) ? (-fontHeight/10000L) - 1 :
629 (-fontHeight/10000L);
630 logFont.lfItalic = This->description.fItalic;
631 logFont.lfUnderline = This->description.fUnderline;
632 logFont.lfStrikeOut = This->description.fStrikethrough;
633 logFont.lfWeight = This->description.sWeight;
634 logFont.lfOutPrecision = OUT_CHARACTER_PRECIS;
635 logFont.lfClipPrecision = CLIP_DEFAULT_PRECIS;
636 logFont.lfQuality = DEFAULT_QUALITY;
637 logFont.lfPitchAndFamily = DEFAULT_PITCH;
638
639 This->gdiFont = CreateFontIndirectW(&logFont);
640 This->dirty = FALSE;
641
642 add_hfontitem(This->gdiFont);
643
644 /* Fixup the name and charset properties so that they match the
645 selected font */
646 old_font = SelectObject(get_dc(), This->gdiFont);
647 GetTextFaceW(hdc, ARRAY_SIZE(text_face), text_face);
648 if(lstrcmpiW(text_face, This->description.lpstrName))
649 {
650 HeapFree(GetProcessHeap(), 0, This->description.lpstrName);
651 This->description.lpstrName = strdupW(text_face);
652 }
653 GetTextMetricsW(hdc, &tm);
654 This->description.sCharset = tm.tmCharSet;
655 /* While we have it handy, stash the realized font height for use by get_Size() */
656 This->nRealHeight = tm.tmHeight - tm.tmInternalLeading; /* corresponds to LOGFONT lfHeight */
657 SelectObject(hdc, old_font);
658 }
659
660 /************************************************************************
661 * OLEFontImpl_get_Name (IFont)
662 *
663 * See Windows documentation for more details on IFont methods.
664 */
OLEFontImpl_get_Name(IFont * iface,BSTR * pname)665 static HRESULT WINAPI OLEFontImpl_get_Name(
666 IFont* iface,
667 BSTR* pname)
668 {
669 OLEFontImpl *this = impl_from_IFont(iface);
670 TRACE("(%p)->(%p)\n", this, pname);
671
672 if (pname==0)
673 return E_POINTER;
674
675 realize_font(this);
676
677 if (this->description.lpstrName!=0)
678 *pname = SysAllocString(this->description.lpstrName);
679 else
680 *pname = 0;
681
682 return S_OK;
683 }
684
685 /************************************************************************
686 * OLEFontImpl_put_Name (IFont)
687 */
OLEFontImpl_put_Name(IFont * iface,BSTR name)688 static HRESULT WINAPI OLEFontImpl_put_Name(
689 IFont* iface,
690 BSTR name)
691 {
692 OLEFontImpl *This = impl_from_IFont(iface);
693 TRACE("(%p)->(%p)\n", This, name);
694
695 if (!name)
696 return CTL_E_INVALIDPROPERTYVALUE;
697
698 HeapFree(GetProcessHeap(), 0, This->description.lpstrName);
699 This->description.lpstrName = strdupW(name);
700 if (!This->description.lpstrName) return E_OUTOFMEMORY;
701
702 TRACE("new name %s\n", debugstr_w(This->description.lpstrName));
703 OLEFont_SendNotify(This, DISPID_FONT_NAME);
704 return S_OK;
705 }
706
707 /************************************************************************
708 * OLEFontImpl_get_Size (IFont)
709 */
OLEFontImpl_get_Size(IFont * iface,CY * psize)710 static HRESULT WINAPI OLEFontImpl_get_Size(
711 IFont* iface,
712 CY* psize)
713 {
714 OLEFontImpl *this = impl_from_IFont(iface);
715 TRACE("(%p)->(%p)\n", this, psize);
716
717 if (!psize) return E_POINTER;
718
719 realize_font(this);
720
721 /*
722 * Convert realized font height in pixels to points descaled by current
723 * scaling ratio then scaled up by 10000.
724 */
725 psize->s.Lo = MulDiv(this->nRealHeight,
726 this->cyHimetric * 72 * 10000,
727 this->cyLogical * 2540);
728 psize->s.Hi = 0;
729
730 return S_OK;
731 }
732
733 /************************************************************************
734 * OLEFontImpl_put_Size (IFont)
735 */
OLEFontImpl_put_Size(IFont * iface,CY size)736 static HRESULT WINAPI OLEFontImpl_put_Size(
737 IFont* iface,
738 CY size)
739 {
740 OLEFontImpl *this = impl_from_IFont(iface);
741 TRACE("(%p)->(%d)\n", this, size.s.Lo);
742 this->description.cySize.s.Hi = 0;
743 this->description.cySize.s.Lo = size.s.Lo;
744 OLEFont_SendNotify(this, DISPID_FONT_SIZE);
745
746 return S_OK;
747 }
748
749 /************************************************************************
750 * OLEFontImpl_get_Bold (IFont)
751 *
752 * See Windows documentation for more details on IFont methods.
753 */
OLEFontImpl_get_Bold(IFont * iface,BOOL * pbold)754 static HRESULT WINAPI OLEFontImpl_get_Bold(
755 IFont* iface,
756 BOOL* pbold)
757 {
758 OLEFontImpl *this = impl_from_IFont(iface);
759 TRACE("(%p)->(%p)\n", this, pbold);
760
761 if (!pbold) return E_POINTER;
762
763 realize_font(this);
764
765 *pbold = this->description.sWeight > 550;
766
767 return S_OK;
768 }
769
770 /************************************************************************
771 * OLEFontImpl_put_Bold (IFont)
772 */
OLEFontImpl_put_Bold(IFont * iface,BOOL bold)773 static HRESULT WINAPI OLEFontImpl_put_Bold(
774 IFont* iface,
775 BOOL bold)
776 {
777 OLEFontImpl *this = impl_from_IFont(iface);
778 TRACE("(%p)->(%d)\n", this, bold);
779 this->description.sWeight = bold ? FW_BOLD : FW_NORMAL;
780 OLEFont_SendNotify(this, DISPID_FONT_BOLD);
781
782 return S_OK;
783 }
784
785 /************************************************************************
786 * OLEFontImpl_get_Italic (IFont)
787 */
OLEFontImpl_get_Italic(IFont * iface,BOOL * pitalic)788 static HRESULT WINAPI OLEFontImpl_get_Italic(
789 IFont* iface,
790 BOOL* pitalic)
791 {
792 OLEFontImpl *this = impl_from_IFont(iface);
793 TRACE("(%p)->(%p)\n", this, pitalic);
794
795 if (pitalic==0)
796 return E_POINTER;
797
798 realize_font(this);
799
800 *pitalic = this->description.fItalic;
801
802 return S_OK;
803 }
804
805 /************************************************************************
806 * OLEFontImpl_put_Italic (IFont)
807 */
OLEFontImpl_put_Italic(IFont * iface,BOOL italic)808 static HRESULT WINAPI OLEFontImpl_put_Italic(
809 IFont* iface,
810 BOOL italic)
811 {
812 OLEFontImpl *this = impl_from_IFont(iface);
813 TRACE("(%p)->(%d)\n", this, italic);
814
815 this->description.fItalic = italic;
816
817 OLEFont_SendNotify(this, DISPID_FONT_ITALIC);
818 return S_OK;
819 }
820
821 /************************************************************************
822 * OLEFontImpl_get_Underline (IFont)
823 */
OLEFontImpl_get_Underline(IFont * iface,BOOL * punderline)824 static HRESULT WINAPI OLEFontImpl_get_Underline(
825 IFont* iface,
826 BOOL* punderline)
827 {
828 OLEFontImpl *this = impl_from_IFont(iface);
829 TRACE("(%p)->(%p)\n", this, punderline);
830
831 if (punderline==0)
832 return E_POINTER;
833
834 realize_font(this);
835
836 *punderline = this->description.fUnderline;
837
838 return S_OK;
839 }
840
841 /************************************************************************
842 * OLEFontImpl_put_Underline (IFont)
843 */
OLEFontImpl_put_Underline(IFont * iface,BOOL underline)844 static HRESULT WINAPI OLEFontImpl_put_Underline(
845 IFont* iface,
846 BOOL underline)
847 {
848 OLEFontImpl *this = impl_from_IFont(iface);
849 TRACE("(%p)->(%d)\n", this, underline);
850
851 this->description.fUnderline = underline;
852
853 OLEFont_SendNotify(this, DISPID_FONT_UNDER);
854 return S_OK;
855 }
856
857 /************************************************************************
858 * OLEFontImpl_get_Strikethrough (IFont)
859 */
OLEFontImpl_get_Strikethrough(IFont * iface,BOOL * pstrikethrough)860 static HRESULT WINAPI OLEFontImpl_get_Strikethrough(
861 IFont* iface,
862 BOOL* pstrikethrough)
863 {
864 OLEFontImpl *this = impl_from_IFont(iface);
865 TRACE("(%p)->(%p)\n", this, pstrikethrough);
866
867 if (pstrikethrough==0)
868 return E_POINTER;
869
870 realize_font(this);
871
872 *pstrikethrough = this->description.fStrikethrough;
873
874 return S_OK;
875 }
876
877 /************************************************************************
878 * OLEFontImpl_put_Strikethrough (IFont)
879 */
OLEFontImpl_put_Strikethrough(IFont * iface,BOOL strikethrough)880 static HRESULT WINAPI OLEFontImpl_put_Strikethrough(
881 IFont* iface,
882 BOOL strikethrough)
883 {
884 OLEFontImpl *this = impl_from_IFont(iface);
885 TRACE("(%p)->(%d)\n", this, strikethrough);
886
887 this->description.fStrikethrough = strikethrough;
888 OLEFont_SendNotify(this, DISPID_FONT_STRIKE);
889
890 return S_OK;
891 }
892
893 /************************************************************************
894 * OLEFontImpl_get_Weight (IFont)
895 */
OLEFontImpl_get_Weight(IFont * iface,short * pweight)896 static HRESULT WINAPI OLEFontImpl_get_Weight(
897 IFont* iface,
898 short* pweight)
899 {
900 OLEFontImpl *this = impl_from_IFont(iface);
901 TRACE("(%p)->(%p)\n", this, pweight);
902
903 if (pweight==0)
904 return E_POINTER;
905
906 realize_font(this);
907
908 *pweight = this->description.sWeight;
909
910 return S_OK;
911 }
912
913 /************************************************************************
914 * OLEFontImpl_put_Weight (IFont)
915 */
OLEFontImpl_put_Weight(IFont * iface,short weight)916 static HRESULT WINAPI OLEFontImpl_put_Weight(
917 IFont* iface,
918 short weight)
919 {
920 OLEFontImpl *this = impl_from_IFont(iface);
921 TRACE("(%p)->(%d)\n", this, weight);
922
923 this->description.sWeight = weight;
924
925 OLEFont_SendNotify(this, DISPID_FONT_WEIGHT);
926 return S_OK;
927 }
928
929 /************************************************************************
930 * OLEFontImpl_get_Charset (IFont)
931 */
OLEFontImpl_get_Charset(IFont * iface,short * pcharset)932 static HRESULT WINAPI OLEFontImpl_get_Charset(
933 IFont* iface,
934 short* pcharset)
935 {
936 OLEFontImpl *this = impl_from_IFont(iface);
937 TRACE("(%p)->(%p)\n", this, pcharset);
938
939 if (pcharset==0)
940 return E_POINTER;
941
942 realize_font(this);
943
944 *pcharset = this->description.sCharset;
945
946 return S_OK;
947 }
948
949 /************************************************************************
950 * OLEFontImpl_put_Charset (IFont)
951 */
OLEFontImpl_put_Charset(IFont * iface,short charset)952 static HRESULT WINAPI OLEFontImpl_put_Charset(
953 IFont* iface,
954 short charset)
955 {
956 OLEFontImpl *this = impl_from_IFont(iface);
957 TRACE("(%p)->(%d)\n", this, charset);
958
959 this->description.sCharset = charset;
960 OLEFont_SendNotify(this, DISPID_FONT_CHARSET);
961
962 return S_OK;
963 }
964
965 /************************************************************************
966 * OLEFontImpl_get_hFont (IFont)
967 */
OLEFontImpl_get_hFont(IFont * iface,HFONT * phfont)968 static HRESULT WINAPI OLEFontImpl_get_hFont(
969 IFont* iface,
970 HFONT* phfont)
971 {
972 OLEFontImpl *this = impl_from_IFont(iface);
973 TRACE("(%p)->(%p)\n", this, phfont);
974 if (phfont==NULL)
975 return E_POINTER;
976
977 realize_font(this);
978
979 *phfont = this->gdiFont;
980 TRACE("Returning %p\n", *phfont);
981 return S_OK;
982 }
983
984 /************************************************************************
985 * OLEFontImpl_Clone (IFont)
986 */
OLEFontImpl_Clone(IFont * iface,IFont ** ppfont)987 static HRESULT WINAPI OLEFontImpl_Clone(
988 IFont* iface,
989 IFont** ppfont)
990 {
991 OLEFontImpl *this = impl_from_IFont(iface);
992 OLEFontImpl* newObject;
993
994 TRACE("(%p)->(%p)\n", this, ppfont);
995
996 if (ppfont == NULL)
997 return E_POINTER;
998
999 *ppfont = NULL;
1000
1001 newObject = HeapAlloc(GetProcessHeap(), 0, sizeof(OLEFontImpl));
1002 if (newObject==NULL)
1003 return E_OUTOFMEMORY;
1004
1005 *newObject = *this;
1006 /* allocate separate buffer */
1007 newObject->description.lpstrName = strdupW(this->description.lpstrName);
1008
1009 /* Increment internal ref in hfont item list */
1010 if(newObject->gdiFont) inc_int_ref(newObject->gdiFont);
1011
1012 InterlockedIncrement(&ifont_cnt);
1013
1014 newObject->pPropertyNotifyCP = NULL;
1015 newObject->pFontEventsCP = NULL;
1016 CreateConnectionPoint((IUnknown*)&newObject->IFont_iface, &IID_IPropertyNotifySink,
1017 &newObject->pPropertyNotifyCP);
1018 CreateConnectionPoint((IUnknown*)&newObject->IFont_iface, &IID_IFontEventsDisp,
1019 &newObject->pFontEventsCP);
1020
1021 if (!newObject->pPropertyNotifyCP || !newObject->pFontEventsCP)
1022 {
1023 OLEFontImpl_Destroy(newObject);
1024 return E_OUTOFMEMORY;
1025 }
1026
1027 /* The cloned object starts with a reference count of 1 */
1028 newObject->ref = 1;
1029
1030 *ppfont = &newObject->IFont_iface;
1031
1032 return S_OK;
1033 }
1034
1035 /************************************************************************
1036 * OLEFontImpl_IsEqual (IFont)
1037 */
OLEFontImpl_IsEqual(IFont * iface,IFont * pFontOther)1038 static HRESULT WINAPI OLEFontImpl_IsEqual(
1039 IFont* iface,
1040 IFont* pFontOther)
1041 {
1042 OLEFontImpl *left = impl_from_IFont(iface);
1043 OLEFontImpl *right = impl_from_IFont(pFontOther);
1044 INT ret;
1045 INT left_len,right_len;
1046
1047 if(pFontOther == NULL)
1048 return E_POINTER;
1049 else if (left->description.cySize.s.Lo != right->description.cySize.s.Lo)
1050 return S_FALSE;
1051 else if (left->description.cySize.s.Hi != right->description.cySize.s.Hi)
1052 return S_FALSE;
1053 else if (left->description.sWeight != right->description.sWeight)
1054 return S_FALSE;
1055 else if (left->description.sCharset != right->description.sCharset)
1056 return S_FALSE;
1057 else if (left->description.fItalic != right->description.fItalic)
1058 return S_FALSE;
1059 else if (left->description.fUnderline != right->description.fUnderline)
1060 return S_FALSE;
1061 else if (left->description.fStrikethrough != right->description.fStrikethrough)
1062 return S_FALSE;
1063
1064 /* Check from string */
1065 left_len = lstrlenW(left->description.lpstrName);
1066 right_len = lstrlenW(right->description.lpstrName);
1067 ret = CompareStringW(0,0,left->description.lpstrName, left_len,
1068 right->description.lpstrName, right_len);
1069 if (ret != CSTR_EQUAL)
1070 return S_FALSE;
1071
1072 return S_OK;
1073 }
1074
1075 /************************************************************************
1076 * OLEFontImpl_SetRatio (IFont)
1077 */
OLEFontImpl_SetRatio(IFont * iface,LONG cyLogical,LONG cyHimetric)1078 static HRESULT WINAPI OLEFontImpl_SetRatio(
1079 IFont* iface,
1080 LONG cyLogical,
1081 LONG cyHimetric)
1082 {
1083 OLEFontImpl *this = impl_from_IFont(iface);
1084 TRACE("(%p)->(%d, %d)\n", this, cyLogical, cyHimetric);
1085
1086 if(cyLogical == 0 || cyHimetric == 0)
1087 return E_FAIL;
1088
1089 /* cyLogical and cyHimetric both set to 1 is a special case that
1090 does not change the scaling but also does not fail */
1091 if(cyLogical == 1 && cyHimetric == 1)
1092 return S_OK;
1093
1094 this->cyLogical = cyLogical;
1095 this->cyHimetric = cyHimetric;
1096 this->dirty = TRUE;
1097
1098 return S_OK;
1099 }
1100
1101 /************************************************************************
1102 * OLEFontImpl_QueryTextMetrics (IFont)
1103 */
OLEFontImpl_QueryTextMetrics(IFont * iface,TEXTMETRICOLE * ptm)1104 static HRESULT WINAPI OLEFontImpl_QueryTextMetrics(
1105 IFont* iface,
1106 TEXTMETRICOLE* ptm)
1107 {
1108 HDC hdcRef;
1109 HFONT hOldFont, hNewFont;
1110
1111 hdcRef = GetDC(0);
1112 IFont_get_hFont(iface, &hNewFont);
1113 hOldFont = SelectObject(hdcRef, hNewFont);
1114 GetTextMetricsW(hdcRef, ptm);
1115 SelectObject(hdcRef, hOldFont);
1116 ReleaseDC(0, hdcRef);
1117 return S_OK;
1118 }
1119
1120 /************************************************************************
1121 * OLEFontImpl_AddRefHfont (IFont)
1122 */
OLEFontImpl_AddRefHfont(IFont * iface,HFONT hfont)1123 static HRESULT WINAPI OLEFontImpl_AddRefHfont(
1124 IFont* iface,
1125 HFONT hfont)
1126 {
1127 OLEFontImpl *this = impl_from_IFont(iface);
1128
1129 TRACE("(%p)->(%p)\n", this, hfont);
1130
1131 if (!hfont) return E_INVALIDARG;
1132
1133 return inc_ext_ref(hfont);
1134 }
1135
1136 /************************************************************************
1137 * OLEFontImpl_ReleaseHfont (IFont)
1138 */
OLEFontImpl_ReleaseHfont(IFont * iface,HFONT hfont)1139 static HRESULT WINAPI OLEFontImpl_ReleaseHfont(
1140 IFont* iface,
1141 HFONT hfont)
1142 {
1143 OLEFontImpl *this = impl_from_IFont(iface);
1144
1145 TRACE("(%p)->(%p)\n", this, hfont);
1146
1147 if (!hfont) return E_INVALIDARG;
1148
1149 return dec_ext_ref(hfont);
1150 }
1151
1152 /************************************************************************
1153 * OLEFontImpl_SetHdc (IFont)
1154 */
OLEFontImpl_SetHdc(IFont * iface,HDC hdc)1155 static HRESULT WINAPI OLEFontImpl_SetHdc(
1156 IFont* iface,
1157 HDC hdc)
1158 {
1159 OLEFontImpl *this = impl_from_IFont(iface);
1160 FIXME("(%p)->(%p): Stub\n", this, hdc);
1161 return E_NOTIMPL;
1162 }
1163
1164 static const IFontVtbl OLEFontImpl_VTable =
1165 {
1166 OLEFontImpl_QueryInterface,
1167 OLEFontImpl_AddRef,
1168 OLEFontImpl_Release,
1169 OLEFontImpl_get_Name,
1170 OLEFontImpl_put_Name,
1171 OLEFontImpl_get_Size,
1172 OLEFontImpl_put_Size,
1173 OLEFontImpl_get_Bold,
1174 OLEFontImpl_put_Bold,
1175 OLEFontImpl_get_Italic,
1176 OLEFontImpl_put_Italic,
1177 OLEFontImpl_get_Underline,
1178 OLEFontImpl_put_Underline,
1179 OLEFontImpl_get_Strikethrough,
1180 OLEFontImpl_put_Strikethrough,
1181 OLEFontImpl_get_Weight,
1182 OLEFontImpl_put_Weight,
1183 OLEFontImpl_get_Charset,
1184 OLEFontImpl_put_Charset,
1185 OLEFontImpl_get_hFont,
1186 OLEFontImpl_Clone,
1187 OLEFontImpl_IsEqual,
1188 OLEFontImpl_SetRatio,
1189 OLEFontImpl_QueryTextMetrics,
1190 OLEFontImpl_AddRefHfont,
1191 OLEFontImpl_ReleaseHfont,
1192 OLEFontImpl_SetHdc
1193 };
1194
1195 /************************************************************************
1196 * OLEFontImpl_IDispatch_QueryInterface (IUnknown)
1197 */
OLEFontImpl_IDispatch_QueryInterface(IDispatch * iface,REFIID riid,VOID ** ppvoid)1198 static HRESULT WINAPI OLEFontImpl_IDispatch_QueryInterface(
1199 IDispatch* iface,
1200 REFIID riid,
1201 VOID** ppvoid)
1202 {
1203 OLEFontImpl *this = impl_from_IDispatch(iface);
1204 return IFont_QueryInterface(&this->IFont_iface, riid, ppvoid);
1205 }
1206
1207 /************************************************************************
1208 * OLEFontImpl_IDispatch_Release (IUnknown)
1209 */
OLEFontImpl_IDispatch_Release(IDispatch * iface)1210 static ULONG WINAPI OLEFontImpl_IDispatch_Release(
1211 IDispatch* iface)
1212 {
1213 OLEFontImpl *this = impl_from_IDispatch(iface);
1214 return IFont_Release(&this->IFont_iface);
1215 }
1216
1217 /************************************************************************
1218 * OLEFontImpl_IDispatch_AddRef (IUnknown)
1219 */
OLEFontImpl_IDispatch_AddRef(IDispatch * iface)1220 static ULONG WINAPI OLEFontImpl_IDispatch_AddRef(
1221 IDispatch* iface)
1222 {
1223 OLEFontImpl *this = impl_from_IDispatch(iface);
1224 return IFont_AddRef(&this->IFont_iface);
1225 }
1226
1227 /************************************************************************
1228 * OLEFontImpl_GetTypeInfoCount (IDispatch)
1229 */
OLEFontImpl_GetTypeInfoCount(IDispatch * iface,unsigned int * pctinfo)1230 static HRESULT WINAPI OLEFontImpl_GetTypeInfoCount(
1231 IDispatch* iface,
1232 unsigned int* pctinfo)
1233 {
1234 OLEFontImpl *this = impl_from_IDispatch(iface);
1235 TRACE("(%p)->(%p)\n", this, pctinfo);
1236 *pctinfo = 1;
1237
1238 return S_OK;
1239 }
1240
1241 /************************************************************************
1242 * OLEFontImpl_GetTypeInfo (IDispatch)
1243 */
OLEFontImpl_GetTypeInfo(IDispatch * iface,UINT iTInfo,LCID lcid,ITypeInfo ** ppTInfo)1244 static HRESULT WINAPI OLEFontImpl_GetTypeInfo(
1245 IDispatch* iface,
1246 UINT iTInfo,
1247 LCID lcid,
1248 ITypeInfo** ppTInfo)
1249 {
1250 static const WCHAR stdole2tlb[] = {'s','t','d','o','l','e','2','.','t','l','b',0};
1251 ITypeLib *tl;
1252 HRESULT hres;
1253
1254 OLEFontImpl *this = impl_from_IDispatch(iface);
1255 TRACE("(%p, iTInfo=%d, lcid=%04x, %p)\n", this, iTInfo, (int)lcid, ppTInfo);
1256 if (iTInfo != 0)
1257 return E_FAIL;
1258 hres = LoadTypeLib(stdole2tlb, &tl);
1259 if (FAILED(hres)) {
1260 ERR("Could not load the stdole2.tlb?\n");
1261 return hres;
1262 }
1263 hres = ITypeLib_GetTypeInfoOfGuid(tl, &IID_IFontDisp, ppTInfo);
1264 ITypeLib_Release(tl);
1265 if (FAILED(hres)) {
1266 FIXME("Did not IDispatch typeinfo from typelib, hres %x\n",hres);
1267 }
1268 return hres;
1269 }
1270
1271 /************************************************************************
1272 * OLEFontImpl_GetIDsOfNames (IDispatch)
1273 */
OLEFontImpl_GetIDsOfNames(IDispatch * iface,REFIID riid,LPOLESTR * rgszNames,UINT cNames,LCID lcid,DISPID * rgDispId)1274 static HRESULT WINAPI OLEFontImpl_GetIDsOfNames(
1275 IDispatch* iface,
1276 REFIID riid,
1277 LPOLESTR* rgszNames,
1278 UINT cNames,
1279 LCID lcid,
1280 DISPID* rgDispId)
1281 {
1282 ITypeInfo * pTInfo;
1283 HRESULT hres;
1284
1285 OLEFontImpl *this = impl_from_IDispatch(iface);
1286
1287 TRACE("(%p,%s,%p,cNames=%d,lcid=%04x,%p)\n", this, debugstr_guid(riid),
1288 rgszNames, cNames, (int)lcid, rgDispId);
1289
1290 if (cNames == 0) return E_INVALIDARG;
1291
1292 hres = IDispatch_GetTypeInfo(iface, 0, lcid, &pTInfo);
1293 if (FAILED(hres))
1294 {
1295 ERR("GetTypeInfo failed.\n");
1296 return hres;
1297 }
1298
1299 /* convert names to DISPIDs */
1300 hres = DispGetIDsOfNames (pTInfo, rgszNames, cNames, rgDispId);
1301 ITypeInfo_Release(pTInfo);
1302
1303 return hres;
1304 }
1305
1306 /************************************************************************
1307 * OLEFontImpl_Invoke (IDispatch)
1308 *
1309 */
OLEFontImpl_Invoke(IDispatch * iface,DISPID dispIdMember,REFIID riid,LCID lcid,WORD wFlags,DISPPARAMS * pDispParams,VARIANT * pVarResult,EXCEPINFO * pExepInfo,UINT * puArgErr)1310 static HRESULT WINAPI OLEFontImpl_Invoke(
1311 IDispatch* iface,
1312 DISPID dispIdMember,
1313 REFIID riid,
1314 LCID lcid,
1315 WORD wFlags,
1316 DISPPARAMS* pDispParams,
1317 VARIANT* pVarResult,
1318 EXCEPINFO* pExepInfo,
1319 UINT* puArgErr)
1320 {
1321 OLEFontImpl *this = impl_from_IDispatch(iface);
1322 HRESULT hr;
1323
1324 TRACE("%p->(%d,%s,0x%x,0x%x,%p,%p,%p,%p)\n", this, dispIdMember,
1325 debugstr_guid(riid), lcid, wFlags, pDispParams, pVarResult, pExepInfo,
1326 puArgErr);
1327
1328 /* validate parameters */
1329
1330 if (!IsEqualIID(riid, &IID_NULL))
1331 {
1332 ERR("riid was %s instead of IID_NULL\n", debugstr_guid(riid));
1333 return DISP_E_UNKNOWNINTERFACE;
1334 }
1335
1336 if (wFlags & DISPATCH_PROPERTYGET)
1337 {
1338 if (!pVarResult)
1339 {
1340 ERR("null pVarResult not allowed when DISPATCH_PROPERTYGET specified\n");
1341 return DISP_E_PARAMNOTOPTIONAL;
1342 }
1343 }
1344 else if (wFlags & DISPATCH_PROPERTYPUT)
1345 {
1346 if (!pDispParams)
1347 {
1348 ERR("null pDispParams not allowed when DISPATCH_PROPERTYPUT specified\n");
1349 return DISP_E_PARAMNOTOPTIONAL;
1350 }
1351 if (pDispParams->cArgs != 1)
1352 {
1353 ERR("param count for DISPATCH_PROPERTYPUT was %d instead of 1\n", pDispParams->cArgs);
1354 return DISP_E_BADPARAMCOUNT;
1355 }
1356 }
1357 else
1358 {
1359 ERR("one of DISPATCH_PROPERTYGET or DISPATCH_PROPERTYPUT must be specified\n");
1360 return DISP_E_MEMBERNOTFOUND;
1361 }
1362
1363 switch (dispIdMember) {
1364 case DISPID_FONT_NAME:
1365 if (wFlags & DISPATCH_PROPERTYGET) {
1366 V_VT(pVarResult) = VT_BSTR;
1367 return IFont_get_Name(&this->IFont_iface, &V_BSTR(pVarResult));
1368 } else {
1369 VARIANTARG vararg;
1370
1371 VariantInit(&vararg);
1372 hr = VariantChangeTypeEx(&vararg, &pDispParams->rgvarg[0], lcid, 0, VT_BSTR);
1373 if (FAILED(hr))
1374 return hr;
1375
1376 hr = IFont_put_Name(&this->IFont_iface, V_BSTR(&vararg));
1377
1378 VariantClear(&vararg);
1379 return hr;
1380 }
1381 break;
1382 case DISPID_FONT_BOLD:
1383 if (wFlags & DISPATCH_PROPERTYGET) {
1384 BOOL value;
1385 hr = IFont_get_Bold(&this->IFont_iface, &value);
1386 V_VT(pVarResult) = VT_BOOL;
1387 V_BOOL(pVarResult) = value ? VARIANT_TRUE : VARIANT_FALSE;
1388 return hr;
1389 } else {
1390 VARIANTARG vararg;
1391
1392 VariantInit(&vararg);
1393 hr = VariantChangeTypeEx(&vararg, &pDispParams->rgvarg[0], lcid, 0, VT_BOOL);
1394 if (FAILED(hr))
1395 return hr;
1396
1397 hr = IFont_put_Bold(&this->IFont_iface, V_BOOL(&vararg));
1398
1399 VariantClear(&vararg);
1400 return hr;
1401 }
1402 break;
1403 case DISPID_FONT_ITALIC:
1404 if (wFlags & DISPATCH_PROPERTYGET) {
1405 BOOL value;
1406 hr = IFont_get_Italic(&this->IFont_iface, &value);
1407 V_VT(pVarResult) = VT_BOOL;
1408 V_BOOL(pVarResult) = value ? VARIANT_TRUE : VARIANT_FALSE;
1409 return hr;
1410 } else {
1411 VARIANTARG vararg;
1412
1413 VariantInit(&vararg);
1414 hr = VariantChangeTypeEx(&vararg, &pDispParams->rgvarg[0], lcid, 0, VT_BOOL);
1415 if (FAILED(hr))
1416 return hr;
1417
1418 hr = IFont_put_Italic(&this->IFont_iface, V_BOOL(&vararg));
1419
1420 VariantClear(&vararg);
1421 return hr;
1422 }
1423 break;
1424 case DISPID_FONT_UNDER:
1425 if (wFlags & DISPATCH_PROPERTYGET) {
1426 BOOL value;
1427 hr = IFont_get_Underline(&this->IFont_iface, &value);
1428 V_VT(pVarResult) = VT_BOOL;
1429 V_BOOL(pVarResult) = value ? VARIANT_TRUE : VARIANT_FALSE;
1430 return hr;
1431 } else {
1432 VARIANTARG vararg;
1433
1434 VariantInit(&vararg);
1435 hr = VariantChangeTypeEx(&vararg, &pDispParams->rgvarg[0], lcid, 0, VT_BOOL);
1436 if (FAILED(hr))
1437 return hr;
1438
1439 hr = IFont_put_Underline(&this->IFont_iface, V_BOOL(&vararg));
1440
1441 VariantClear(&vararg);
1442 return hr;
1443 }
1444 break;
1445 case DISPID_FONT_STRIKE:
1446 if (wFlags & DISPATCH_PROPERTYGET) {
1447 BOOL value;
1448 hr = IFont_get_Strikethrough(&this->IFont_iface, &value);
1449 V_VT(pVarResult) = VT_BOOL;
1450 V_BOOL(pVarResult) = value ? VARIANT_TRUE : VARIANT_FALSE;
1451 return hr;
1452 } else {
1453 VARIANTARG vararg;
1454
1455 VariantInit(&vararg);
1456 hr = VariantChangeTypeEx(&vararg, &pDispParams->rgvarg[0], lcid, 0, VT_BOOL);
1457 if (FAILED(hr))
1458 return hr;
1459
1460 hr = IFont_put_Strikethrough(&this->IFont_iface, V_BOOL(&vararg));
1461
1462 VariantClear(&vararg);
1463 return hr;
1464 }
1465 break;
1466 case DISPID_FONT_SIZE:
1467 if (wFlags & DISPATCH_PROPERTYGET) {
1468 V_VT(pVarResult) = VT_CY;
1469 return IFont_get_Size(&this->IFont_iface, &V_CY(pVarResult));
1470 } else {
1471 VARIANTARG vararg;
1472
1473 VariantInit(&vararg);
1474 hr = VariantChangeTypeEx(&vararg, &pDispParams->rgvarg[0], lcid, 0, VT_CY);
1475 if (FAILED(hr))
1476 return hr;
1477
1478 hr = IFont_put_Size(&this->IFont_iface, V_CY(&vararg));
1479
1480 VariantClear(&vararg);
1481 return hr;
1482 }
1483 break;
1484 case DISPID_FONT_WEIGHT:
1485 if (wFlags & DISPATCH_PROPERTYGET) {
1486 V_VT(pVarResult) = VT_I2;
1487 return IFont_get_Weight(&this->IFont_iface, &V_I2(pVarResult));
1488 } else {
1489 VARIANTARG vararg;
1490
1491 VariantInit(&vararg);
1492 hr = VariantChangeTypeEx(&vararg, &pDispParams->rgvarg[0], lcid, 0, VT_I2);
1493 if (FAILED(hr))
1494 return hr;
1495
1496 hr = IFont_put_Weight(&this->IFont_iface, V_I2(&vararg));
1497
1498 VariantClear(&vararg);
1499 return hr;
1500 }
1501 break;
1502 case DISPID_FONT_CHARSET:
1503 if (wFlags & DISPATCH_PROPERTYGET) {
1504 V_VT(pVarResult) = VT_I2;
1505 return OLEFontImpl_get_Charset(&this->IFont_iface, &V_I2(pVarResult));
1506 } else {
1507 VARIANTARG vararg;
1508
1509 VariantInit(&vararg);
1510 hr = VariantChangeTypeEx(&vararg, &pDispParams->rgvarg[0], lcid, 0, VT_I2);
1511 if (FAILED(hr))
1512 return hr;
1513
1514 hr = IFont_put_Charset(&this->IFont_iface, V_I2(&vararg));
1515
1516 VariantClear(&vararg);
1517 return hr;
1518 }
1519 break;
1520 default:
1521 ERR("member not found for dispid 0x%x\n", dispIdMember);
1522 return DISP_E_MEMBERNOTFOUND;
1523 }
1524 }
1525
1526 static const IDispatchVtbl OLEFontImpl_IDispatch_VTable =
1527 {
1528 OLEFontImpl_IDispatch_QueryInterface,
1529 OLEFontImpl_IDispatch_AddRef,
1530 OLEFontImpl_IDispatch_Release,
1531 OLEFontImpl_GetTypeInfoCount,
1532 OLEFontImpl_GetTypeInfo,
1533 OLEFontImpl_GetIDsOfNames,
1534 OLEFontImpl_Invoke
1535 };
1536
1537 /************************************************************************
1538 * OLEFontImpl_IPersistStream_QueryInterface (IUnknown)
1539 */
OLEFontImpl_IPersistStream_QueryInterface(IPersistStream * iface,REFIID riid,VOID ** ppvoid)1540 static HRESULT WINAPI OLEFontImpl_IPersistStream_QueryInterface(
1541 IPersistStream* iface,
1542 REFIID riid,
1543 VOID** ppvoid)
1544 {
1545 OLEFontImpl *this = impl_from_IPersistStream(iface);
1546
1547 return IFont_QueryInterface(&this->IFont_iface, riid, ppvoid);
1548 }
1549
1550 /************************************************************************
1551 * OLEFontImpl_IPersistStream_Release (IUnknown)
1552 */
OLEFontImpl_IPersistStream_Release(IPersistStream * iface)1553 static ULONG WINAPI OLEFontImpl_IPersistStream_Release(
1554 IPersistStream* iface)
1555 {
1556 OLEFontImpl *this = impl_from_IPersistStream(iface);
1557
1558 return IFont_Release(&this->IFont_iface);
1559 }
1560
1561 /************************************************************************
1562 * OLEFontImpl_IPersistStream_AddRef (IUnknown)
1563 */
OLEFontImpl_IPersistStream_AddRef(IPersistStream * iface)1564 static ULONG WINAPI OLEFontImpl_IPersistStream_AddRef(
1565 IPersistStream* iface)
1566 {
1567 OLEFontImpl *this = impl_from_IPersistStream(iface);
1568
1569 return IFont_AddRef(&this->IFont_iface);
1570 }
1571
1572 /************************************************************************
1573 * OLEFontImpl_GetClassID (IPersistStream)
1574 */
OLEFontImpl_GetClassID(IPersistStream * iface,CLSID * pClassID)1575 static HRESULT WINAPI OLEFontImpl_GetClassID(
1576 IPersistStream* iface,
1577 CLSID* pClassID)
1578 {
1579 TRACE("(%p,%p)\n",iface,pClassID);
1580 if (pClassID==0)
1581 return E_POINTER;
1582
1583 *pClassID = CLSID_StdFont;
1584
1585 return S_OK;
1586 }
1587
1588 /************************************************************************
1589 * OLEFontImpl_IsDirty (IPersistStream)
1590 *
1591 * See Windows documentation for more details on IPersistStream methods.
1592 */
OLEFontImpl_IsDirty(IPersistStream * iface)1593 static HRESULT WINAPI OLEFontImpl_IsDirty(
1594 IPersistStream* iface)
1595 {
1596 TRACE("(%p)\n",iface);
1597 return S_OK;
1598 }
1599
1600 /************************************************************************
1601 * OLEFontImpl_Load (IPersistStream)
1602 *
1603 * See Windows documentation for more details on IPersistStream methods.
1604 *
1605 * This is the format of the standard font serialization as far as I
1606 * know
1607 *
1608 * Offset Type Value Comment
1609 * 0x0000 Byte Unknown Probably a version number, contains 0x01
1610 * 0x0001 Short Charset Charset value from the FONTDESC structure
1611 * 0x0003 Byte Attributes Flags defined as follows:
1612 * 00000010 - Italic
1613 * 00000100 - Underline
1614 * 00001000 - Strikethrough
1615 * 0x0004 Short Weight Weight value from FONTDESC structure
1616 * 0x0006 DWORD size "Low" portion of the cySize member of the FONTDESC
1617 * structure/
1618 * 0x000A Byte name length Length of the font name string (no null character)
1619 * 0x000B String name Name of the font (ASCII, no nul character)
1620 */
OLEFontImpl_Load(IPersistStream * iface,IStream * pLoadStream)1621 static HRESULT WINAPI OLEFontImpl_Load(
1622 IPersistStream* iface,
1623 IStream* pLoadStream)
1624 {
1625 OLEFontImpl *this = impl_from_IPersistStream(iface);
1626 BYTE version, attributes, string_size;
1627 char readBuffer[0x100];
1628 ULONG cbRead;
1629 INT len;
1630
1631 /* Version */
1632 IStream_Read(pLoadStream, &version, sizeof(BYTE), &cbRead);
1633 if ((cbRead != sizeof(BYTE)) || (version != 0x01)) return E_FAIL;
1634
1635 /* Charset */
1636 IStream_Read(pLoadStream, &this->description.sCharset, sizeof(WORD), &cbRead);
1637 if (cbRead != sizeof(WORD)) return E_FAIL;
1638
1639 /* Attributes */
1640 IStream_Read(pLoadStream, &attributes, sizeof(BYTE), &cbRead);
1641 if (cbRead != sizeof(BYTE)) return E_FAIL;
1642
1643 this->description.fItalic = (attributes & FONTPERSIST_ITALIC) != 0;
1644 this->description.fStrikethrough = (attributes & FONTPERSIST_STRIKETHROUGH) != 0;
1645 this->description.fUnderline = (attributes & FONTPERSIST_UNDERLINE) != 0;
1646
1647 /* Weight */
1648 IStream_Read(pLoadStream, &this->description.sWeight, sizeof(WORD), &cbRead);
1649 if (cbRead != sizeof(WORD)) return E_FAIL;
1650
1651 /* Size */
1652 IStream_Read(pLoadStream, &this->description.cySize.s.Lo, sizeof(DWORD), &cbRead);
1653 if (cbRead != sizeof(DWORD)) return E_FAIL;
1654
1655 this->description.cySize.s.Hi = 0;
1656
1657 /* Name */
1658 IStream_Read(pLoadStream, &string_size, sizeof(BYTE), &cbRead);
1659 if (cbRead != sizeof(BYTE)) return E_FAIL;
1660
1661 IStream_Read(pLoadStream, readBuffer, string_size, &cbRead);
1662 if (cbRead != string_size) return E_FAIL;
1663
1664 HeapFree(GetProcessHeap(), 0, this->description.lpstrName);
1665
1666 len = MultiByteToWideChar( CP_ACP, 0, readBuffer, string_size, NULL, 0 );
1667 this->description.lpstrName = HeapAlloc( GetProcessHeap(), 0, (len+1) * sizeof(WCHAR) );
1668 MultiByteToWideChar( CP_ACP, 0, readBuffer, string_size, this->description.lpstrName, len );
1669 this->description.lpstrName[len] = 0;
1670
1671 /* Ensure use of this font causes a new one to be created */
1672 dec_int_ref(this->gdiFont);
1673 this->gdiFont = 0;
1674 this->dirty = TRUE;
1675
1676 return S_OK;
1677 }
1678
1679 /************************************************************************
1680 * OLEFontImpl_Save (IPersistStream)
1681 */
OLEFontImpl_Save(IPersistStream * iface,IStream * pOutStream,BOOL fClearDirty)1682 static HRESULT WINAPI OLEFontImpl_Save(
1683 IPersistStream* iface,
1684 IStream* pOutStream,
1685 BOOL fClearDirty)
1686 {
1687 OLEFontImpl *this = impl_from_IPersistStream(iface);
1688 BYTE attributes, string_size;
1689 const BYTE version = 0x01;
1690 char* writeBuffer = NULL;
1691 ULONG written;
1692
1693 TRACE("(%p)->(%p %d)\n", this, pOutStream, fClearDirty);
1694
1695 /* Version */
1696 IStream_Write(pOutStream, &version, sizeof(BYTE), &written);
1697 if (written != sizeof(BYTE)) return E_FAIL;
1698
1699 /* Charset */
1700 IStream_Write(pOutStream, &this->description.sCharset, sizeof(WORD), &written);
1701 if (written != sizeof(WORD)) return E_FAIL;
1702
1703 /* Attributes */
1704 attributes = 0;
1705
1706 if (this->description.fItalic)
1707 attributes |= FONTPERSIST_ITALIC;
1708
1709 if (this->description.fStrikethrough)
1710 attributes |= FONTPERSIST_STRIKETHROUGH;
1711
1712 if (this->description.fUnderline)
1713 attributes |= FONTPERSIST_UNDERLINE;
1714
1715 IStream_Write(pOutStream, &attributes, sizeof(BYTE), &written);
1716 if (written != sizeof(BYTE)) return E_FAIL;
1717
1718 /* Weight */
1719 IStream_Write(pOutStream, &this->description.sWeight, sizeof(WORD), &written);
1720 if (written != sizeof(WORD)) return E_FAIL;
1721
1722 /* Size */
1723 IStream_Write(pOutStream, &this->description.cySize.s.Lo, sizeof(DWORD), &written);
1724 if (written != sizeof(DWORD)) return E_FAIL;
1725
1726 /* FontName */
1727 if (this->description.lpstrName)
1728 string_size = WideCharToMultiByte( CP_ACP, 0, this->description.lpstrName,
1729 lstrlenW(this->description.lpstrName), NULL, 0, NULL, NULL );
1730 else
1731 string_size = 0;
1732
1733 IStream_Write(pOutStream, &string_size, sizeof(BYTE), &written);
1734 if (written != sizeof(BYTE)) return E_FAIL;
1735
1736 if (string_size)
1737 {
1738 if (!(writeBuffer = HeapAlloc( GetProcessHeap(), 0, string_size ))) return E_OUTOFMEMORY;
1739 WideCharToMultiByte( CP_ACP, 0, this->description.lpstrName,
1740 lstrlenW(this->description.lpstrName),
1741 writeBuffer, string_size, NULL, NULL );
1742
1743 IStream_Write(pOutStream, writeBuffer, string_size, &written);
1744 HeapFree(GetProcessHeap(), 0, writeBuffer);
1745
1746 if (written != string_size) return E_FAIL;
1747 }
1748
1749 return S_OK;
1750 }
1751
1752 /************************************************************************
1753 * OLEFontImpl_GetSizeMax (IPersistStream)
1754 */
OLEFontImpl_GetSizeMax(IPersistStream * iface,ULARGE_INTEGER * pcbSize)1755 static HRESULT WINAPI OLEFontImpl_GetSizeMax(
1756 IPersistStream* iface,
1757 ULARGE_INTEGER* pcbSize)
1758 {
1759 OLEFontImpl *this = impl_from_IPersistStream(iface);
1760
1761 if (pcbSize==NULL)
1762 return E_POINTER;
1763
1764 pcbSize->u.HighPart = 0;
1765 pcbSize->u.LowPart = 0;
1766
1767 pcbSize->u.LowPart += sizeof(BYTE); /* Version */
1768 pcbSize->u.LowPart += sizeof(WORD); /* Lang code */
1769 pcbSize->u.LowPart += sizeof(BYTE); /* Flags */
1770 pcbSize->u.LowPart += sizeof(WORD); /* Weight */
1771 pcbSize->u.LowPart += sizeof(DWORD); /* Size */
1772 pcbSize->u.LowPart += sizeof(BYTE); /* StrLength */
1773
1774 if (this->description.lpstrName!=0)
1775 pcbSize->u.LowPart += WideCharToMultiByte( CP_ACP, 0, this->description.lpstrName,
1776 lstrlenW(this->description.lpstrName),
1777 NULL, 0, NULL, NULL );
1778
1779 return S_OK;
1780 }
1781
1782 static const IPersistStreamVtbl OLEFontImpl_IPersistStream_VTable =
1783 {
1784 OLEFontImpl_IPersistStream_QueryInterface,
1785 OLEFontImpl_IPersistStream_AddRef,
1786 OLEFontImpl_IPersistStream_Release,
1787 OLEFontImpl_GetClassID,
1788 OLEFontImpl_IsDirty,
1789 OLEFontImpl_Load,
1790 OLEFontImpl_Save,
1791 OLEFontImpl_GetSizeMax
1792 };
1793
1794 /************************************************************************
1795 * OLEFontImpl_IConnectionPointContainer_QueryInterface (IUnknown)
1796 */
OLEFontImpl_IConnectionPointContainer_QueryInterface(IConnectionPointContainer * iface,REFIID riid,VOID ** ppvoid)1797 static HRESULT WINAPI OLEFontImpl_IConnectionPointContainer_QueryInterface(
1798 IConnectionPointContainer* iface,
1799 REFIID riid,
1800 VOID** ppvoid)
1801 {
1802 OLEFontImpl *this = impl_from_IConnectionPointContainer(iface);
1803
1804 return IFont_QueryInterface(&this->IFont_iface, riid, ppvoid);
1805 }
1806
1807 /************************************************************************
1808 * OLEFontImpl_IConnectionPointContainer_Release (IUnknown)
1809 */
OLEFontImpl_IConnectionPointContainer_Release(IConnectionPointContainer * iface)1810 static ULONG WINAPI OLEFontImpl_IConnectionPointContainer_Release(
1811 IConnectionPointContainer* iface)
1812 {
1813 OLEFontImpl *this = impl_from_IConnectionPointContainer(iface);
1814
1815 return IFont_Release(&this->IFont_iface);
1816 }
1817
1818 /************************************************************************
1819 * OLEFontImpl_IConnectionPointContainer_AddRef (IUnknown)
1820 */
OLEFontImpl_IConnectionPointContainer_AddRef(IConnectionPointContainer * iface)1821 static ULONG WINAPI OLEFontImpl_IConnectionPointContainer_AddRef(
1822 IConnectionPointContainer* iface)
1823 {
1824 OLEFontImpl *this = impl_from_IConnectionPointContainer(iface);
1825
1826 return IFont_AddRef(&this->IFont_iface);
1827 }
1828
1829 /************************************************************************
1830 * OLEFontImpl_EnumConnectionPoints (IConnectionPointContainer)
1831 */
OLEFontImpl_EnumConnectionPoints(IConnectionPointContainer * iface,IEnumConnectionPoints ** ppEnum)1832 static HRESULT WINAPI OLEFontImpl_EnumConnectionPoints(
1833 IConnectionPointContainer* iface,
1834 IEnumConnectionPoints **ppEnum)
1835 {
1836 OLEFontImpl *this = impl_from_IConnectionPointContainer(iface);
1837
1838 FIXME("(%p)->(%p): stub\n", this, ppEnum);
1839 return E_NOTIMPL;
1840 }
1841
1842 /************************************************************************
1843 * OLEFontImpl_FindConnectionPoint (IConnectionPointContainer)
1844 */
OLEFontImpl_FindConnectionPoint(IConnectionPointContainer * iface,REFIID riid,IConnectionPoint ** ppCp)1845 static HRESULT WINAPI OLEFontImpl_FindConnectionPoint(
1846 IConnectionPointContainer* iface,
1847 REFIID riid,
1848 IConnectionPoint **ppCp)
1849 {
1850 OLEFontImpl *this = impl_from_IConnectionPointContainer(iface);
1851 TRACE("(%p)->(%s, %p)\n", this, debugstr_guid(riid), ppCp);
1852
1853 if(IsEqualIID(riid, &IID_IPropertyNotifySink)) {
1854 return IConnectionPoint_QueryInterface(this->pPropertyNotifyCP, &IID_IConnectionPoint,
1855 (void**)ppCp);
1856 } else if(IsEqualIID(riid, &IID_IFontEventsDisp)) {
1857 return IConnectionPoint_QueryInterface(this->pFontEventsCP, &IID_IConnectionPoint,
1858 (void**)ppCp);
1859 } else {
1860 FIXME("no connection point for %s\n", debugstr_guid(riid));
1861 return CONNECT_E_NOCONNECTION;
1862 }
1863 }
1864
1865 static const IConnectionPointContainerVtbl
1866 OLEFontImpl_IConnectionPointContainer_VTable =
1867 {
1868 OLEFontImpl_IConnectionPointContainer_QueryInterface,
1869 OLEFontImpl_IConnectionPointContainer_AddRef,
1870 OLEFontImpl_IConnectionPointContainer_Release,
1871 OLEFontImpl_EnumConnectionPoints,
1872 OLEFontImpl_FindConnectionPoint
1873 };
1874
1875 /************************************************************************
1876 * OLEFontImpl implementation of IPersistPropertyBag.
1877 */
OLEFontImpl_IPersistPropertyBag_QueryInterface(IPersistPropertyBag * iface,REFIID riid,LPVOID * ppvObj)1878 static HRESULT WINAPI OLEFontImpl_IPersistPropertyBag_QueryInterface(
1879 IPersistPropertyBag *iface, REFIID riid, LPVOID *ppvObj
1880 ) {
1881 OLEFontImpl *this = impl_from_IPersistPropertyBag(iface);
1882 return IFont_QueryInterface(&this->IFont_iface,riid,ppvObj);
1883 }
1884
OLEFontImpl_IPersistPropertyBag_AddRef(IPersistPropertyBag * iface)1885 static ULONG WINAPI OLEFontImpl_IPersistPropertyBag_AddRef(
1886 IPersistPropertyBag *iface
1887 ) {
1888 OLEFontImpl *this = impl_from_IPersistPropertyBag(iface);
1889 return IFont_AddRef(&this->IFont_iface);
1890 }
1891
OLEFontImpl_IPersistPropertyBag_Release(IPersistPropertyBag * iface)1892 static ULONG WINAPI OLEFontImpl_IPersistPropertyBag_Release(
1893 IPersistPropertyBag *iface
1894 ) {
1895 OLEFontImpl *this = impl_from_IPersistPropertyBag(iface);
1896 return IFont_Release(&this->IFont_iface);
1897 }
1898
OLEFontImpl_IPersistPropertyBag_GetClassID(IPersistPropertyBag * iface,CLSID * classid)1899 static HRESULT WINAPI OLEFontImpl_IPersistPropertyBag_GetClassID(
1900 IPersistPropertyBag *iface, CLSID *classid
1901 ) {
1902 FIXME("(%p,%p), stub!\n", iface, classid);
1903 return E_FAIL;
1904 }
1905
OLEFontImpl_IPersistPropertyBag_InitNew(IPersistPropertyBag * iface)1906 static HRESULT WINAPI OLEFontImpl_IPersistPropertyBag_InitNew(
1907 IPersistPropertyBag *iface
1908 ) {
1909 FIXME("(%p), stub!\n", iface);
1910 return S_OK;
1911 }
1912
OLEFontImpl_IPersistPropertyBag_Load(IPersistPropertyBag * iface,IPropertyBag * pPropBag,IErrorLog * pErrorLog)1913 static HRESULT WINAPI OLEFontImpl_IPersistPropertyBag_Load(
1914 IPersistPropertyBag *iface, IPropertyBag* pPropBag, IErrorLog* pErrorLog
1915 ) {
1916 /* (from Visual Basic 6 property bag)
1917 Name = "MS Sans Serif"
1918 Size = 13.8
1919 Charset = 0
1920 Weight = 400
1921 Underline = 0 'False
1922 Italic = 0 'False
1923 Strikethrough = 0 'False
1924 */
1925 static const WCHAR sAttrName[] = {'N','a','m','e',0};
1926 static const WCHAR sAttrSize[] = {'S','i','z','e',0};
1927 static const WCHAR sAttrCharset[] = {'C','h','a','r','s','e','t',0};
1928 static const WCHAR sAttrWeight[] = {'W','e','i','g','h','t',0};
1929 static const WCHAR sAttrUnderline[] = {'U','n','d','e','r','l','i','n','e',0};
1930 static const WCHAR sAttrItalic[] = {'I','t','a','l','i','c',0};
1931 static const WCHAR sAttrStrikethrough[] = {'S','t','r','i','k','e','t','h','r','o','u','g','h',0};
1932 OLEFontImpl *this = impl_from_IPersistPropertyBag(iface);
1933 VARIANT value;
1934 HRESULT iRes;
1935
1936 VariantInit(&value);
1937
1938 iRes = IPropertyBag_Read(pPropBag, sAttrName, &value, pErrorLog);
1939 if (iRes == S_OK)
1940 {
1941 iRes = VariantChangeType(&value, &value, 0, VT_BSTR);
1942 if (iRes == S_OK)
1943 iRes = IFont_put_Name(&this->IFont_iface, V_BSTR(&value));
1944 }
1945 else if (iRes == E_INVALIDARG)
1946 iRes = S_OK;
1947
1948 VariantClear(&value);
1949
1950 if (iRes == S_OK) {
1951 iRes = IPropertyBag_Read(pPropBag, sAttrSize, &value, pErrorLog);
1952 if (iRes == S_OK)
1953 {
1954 iRes = VariantChangeType(&value, &value, 0, VT_CY);
1955 if (iRes == S_OK)
1956 iRes = IFont_put_Size(&this->IFont_iface, V_CY(&value));
1957 }
1958 else if (iRes == E_INVALIDARG)
1959 iRes = S_OK;
1960
1961 VariantClear(&value);
1962 }
1963
1964 if (iRes == S_OK) {
1965 iRes = IPropertyBag_Read(pPropBag, sAttrCharset, &value, pErrorLog);
1966 if (iRes == S_OK)
1967 {
1968 iRes = VariantChangeType(&value, &value, 0, VT_I2);
1969 if (iRes == S_OK)
1970 iRes = IFont_put_Charset(&this->IFont_iface, V_I2(&value));
1971 }
1972 else if (iRes == E_INVALIDARG)
1973 iRes = S_OK;
1974
1975 VariantClear(&value);
1976 }
1977
1978 if (iRes == S_OK) {
1979 iRes = IPropertyBag_Read(pPropBag, sAttrWeight, &value, pErrorLog);
1980 if (iRes == S_OK)
1981 {
1982 iRes = VariantChangeType(&value, &value, 0, VT_I2);
1983 if (iRes == S_OK)
1984 iRes = IFont_put_Weight(&this->IFont_iface, V_I2(&value));
1985 }
1986 else if (iRes == E_INVALIDARG)
1987 iRes = S_OK;
1988
1989 VariantClear(&value);
1990 }
1991
1992 if (iRes == S_OK) {
1993 iRes = IPropertyBag_Read(pPropBag, sAttrUnderline, &value, pErrorLog);
1994 if (iRes == S_OK)
1995 {
1996 iRes = VariantChangeType(&value, &value, 0, VT_BOOL);
1997 if (iRes == S_OK)
1998 iRes = IFont_put_Underline(&this->IFont_iface, V_BOOL(&value));
1999 }
2000 else if (iRes == E_INVALIDARG)
2001 iRes = S_OK;
2002
2003 VariantClear(&value);
2004 }
2005
2006 if (iRes == S_OK) {
2007 iRes = IPropertyBag_Read(pPropBag, sAttrItalic, &value, pErrorLog);
2008 if (iRes == S_OK)
2009 {
2010 iRes = VariantChangeType(&value, &value, 0, VT_BOOL);
2011 if (iRes == S_OK)
2012 iRes = IFont_put_Italic(&this->IFont_iface, V_BOOL(&value));
2013 }
2014 else if (iRes == E_INVALIDARG)
2015 iRes = S_OK;
2016
2017 VariantClear(&value);
2018 }
2019
2020 if (iRes == S_OK) {
2021 iRes = IPropertyBag_Read(pPropBag, sAttrStrikethrough, &value, pErrorLog);
2022 if (iRes == S_OK)
2023 {
2024 iRes = VariantChangeType(&value, &value, 0, VT_BOOL);
2025 if (iRes == S_OK)
2026 IFont_put_Strikethrough(&this->IFont_iface, V_BOOL(&value));
2027 }
2028 else if (iRes == E_INVALIDARG)
2029 iRes = S_OK;
2030
2031 VariantClear(&value);
2032 }
2033
2034 if (FAILED(iRes))
2035 WARN("-- 0x%08x\n", iRes);
2036 return iRes;
2037 }
2038
OLEFontImpl_IPersistPropertyBag_Save(IPersistPropertyBag * iface,IPropertyBag * pPropBag,BOOL fClearDirty,BOOL fSaveAllProperties)2039 static HRESULT WINAPI OLEFontImpl_IPersistPropertyBag_Save(
2040 IPersistPropertyBag *iface, IPropertyBag* pPropBag, BOOL fClearDirty,
2041 BOOL fSaveAllProperties
2042 ) {
2043 FIXME("(%p,%p,%d,%d), stub!\n", iface, pPropBag, fClearDirty, fSaveAllProperties);
2044 return E_FAIL;
2045 }
2046
2047 static const IPersistPropertyBagVtbl OLEFontImpl_IPersistPropertyBag_VTable =
2048 {
2049 OLEFontImpl_IPersistPropertyBag_QueryInterface,
2050 OLEFontImpl_IPersistPropertyBag_AddRef,
2051 OLEFontImpl_IPersistPropertyBag_Release,
2052
2053 OLEFontImpl_IPersistPropertyBag_GetClassID,
2054 OLEFontImpl_IPersistPropertyBag_InitNew,
2055 OLEFontImpl_IPersistPropertyBag_Load,
2056 OLEFontImpl_IPersistPropertyBag_Save
2057 };
2058
2059 /************************************************************************
2060 * OLEFontImpl implementation of IPersistStreamInit.
2061 */
OLEFontImpl_IPersistStreamInit_QueryInterface(IPersistStreamInit * iface,REFIID riid,LPVOID * ppvObj)2062 static HRESULT WINAPI OLEFontImpl_IPersistStreamInit_QueryInterface(
2063 IPersistStreamInit *iface, REFIID riid, LPVOID *ppvObj
2064 ) {
2065 OLEFontImpl *this = impl_from_IPersistStreamInit(iface);
2066 return IFont_QueryInterface(&this->IFont_iface,riid,ppvObj);
2067 }
2068
OLEFontImpl_IPersistStreamInit_AddRef(IPersistStreamInit * iface)2069 static ULONG WINAPI OLEFontImpl_IPersistStreamInit_AddRef(
2070 IPersistStreamInit *iface
2071 ) {
2072 OLEFontImpl *this = impl_from_IPersistStreamInit(iface);
2073 return IFont_AddRef(&this->IFont_iface);
2074 }
2075
OLEFontImpl_IPersistStreamInit_Release(IPersistStreamInit * iface)2076 static ULONG WINAPI OLEFontImpl_IPersistStreamInit_Release(
2077 IPersistStreamInit *iface
2078 ) {
2079 OLEFontImpl *this = impl_from_IPersistStreamInit(iface);
2080 return IFont_Release(&this->IFont_iface);
2081 }
2082
OLEFontImpl_IPersistStreamInit_GetClassID(IPersistStreamInit * iface,CLSID * classid)2083 static HRESULT WINAPI OLEFontImpl_IPersistStreamInit_GetClassID(
2084 IPersistStreamInit *iface, CLSID *classid
2085 ) {
2086 FIXME("(%p,%p), stub!\n", iface, classid);
2087 return E_FAIL;
2088 }
2089
OLEFontImpl_IPersistStreamInit_IsDirty(IPersistStreamInit * iface)2090 static HRESULT WINAPI OLEFontImpl_IPersistStreamInit_IsDirty(
2091 IPersistStreamInit *iface
2092 ) {
2093 FIXME("(%p), stub!\n", iface);
2094 return E_FAIL;
2095 }
2096
OLEFontImpl_IPersistStreamInit_Load(IPersistStreamInit * iface,LPSTREAM pStm)2097 static HRESULT WINAPI OLEFontImpl_IPersistStreamInit_Load(
2098 IPersistStreamInit *iface, LPSTREAM pStm
2099 ) {
2100 FIXME("(%p,%p), stub!\n", iface, pStm);
2101 return E_FAIL;
2102 }
2103
OLEFontImpl_IPersistStreamInit_Save(IPersistStreamInit * iface,LPSTREAM pStm,BOOL fClearDirty)2104 static HRESULT WINAPI OLEFontImpl_IPersistStreamInit_Save(
2105 IPersistStreamInit *iface, LPSTREAM pStm, BOOL fClearDirty
2106 ) {
2107 FIXME("(%p,%p,%d), stub!\n", iface, pStm, fClearDirty);
2108 return E_FAIL;
2109 }
2110
OLEFontImpl_IPersistStreamInit_GetSizeMax(IPersistStreamInit * iface,ULARGE_INTEGER * pcbSize)2111 static HRESULT WINAPI OLEFontImpl_IPersistStreamInit_GetSizeMax(
2112 IPersistStreamInit *iface, ULARGE_INTEGER *pcbSize
2113 ) {
2114 FIXME("(%p,%p), stub!\n", iface, pcbSize);
2115 return E_FAIL;
2116 }
2117
OLEFontImpl_IPersistStreamInit_InitNew(IPersistStreamInit * iface)2118 static HRESULT WINAPI OLEFontImpl_IPersistStreamInit_InitNew(
2119 IPersistStreamInit *iface
2120 ) {
2121 FIXME("(%p), stub!\n", iface);
2122 return S_OK;
2123 }
2124
2125 static const IPersistStreamInitVtbl OLEFontImpl_IPersistStreamInit_VTable =
2126 {
2127 OLEFontImpl_IPersistStreamInit_QueryInterface,
2128 OLEFontImpl_IPersistStreamInit_AddRef,
2129 OLEFontImpl_IPersistStreamInit_Release,
2130
2131 OLEFontImpl_IPersistStreamInit_GetClassID,
2132 OLEFontImpl_IPersistStreamInit_IsDirty,
2133 OLEFontImpl_IPersistStreamInit_Load,
2134 OLEFontImpl_IPersistStreamInit_Save,
2135 OLEFontImpl_IPersistStreamInit_GetSizeMax,
2136 OLEFontImpl_IPersistStreamInit_InitNew
2137 };
2138
2139 /************************************************************************
2140 * OLEFontImpl_Construct
2141 *
2142 * This method will construct a new instance of the OLEFontImpl
2143 * class.
2144 *
2145 * The caller of this method must release the object when it's
2146 * done with it.
2147 */
OLEFontImpl_Construct(const FONTDESC * fontDesc)2148 static OLEFontImpl* OLEFontImpl_Construct(const FONTDESC *fontDesc)
2149 {
2150 OLEFontImpl* newObject;
2151
2152 newObject = HeapAlloc(GetProcessHeap(), 0, sizeof(OLEFontImpl));
2153
2154 if (newObject==0)
2155 return newObject;
2156
2157 newObject->IFont_iface.lpVtbl = &OLEFontImpl_VTable;
2158 newObject->IDispatch_iface.lpVtbl = &OLEFontImpl_IDispatch_VTable;
2159 newObject->IPersistStream_iface.lpVtbl = &OLEFontImpl_IPersistStream_VTable;
2160 newObject->IConnectionPointContainer_iface.lpVtbl = &OLEFontImpl_IConnectionPointContainer_VTable;
2161 newObject->IPersistPropertyBag_iface.lpVtbl = &OLEFontImpl_IPersistPropertyBag_VTable;
2162 newObject->IPersistStreamInit_iface.lpVtbl = &OLEFontImpl_IPersistStreamInit_VTable;
2163
2164 newObject->ref = 1;
2165
2166 newObject->description.cbSizeofstruct = sizeof(FONTDESC);
2167 newObject->description.lpstrName = strdupW(fontDesc->lpstrName);
2168 newObject->description.cySize = fontDesc->cySize;
2169 newObject->description.sWeight = fontDesc->sWeight;
2170 newObject->description.sCharset = fontDesc->sCharset;
2171 newObject->description.fItalic = fontDesc->fItalic;
2172 newObject->description.fUnderline = fontDesc->fUnderline;
2173 newObject->description.fStrikethrough = fontDesc->fStrikethrough;
2174
2175 newObject->gdiFont = 0;
2176 newObject->dirty = TRUE;
2177 newObject->cyLogical = GetDeviceCaps(get_dc(), LOGPIXELSY);
2178 newObject->cyHimetric = 2540L;
2179 newObject->pPropertyNotifyCP = NULL;
2180 newObject->pFontEventsCP = NULL;
2181
2182 CreateConnectionPoint((IUnknown*)&newObject->IFont_iface, &IID_IPropertyNotifySink, &newObject->pPropertyNotifyCP);
2183 CreateConnectionPoint((IUnknown*)&newObject->IFont_iface, &IID_IFontEventsDisp, &newObject->pFontEventsCP);
2184
2185 if (!newObject->pPropertyNotifyCP || !newObject->pFontEventsCP)
2186 {
2187 OLEFontImpl_Destroy(newObject);
2188 return NULL;
2189 }
2190
2191 InterlockedIncrement(&ifont_cnt);
2192
2193 TRACE("returning %p\n", newObject);
2194 return newObject;
2195 }
2196
2197 /************************************************************************
2198 * OLEFontImpl_Destroy
2199 *
2200 * This method is called by the Release method when the reference
2201 * count goes down to 0. It will free all resources used by
2202 * this object.
2203 */
OLEFontImpl_Destroy(OLEFontImpl * fontDesc)2204 static void OLEFontImpl_Destroy(OLEFontImpl* fontDesc)
2205 {
2206 TRACE("(%p)\n", fontDesc);
2207
2208 HeapFree(GetProcessHeap(), 0, fontDesc->description.lpstrName);
2209
2210 if (fontDesc->pPropertyNotifyCP)
2211 IConnectionPoint_Release(fontDesc->pPropertyNotifyCP);
2212 if (fontDesc->pFontEventsCP)
2213 IConnectionPoint_Release(fontDesc->pFontEventsCP);
2214
2215 HeapFree(GetProcessHeap(), 0, fontDesc);
2216 }
2217
2218 /*******************************************************************************
2219 * StdFont ClassFactory
2220 */
2221 typedef struct
2222 {
2223 /* IUnknown fields */
2224 IClassFactory IClassFactory_iface;
2225 LONG ref;
2226 } IClassFactoryImpl;
2227
impl_from_IClassFactory(IClassFactory * iface)2228 static inline IClassFactoryImpl *impl_from_IClassFactory(IClassFactory *iface)
2229 {
2230 return CONTAINING_RECORD(iface, IClassFactoryImpl, IClassFactory_iface);
2231 }
2232
SFCF_QueryInterface(IClassFactory * iface,REFIID riid,void ** obj)2233 static HRESULT WINAPI SFCF_QueryInterface(IClassFactory *iface, REFIID riid, void **obj)
2234 {
2235 IClassFactoryImpl *This = impl_from_IClassFactory(iface);
2236
2237 TRACE("(%p)->(%s, %p)\n", This, debugstr_guid(riid), obj);
2238
2239 *obj = NULL;
2240
2241 if (IsEqualIID(&IID_IClassFactory, riid) || IsEqualIID(&IID_IUnknown, riid))
2242 {
2243 *obj = iface;
2244 IClassFactory_AddRef(iface);
2245 return S_OK;
2246 }
2247
2248 return E_NOINTERFACE;
2249 }
2250
2251 static ULONG WINAPI
SFCF_AddRef(LPCLASSFACTORY iface)2252 SFCF_AddRef(LPCLASSFACTORY iface) {
2253 IClassFactoryImpl *This = impl_from_IClassFactory(iface);
2254 return InterlockedIncrement(&This->ref);
2255 }
2256
SFCF_Release(LPCLASSFACTORY iface)2257 static ULONG WINAPI SFCF_Release(LPCLASSFACTORY iface) {
2258 IClassFactoryImpl *This = impl_from_IClassFactory(iface);
2259 /* static class, won't be freed */
2260 return InterlockedDecrement(&This->ref);
2261 }
2262
SFCF_CreateInstance(LPCLASSFACTORY iface,LPUNKNOWN pOuter,REFIID riid,LPVOID * ppobj)2263 static HRESULT WINAPI SFCF_CreateInstance(
2264 LPCLASSFACTORY iface,LPUNKNOWN pOuter,REFIID riid,LPVOID *ppobj
2265 ) {
2266 return OleCreateFontIndirect(NULL,riid,ppobj);
2267
2268 }
2269
SFCF_LockServer(LPCLASSFACTORY iface,BOOL dolock)2270 static HRESULT WINAPI SFCF_LockServer(LPCLASSFACTORY iface,BOOL dolock) {
2271 IClassFactoryImpl *This = impl_from_IClassFactory(iface);
2272 FIXME("(%p)->(%d),stub!\n",This,dolock);
2273 return S_OK;
2274 }
2275
2276 static const IClassFactoryVtbl SFCF_Vtbl = {
2277 SFCF_QueryInterface,
2278 SFCF_AddRef,
2279 SFCF_Release,
2280 SFCF_CreateInstance,
2281 SFCF_LockServer
2282 };
2283 static IClassFactoryImpl STDFONT_CF = {{&SFCF_Vtbl}, 1 };
2284
_get_STDFONT_CF(LPVOID * ppv)2285 void _get_STDFONT_CF(LPVOID *ppv) { *ppv = &STDFONT_CF; }
2286