f9c5625701c4ed541d19eefb08f45956eee49a2a
[reactos.git] / modules / rostests / winetests / oleaut32 / olefont.c
1 /*
2 * OLEFONT test program
3 *
4 * Copyright 2003 Marcus Meissner
5 * Copyright 2006 (Google) Benjamin Arai
6 *
7 *
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2.1 of the License, or (at your option) any later version.
12 *
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
17 *
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with this library; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 */
22
23 #include <stdarg.h>
24 #include <stdio.h>
25 #include <math.h>
26 #include <float.h>
27
28 #define COBJMACROS
29
30 #include <wine/test.h>
31 #include <windef.h>
32 #include <winbase.h>
33 #include <winuser.h>
34 #include <wingdi.h>
35 #include <winnls.h>
36 #include <winerror.h>
37 #include <winnt.h>
38 #include <initguid.h>
39 #include <wtypes.h>
40 #include <olectl.h>
41 #include <ocidl.h>
42
43 DEFINE_GUID(GUID_NULL,0,0,0,0,0,0,0,0,0,0,0);
44
45 static WCHAR MSSansSerif_font[] = {'M','S',' ','S','a','n','s',' ','S','e','r','i','f',0};
46 static WCHAR system_font[] = { 'S','y','s','t','e','m',0 };
47 static WCHAR arial_font[] = { 'A','r','i','a','l',0 };
48 static WCHAR marlett_font[] = { 'M','a','r','l','e','t','t',0 };
49
50 static HMODULE hOleaut32;
51
52 static HRESULT (WINAPI *pOleCreateFontIndirect)(LPFONTDESC,REFIID,LPVOID*);
53
54 #define EXPECT_HR(hr,hr_exp) \
55 ok(hr == hr_exp, "got 0x%08x, expected 0x%08x\n", hr, hr_exp)
56
57 /* Create a font with cySize given by lo_size, hi_size, */
58 /* SetRatio to ratio_logical, ratio_himetric, */
59 /* check that resulting hfont has height hfont_height. */
60 /* Various checks along the way. */
61 static void test_ifont_size(LONGLONG size, LONG ratio_logical, LONG ratio_himetric,
62 LONG hfont_height, const char * test_name)
63 {
64 FONTDESC fd;
65 LPVOID pvObj = NULL;
66 IFont* ifnt = NULL;
67 HFONT hfont;
68 LOGFONTA lf;
69 CY psize;
70 HRESULT hres;
71 DWORD rtnval;
72
73 fd.cbSizeofstruct = sizeof(FONTDESC);
74 fd.lpstrName = arial_font; /* using scalable instead of bitmap font reduces errors due to font realization */
75 fd.cySize.int64 = size;
76 fd.sWeight = 0;
77 fd.sCharset = 0;
78 fd.fItalic = FALSE;
79 fd.fUnderline = FALSE;
80 fd.fStrikethrough = FALSE;
81
82 /* Create font, test that it worked. */
83 hres = pOleCreateFontIndirect(&fd, &IID_IFont, &pvObj);
84 ifnt = pvObj;
85 ok(hres == S_OK,"%s: OCFI returns 0x%08x instead of S_OK.\n",
86 test_name, hres);
87 ok(pvObj != NULL,"%s: OCFI returns NULL.\n", test_name);
88
89 /* Change the scaling ratio */
90 hres = IFont_SetRatio(ifnt, ratio_logical, ratio_himetric);
91 ok((ratio_logical && ratio_himetric) ? hres == S_OK : hres == E_FAIL,
92 "%s: IFont_SetRatio unexpectedly returned 0x%08x.\n", test_name, hres);
93
94 /* Read back size. */
95 hres = IFont_get_Size(ifnt, &psize);
96 ok(hres == S_OK,"%s: IFont_get_size returns 0x%08x instead of S_OK.\n",
97 test_name, hres);
98
99 /* Check returned size - allow for errors due to rounding & font realization. */
100 ok((psize.int64 - size) < 10000 && (psize.int64 - size) > -10000,
101 "%s: IFont_get_Size: Lo=%d, Hi=%d; expected Lo=%d, Hi=%d.\n",
102 test_name, S(psize).Lo, S(psize).Hi, fd.cySize.Lo, fd.cySize.Hi);
103
104 /* Check hFont size. */
105 hres = IFont_get_hFont (ifnt, &hfont);
106 ok(hres == S_OK, "%s: IFont_get_hFont returns 0x%08x instead of S_OK.\n",
107 test_name, hres);
108 rtnval = GetObjectA(hfont, sizeof(LOGFONTA), &lf);
109 ok(rtnval > 0, "GetObject(hfont) failed\n");
110
111 /* Since font scaling may encounter rounding errors, allow 1 pixel deviation. */
112 ok(abs(lf.lfHeight - hfont_height) <= 1,
113 "%s: hFont has lf.lfHeight=%d, expected %d.\n",
114 test_name, lf.lfHeight, hfont_height);
115
116 /* Free IFont. */
117 IFont_Release(ifnt);
118 }
119
120 static void test_ifont_sizes(void)
121 {
122 /* Test various size operations and conversions. */
123 /* Add more as needed. */
124
125 /* Results of first 2 tests depend on display resolution. */
126 HDC hdc = GetDC(0);
127 LONG dpi = GetDeviceCaps(hdc, LOGPIXELSY); /* expected results depend on display DPI */
128 ReleaseDC(0, hdc);
129 if(dpi == 96) /* normal resolution display */
130 {
131 test_ifont_size(180000, 0, 0, -24, "default"); /* normal font */
132 test_ifont_size(186000, 0, 0, -25, "rounding"); /* test rounding */
133 } else if(dpi == 72) /* low resolution display */
134 {
135 test_ifont_size(180000, 0, 0, -18, "default"); /* normal font */
136 test_ifont_size(186000, 0, 0, -19, "rounding"); /* test rounding */
137 } else if(dpi == 120) /* high resolution display */
138 {
139 test_ifont_size(180000, 0, 0, -30, "default"); /* normal font */
140 test_ifont_size(186000, 0, 0, -31, "rounding"); /* test rounding */
141 } else
142 skip("Skipping resolution dependent font size tests - display resolution is %d\n", dpi);
143
144 /* Next 4 tests specify a scaling ratio, so display resolution is not a factor. */
145 test_ifont_size(180000, 72, 2540, -18, "ratio1"); /* change ratio */
146 test_ifont_size(180000, 144, 2540, -36, "ratio2"); /* another ratio */
147 test_ifont_size(180000, 72, 1270, -36, "ratio3"); /* yet another ratio */
148 test_ifont_size(186000, 72, 2540, -19, "rounding+ratio"); /* test rounding with ratio */
149
150 /* test various combinations of logical == himetric */
151 test_ifont_size(180000, 10, 10, -635, "identical ratio 1");
152 test_ifont_size(240000, 10, 10, -848, "identical ratio 2");
153 test_ifont_size(300000, 10, 10, -1058, "identical ratio 3");
154
155 /* test various combinations of logical and himetric both set to 1 */
156 test_ifont_size(180000, 1, 1, -24, "1:1 ratio 1");
157 test_ifont_size(240000, 1, 1, -32, "1:1 ratio 2");
158 test_ifont_size(300000, 1, 1, -40, "1:1 ratio 3");
159
160 /* test various combinations of logical set to 1 */
161 test_ifont_size(180000, 1, 0, -24, "1:0 ratio 1");
162 test_ifont_size(240000, 1, 0, -32, "1:0 ratio 2");
163 test_ifont_size(300000, 1, 0, -40, "1:0 ratio 3");
164
165 /* test various combinations of himetric set to 1 */
166 test_ifont_size(180000, 0, 1, -24, "0:1 ratio 1");
167 test_ifont_size(240000, 0, 1, -32, "0:1 ratio 2");
168 test_ifont_size(300000, 0, 1, -40, "0:1 ratio 3");
169
170 /* test various combinations of 2:1 logical:himetric */
171 test_ifont_size(180000, 2, 1, -1270, "2:1 ratio 1");
172 test_ifont_size(240000, 2, 1, -1694, "2:1 ratio 2");
173 test_ifont_size(300000, 2, 1, -2117, "2:1 ratio 3");
174
175 /* test various combinations of 1:2 logical:himetric */
176 test_ifont_size(180000, 1, 2, -318, "1:2 ratio 1");
177 test_ifont_size(240000, 1, 2, -424, "1:2 ratio 2");
178 test_ifont_size(300000, 1, 2, -529, "1:2 ratio 3");
179
180 /* test various combinations of logical and himetric both set to 2 */
181 test_ifont_size(180000, 2, 2, -635, "2:2 ratio 1");
182 test_ifont_size(240000, 2, 2, -848, "2:2 ratio 2");
183 test_ifont_size(300000, 2, 2, -1058, "2:2 ratio 3");
184 }
185
186 static void test_QueryInterface(void)
187 {
188 LPVOID pvObj = NULL;
189 HRESULT hr;
190 IFont* font = NULL;
191 LONG ref;
192
193 hr = pOleCreateFontIndirect(NULL, &IID_IFont, NULL);
194 EXPECT_HR(hr, E_POINTER);
195
196 hr = pOleCreateFontIndirect(NULL, &IID_IFont, &pvObj);
197 font = pvObj;
198
199 EXPECT_HR(hr, S_OK);
200 ok(font != NULL,"OCFI (NULL,..) returns NULL, instead of !NULL\n");
201
202 pvObj = NULL;
203 hr = IFont_QueryInterface( font, &IID_IFont, &pvObj);
204 EXPECT_HR(hr, S_OK);
205
206 /* Test if QueryInterface increments ref counter for IFONTs */
207 ref = IFont_AddRef(font);
208 ok(ref == 3 ||
209 broken(ref == 1), /* win95 */
210 "IFont_QI expected ref value 3 but instead got %d\n", ref);
211 IFont_Release(font);
212
213 ok(pvObj != NULL,"IFont_QI does return NULL, instead of a ptr\n");
214
215 IFont_Release(font);
216 IFont_Release(font);
217 }
218
219 static void test_type_info(void)
220 {
221 LPVOID pvObj = NULL;
222 HRESULT hres;
223 IFontDisp* fontdisp = NULL;
224 ITypeInfo* pTInfo;
225 WCHAR name_Name[] = {'N','a','m','e',0};
226 BSTR names[3];
227 UINT n;
228 LCID en_us = MAKELCID(MAKELANGID(LANG_ENGLISH,SUBLANG_ENGLISH_US),
229 SORT_DEFAULT);
230 DISPPARAMS dispparams;
231 VARIANT varresult;
232
233 pOleCreateFontIndirect(NULL, &IID_IFontDisp, &pvObj);
234 fontdisp = pvObj;
235
236 hres = IFontDisp_GetTypeInfo(fontdisp, 0, en_us, &pTInfo);
237 ok(hres == S_OK, "GTI returned 0x%08x instead of S_OK.\n", hres);
238 ok(pTInfo != NULL, "GTI returned NULL.\n");
239
240 hres = ITypeInfo_GetNames(pTInfo, DISPID_FONT_NAME, names, 3, &n);
241 ok(hres == S_OK, "GetNames returned 0x%08x instead of S_OK.\n", hres);
242 ok(n == 1, "GetNames returned %d names instead of 1.\n", n);
243 ok(!lstrcmpiW(names[0],name_Name), "DISPID_FONT_NAME doesn't get 'Names'.\n");
244 SysFreeString(names[0]);
245
246 ITypeInfo_Release(pTInfo);
247
248 dispparams.cNamedArgs = 0;
249 dispparams.rgdispidNamedArgs = NULL;
250 dispparams.cArgs = 0;
251 dispparams.rgvarg = NULL;
252 VariantInit(&varresult);
253 hres = IFontDisp_Invoke(fontdisp, DISPID_FONT_NAME, &IID_NULL,
254 LOCALE_NEUTRAL, DISPATCH_PROPERTYGET, &dispparams, &varresult,
255 NULL, NULL);
256 ok(hres == S_OK, "IFontDisp_Invoke return 0x%08x instead of S_OK.\n", hres);
257 VariantClear(&varresult);
258
259 IFontDisp_Release(fontdisp);
260 }
261
262 static HRESULT WINAPI FontEventsDisp_QueryInterface(IFontEventsDisp *iface, REFIID riid, void **ppvObject)
263 {
264 if (IsEqualIID(riid, &IID_IFontEventsDisp) || IsEqualIID(riid, &IID_IUnknown) || IsEqualIID(riid, &IID_IDispatch))
265 {
266 IFontEventsDisp_AddRef(iface);
267 *ppvObject = iface;
268 return S_OK;
269 }
270 else
271 {
272 *ppvObject = NULL;
273 return E_NOINTERFACE;
274 }
275 }
276
277 static ULONG WINAPI FontEventsDisp_AddRef(
278 IFontEventsDisp *iface)
279 {
280 return 2;
281 }
282
283 static ULONG WINAPI FontEventsDisp_Release(
284 IFontEventsDisp *iface)
285 {
286 return 1;
287 }
288
289 static HRESULT WINAPI FontEventsDisp_GetTypeInfoCount(IFontEventsDisp *iface, UINT *pctinfo)
290 {
291 ok(0, "unexpected call\n");
292 return E_NOTIMPL;
293 }
294
295 static HRESULT WINAPI FontEventsDisp_GetTypeInfo(IFontEventsDisp *iface, UINT itinfo, LCID lcid, ITypeInfo **pptinfo)
296 {
297 ok(0, "unexpected call\n");
298 return E_NOTIMPL;
299 }
300
301 static HRESULT WINAPI FontEventsDisp_GetIDsOfNames(IFontEventsDisp *iface, REFIID riid, LPOLESTR *names, UINT cNames, LCID lcid,
302 DISPID *dispid)
303 {
304 ok(0, "unexpected call\n");
305 return E_NOTIMPL;
306 }
307
308 static int fonteventsdisp_invoke_called;
309 static BSTR fonteventsdisp_invoke_arg0;
310
311 static HRESULT WINAPI FontEventsDisp_Invoke(
312 IFontEventsDisp *iface,
313 DISPID dispid,
314 REFIID riid,
315 LCID lcid,
316 WORD wFlags,
317 DISPPARAMS *pDispParams,
318 VARIANT *pVarResult,
319 EXCEPINFO *pExcepInfo,
320 UINT *puArgErr)
321 {
322 VARIANTARG *arg0 = &pDispParams->rgvarg[0];
323
324 ok(dispid == DISPID_FONT_CHANGED, "expected DISPID_FONT_CHANGED instead of 0x%x\n", dispid);
325 ok(IsEqualGUID(riid, &GUID_NULL), "got riid %s\n", wine_dbgstr_guid(riid));
326 ok(wFlags == INVOKE_FUNC, "expected INVOKE_FUNC instead of 0x%x\n", wFlags);
327 ok(pDispParams->cArgs == 1, "expected arg count 1, got %d\n", pDispParams->cArgs);
328 ok(V_VT(arg0) == VT_BSTR, "expected VT_BSTR, got %d\n", V_VT(arg0));
329
330 fonteventsdisp_invoke_arg0 = SysAllocString(V_BSTR(arg0));
331 fonteventsdisp_invoke_called++;
332 return S_OK;
333 }
334
335 static IFontEventsDispVtbl FontEventsDisp_Vtbl =
336 {
337 FontEventsDisp_QueryInterface,
338 FontEventsDisp_AddRef,
339 FontEventsDisp_Release,
340 FontEventsDisp_GetTypeInfoCount,
341 FontEventsDisp_GetTypeInfo,
342 FontEventsDisp_GetIDsOfNames,
343 FontEventsDisp_Invoke
344 };
345
346 static IFontEventsDisp FontEventsDisp = { &FontEventsDisp_Vtbl };
347
348 struct font_dispid
349 {
350 DISPID dispid;
351 const WCHAR *name;
352 };
353
354 static void test_font_events_disp(void)
355 {
356 static const WCHAR nameW[] = {'N','a','m','e',0};
357 static const WCHAR sizeW[] = {'S','i','z','e',0};
358 static const WCHAR boldW[] = {'B','o','l','d',0};
359 static const WCHAR italicW[] = {'I','t','a','l','i','c',0};
360 static const WCHAR underlineW[] = {'U','n','d','e','r','l','i','n','e',0};
361 static const WCHAR strikeW[] = {'S','t','r','i','k','e','t','h','r','o','u','g','h',0};
362 static const WCHAR weightW[] = {'W','e','i','g','h','t',0};
363 static const WCHAR charsetW[] = {'C','h','a','r','s','e','t',0};
364
365 static const struct font_dispid font_dispids[] =
366 {
367 { DISPID_FONT_NAME, nameW },
368 { DISPID_FONT_SIZE, sizeW },
369 { DISPID_FONT_BOLD, boldW },
370 { DISPID_FONT_ITALIC, italicW },
371 { DISPID_FONT_UNDER, underlineW },
372 { DISPID_FONT_STRIKE, strikeW },
373 { DISPID_FONT_WEIGHT, weightW },
374 { DISPID_FONT_CHARSET, charsetW }
375 };
376
377 IFont *pFont;
378 IFont *pFont2;
379 IConnectionPointContainer *pCPC;
380 IConnectionPoint *pCP;
381 FONTDESC fontdesc;
382 HRESULT hr;
383 DWORD dwCookie;
384 IFontDisp *pFontDisp;
385 DISPPARAMS dispparams;
386 VARIANTARG vararg;
387 INT i;
388
389 fontdesc.cbSizeofstruct = sizeof(fontdesc);
390 fontdesc.lpstrName = MSSansSerif_font;
391 fontdesc.cySize.int64 = 12 * 10000; /* 12 pt */
392 fontdesc.sWeight = FW_NORMAL;
393 fontdesc.sCharset = 0;
394 fontdesc.fItalic = FALSE;
395 fontdesc.fUnderline = FALSE;
396 fontdesc.fStrikethrough = FALSE;
397
398 hr = pOleCreateFontIndirect(&fontdesc, &IID_IFont, (void **)&pFont);
399 EXPECT_HR(hr, S_OK);
400
401 hr = IFont_QueryInterface(pFont, &IID_IConnectionPointContainer, (void **)&pCPC);
402 EXPECT_HR(hr, S_OK);
403
404 hr = IConnectionPointContainer_FindConnectionPoint(pCPC, &IID_IFontEventsDisp, &pCP);
405 EXPECT_HR(hr, S_OK);
406 IConnectionPointContainer_Release(pCPC);
407
408 hr = IConnectionPoint_Advise(pCP, (IUnknown *)&FontEventsDisp, &dwCookie);
409 EXPECT_HR(hr, S_OK);
410 IConnectionPoint_Release(pCP);
411
412 fonteventsdisp_invoke_called = 0;
413 fonteventsdisp_invoke_arg0 = NULL;
414 hr = IFont_put_Bold(pFont, TRUE);
415 EXPECT_HR(hr, S_OK);
416
417 ok(fonteventsdisp_invoke_called == 1, "IFontEventDisp::Invoke wasn't called once\n");
418 SysFreeString(fonteventsdisp_invoke_arg0);
419
420 hr = IFont_QueryInterface(pFont, &IID_IFontDisp, (void **)&pFontDisp);
421 EXPECT_HR(hr, S_OK);
422
423 for (i = 0; i < sizeof(font_dispids)/sizeof(font_dispids[0]); i++)
424 {
425 switch (font_dispids[i].dispid)
426 {
427 case DISPID_FONT_NAME:
428 {
429 static const WCHAR arialW[] = {'A','r','i','a','l',0};
430 V_VT(&vararg) = VT_BSTR;
431 V_BSTR(&vararg) = SysAllocString(arialW);
432 break;
433 }
434 case DISPID_FONT_SIZE:
435 V_VT(&vararg) = VT_CY;
436 S(V_CY(&vararg)).Lo = 25;
437 S(V_CY(&vararg)).Hi = 0;
438 break;
439 case DISPID_FONT_BOLD:
440 V_VT(&vararg) = VT_BOOL;
441 V_BOOL(&vararg) = VARIANT_FALSE;
442 break;
443 case DISPID_FONT_ITALIC:
444 case DISPID_FONT_UNDER:
445 case DISPID_FONT_STRIKE:
446 V_VT(&vararg) = VT_BOOL;
447 V_BOOL(&vararg) = VARIANT_TRUE;
448 break;
449 case DISPID_FONT_WEIGHT:
450 V_VT(&vararg) = VT_I2;
451 V_I2(&vararg) = FW_BLACK;
452 break;
453 case DISPID_FONT_CHARSET:
454 V_VT(&vararg) = VT_I2;
455 V_I2(&vararg) = 1;
456 break;
457 default:
458 ;
459 }
460
461 dispparams.cNamedArgs = 0;
462 dispparams.rgdispidNamedArgs = NULL;
463 dispparams.cArgs = 1;
464 dispparams.rgvarg = &vararg;
465 fonteventsdisp_invoke_called = 0;
466 hr = IFontDisp_Invoke(pFontDisp, font_dispids[i].dispid, &IID_NULL, 0, DISPATCH_PROPERTYPUT, &dispparams, NULL, NULL, NULL);
467 ok(hr == S_OK, "dispid=%d, got 0x%08x\n", font_dispids[i].dispid, hr);
468 ok(fonteventsdisp_invoke_called == 1, "dispid=%d, DISPID_FONT_CHANGED not called, got %d\n", font_dispids[i].dispid,
469 fonteventsdisp_invoke_called);
470 if (hr == S_OK)
471 {
472 ok(!lstrcmpW(font_dispids[i].name, fonteventsdisp_invoke_arg0), "dispid=%d, got %s, expected %s\n",
473 font_dispids[i].dispid, wine_dbgstr_w(fonteventsdisp_invoke_arg0), wine_dbgstr_w(font_dispids[i].name));
474 SysFreeString(fonteventsdisp_invoke_arg0);
475 }
476 VariantClear(&vararg);
477 }
478
479 IFontDisp_Release(pFontDisp);
480
481 hr = IFont_Clone(pFont, &pFont2);
482 EXPECT_HR(hr, S_OK);
483 IFont_Release(pFont);
484
485 /* this test shows that the notification routine isn't called again */
486 fonteventsdisp_invoke_called = 0;
487 hr = IFont_put_Bold(pFont2, FALSE);
488 EXPECT_HR(hr, S_OK);
489 ok(fonteventsdisp_invoke_called == 0, "got %d\n", fonteventsdisp_invoke_called);
490
491 IFont_Release(pFont2);
492 }
493
494 static void test_names_ids(WCHAR* w_name_1, const char* a_name_1,
495 WCHAR* w_name_2, const char* a_name_2,
496 LCID lcid, DISPID id_1, DISPID id_2,
497 HRESULT hres_expect, int numnames)
498 {
499 LPVOID pvObj = NULL;
500 IFontDisp *fontdisp = NULL;
501 HRESULT hres;
502 DISPID rgDispId[2] = {0xdeadbeef, 0xdeadbeef};
503 LPOLESTR names[2] = {w_name_1, w_name_2};
504
505 pOleCreateFontIndirect(NULL, &IID_IFontDisp, &pvObj);
506 fontdisp = pvObj;
507
508 hres = IFontDisp_GetIDsOfNames(fontdisp, &IID_NULL, names, numnames,
509 lcid, rgDispId);
510
511 /* test hres */
512 ok(hres == hres_expect,
513 "GetIDsOfNames: \"%s\", \"%s\" returns 0x%08x, expected 0x%08x.\n",
514 a_name_1, a_name_2, hres, hres_expect);
515
516 /* test first DISPID */
517 ok(rgDispId[0]==id_1,
518 "GetIDsOfNames: \"%s\" gets DISPID 0x%08x, expected 0x%08x.\n",
519 a_name_1, rgDispId[0], id_1);
520
521 /* test second DISPID is present */
522 if (numnames == 2)
523 {
524 ok(rgDispId[1]==id_2,
525 "GetIDsOfNames: ..., \"%s\" gets DISPID 0x%08x, expected 0x%08x.\n",
526 a_name_2, rgDispId[1], id_2);
527 }
528
529 IFontDisp_Release(fontdisp);
530 }
531
532 static void test_GetIDsOfNames(void)
533 {
534 WCHAR name_Name[] = {'N','a','m','e',0};
535 WCHAR name_Italic[] = {'I','t','a','l','i','c',0};
536 WCHAR name_Size[] = {'S','i','z','e',0};
537 WCHAR name_Bold[] = {'B','o','l','d',0};
538 WCHAR name_Underline[] = {'U','n','d','e','r','l','i','n','e',0};
539 WCHAR name_Strikethrough[] = {'S','t','r','i','k','e','t','h','r','o','u','g','h',0};
540 WCHAR name_Weight[] = {'W','e','i','g','h','t',0};
541 WCHAR name_Charset[] = {'C','h','a','r','s','e','t',0};
542 WCHAR name_Foo[] = {'F','o','o',0};
543 WCHAR name_nAmE[] = {'n','A','m','E',0};
544 WCHAR name_Nom[] = {'N','o','m',0};
545
546 LCID en_us = MAKELCID(MAKELANGID(LANG_ENGLISH,SUBLANG_ENGLISH_US),
547 SORT_DEFAULT);
548 LCID fr_fr = MAKELCID(MAKELANGID(LANG_FRENCH,SUBLANG_FRENCH),
549 SORT_DEFAULT);
550
551 /* Test DISPID_FONTs for the various properties. */
552 test_names_ids(name_Name, "Name", NULL, "", en_us,
553 DISPID_FONT_NAME, 0, S_OK,1);
554 test_names_ids(name_Size, "Size", NULL, "", en_us,
555 DISPID_FONT_SIZE, 0, S_OK,1);
556 test_names_ids(name_Bold, "Bold", NULL, "", en_us,
557 DISPID_FONT_BOLD, 0, S_OK,1);
558 test_names_ids(name_Italic, "Italic", NULL, "", en_us,
559 DISPID_FONT_ITALIC, 0, S_OK,1);
560 test_names_ids(name_Underline, "Underline", NULL, "", en_us,
561 DISPID_FONT_UNDER, 0, S_OK,1);
562 test_names_ids(name_Strikethrough, "Strikethrough", NULL, "", en_us,
563 DISPID_FONT_STRIKE, 0, S_OK,1);
564 test_names_ids(name_Weight, "Weight", NULL, "", en_us,
565 DISPID_FONT_WEIGHT, 0, S_OK,1);
566 test_names_ids(name_Charset, "Charset", NULL, "", en_us,
567 DISPID_FONT_CHARSET, 0, S_OK,1);
568
569 /* Capitalization doesn't matter. */
570 test_names_ids(name_nAmE, "nAmE", NULL, "", en_us,
571 DISPID_FONT_NAME, 0, S_OK,1);
572
573 /* Unknown name. */
574 test_names_ids(name_Foo, "Foo", NULL, "", en_us,
575 DISPID_UNKNOWN, 0, DISP_E_UNKNOWNNAME,1);
576
577 /* Pass several names: first is processed, */
578 /* second gets DISPID_UNKNOWN and doesn't affect retval. */
579 test_names_ids(name_Italic, "Italic", name_Name, "Name", en_us,
580 DISPID_FONT_ITALIC, DISPID_UNKNOWN, S_OK,2);
581 test_names_ids(name_Italic, "Italic", name_Foo, "Foo", en_us,
582 DISPID_FONT_ITALIC, DISPID_UNKNOWN, S_OK,2);
583
584 /* Locale ID has no effect. */
585 test_names_ids(name_Name, "Name", NULL, "", fr_fr,
586 DISPID_FONT_NAME, 0, S_OK,1);
587 test_names_ids(name_Nom, "This is not a font", NULL, "", fr_fr,
588 DISPID_UNKNOWN, 0, DISP_E_UNKNOWNNAME,1);
589
590 /* One of the arguments are invalid */
591 test_names_ids(name_Name, "Name", NULL, "", en_us,
592 0xdeadbeef, 0xdeadbeef, E_INVALIDARG,0);
593 test_names_ids(name_Italic, "Italic", NULL, "", en_us,
594 0xdeadbeef, 0xdeadbeef, E_INVALIDARG,0);
595 test_names_ids(name_Foo, "Foo", NULL, "", en_us,
596 0xdeadbeef, 0xdeadbeef, E_INVALIDARG,0);
597
598 /* Crazy locale ID? */
599 test_names_ids(name_Name, "Name", NULL, "", -1,
600 DISPID_FONT_NAME, 0, S_OK,1);
601 }
602
603 static void test_Invoke(void)
604 {
605 IFontDisp *fontdisp;
606 HRESULT hr;
607 VARIANTARG vararg;
608 DISPPARAMS dispparams;
609 VARIANT varresult;
610
611 hr = pOleCreateFontIndirect(NULL, &IID_IFontDisp, (void **)&fontdisp);
612 EXPECT_HR(hr, S_OK);
613
614 V_VT(&vararg) = VT_BOOL;
615 V_BOOL(&vararg) = VARIANT_FALSE;
616 dispparams.cNamedArgs = 0;
617 dispparams.rgdispidNamedArgs = NULL;
618 dispparams.cArgs = 1;
619 dispparams.rgvarg = &vararg;
620 hr = IFontDisp_Invoke(fontdisp, DISPID_FONT_BOLD, &IID_IFontDisp, 0, DISPATCH_PROPERTYPUT, &dispparams, NULL, NULL, NULL);
621 EXPECT_HR(hr, DISP_E_UNKNOWNINTERFACE);
622
623 dispparams.cArgs = 0;
624 dispparams.rgvarg = NULL;
625 hr = IFontDisp_Invoke(fontdisp, DISPID_FONT_BOLD, &IID_NULL, 0, DISPATCH_PROPERTYPUT, &dispparams, NULL, NULL, NULL);
626 EXPECT_HR(hr, DISP_E_BADPARAMCOUNT);
627
628 hr = IFontDisp_Invoke(fontdisp, DISPID_FONT_BOLD, &IID_NULL, 0, DISPATCH_PROPERTYPUT, NULL, NULL, NULL, NULL);
629 EXPECT_HR(hr, DISP_E_PARAMNOTOPTIONAL);
630
631 hr = IFontDisp_Invoke(fontdisp, DISPID_FONT_BOLD, &IID_NULL, 0, DISPATCH_PROPERTYGET, NULL, NULL, NULL, NULL);
632 EXPECT_HR(hr, DISP_E_PARAMNOTOPTIONAL);
633
634 hr = IFontDisp_Invoke(fontdisp, DISPID_FONT_BOLD, &IID_NULL, 0, DISPATCH_PROPERTYGET, NULL, &varresult, NULL, NULL);
635 EXPECT_HR(hr, S_OK);
636
637 hr = IFontDisp_Invoke(fontdisp, DISPID_FONT_BOLD, &IID_NULL, 0, DISPATCH_METHOD, NULL, &varresult, NULL, NULL);
638 EXPECT_HR(hr, DISP_E_MEMBERNOTFOUND);
639
640 hr = IFontDisp_Invoke(fontdisp, 0xdeadbeef, &IID_NULL, 0, DISPATCH_PROPERTYGET, NULL, &varresult, NULL, NULL);
641 EXPECT_HR(hr, DISP_E_MEMBERNOTFOUND);
642
643 dispparams.cArgs = 1;
644 dispparams.rgvarg = &vararg;
645 hr = IFontDisp_Invoke(fontdisp, DISPID_FONT_BOLD, &IID_NULL, 0, DISPATCH_PROPERTYGET, &dispparams, &varresult, NULL, NULL);
646 EXPECT_HR(hr, S_OK);
647
648 IFontDisp_Release(fontdisp);
649 }
650
651 static void test_IsEqual(void)
652 {
653 FONTDESC fd;
654 IFont* ifnt = NULL;
655 IFont* ifnt2 = NULL;
656 HRESULT hres;
657
658 /* Basic font description */
659 fd.cbSizeofstruct = sizeof(FONTDESC);
660 fd.lpstrName = system_font;
661 S(fd.cySize).Lo = 100;
662 S(fd.cySize).Hi = 100;
663 fd.sWeight = 0;
664 fd.sCharset = 0;
665 fd.fItalic = FALSE;
666 fd.fUnderline = FALSE;
667 fd.fStrikethrough = FALSE;
668
669 /* Create font */
670 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt);
671
672 /* Test equal fonts */
673 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt2);
674 hres = IFont_IsEqual(ifnt,ifnt2);
675 ok(hres == S_OK,
676 "IFont_IsEqual: (EQUAL) Expected S_OK but got 0x%08x\n",hres);
677 IFont_Release(ifnt2);
678
679 /* Check for bad pointer */
680 hres = IFont_IsEqual(ifnt,NULL);
681 ok(hres == E_POINTER,
682 "IFont_IsEqual: (NULL) Expected 0x80004003 but got 0x%08x\n",hres);
683
684 /* Test strName */
685 fd.lpstrName = arial_font;
686 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt2);
687 hres = IFont_IsEqual(ifnt,ifnt2);
688 ok(hres == S_FALSE,
689 "IFont_IsEqual: (strName) Expected S_FALSE but got 0x%08x\n",hres);
690 fd.lpstrName = system_font;
691 IFont_Release(ifnt2);
692
693 /* Test lo font size */
694 S(fd.cySize).Lo = 10000;
695 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt2);
696 hres = IFont_IsEqual(ifnt,ifnt2);
697 ok(hres == S_FALSE,
698 "IFont_IsEqual: (Lo font size) Expected S_FALSE but got 0x%08x\n",hres);
699 S(fd.cySize).Lo = 100;
700 IFont_Release(ifnt2);
701
702 /* Test hi font size */
703 S(fd.cySize).Hi = 10000;
704 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt2);
705 hres = IFont_IsEqual(ifnt,ifnt2);
706 ok(hres == S_FALSE,
707 "IFont_IsEqual: (Hi font size) Expected S_FALSE but got 0x%08x\n",hres);
708 S(fd.cySize).Hi = 100;
709 IFont_Release(ifnt2);
710
711 /* Test font weight */
712 fd.sWeight = 100;
713 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt2);
714 hres = IFont_IsEqual(ifnt,ifnt2);
715 ok(hres == S_FALSE,
716 "IFont_IsEqual: (Weight) Expected S_FALSE but got 0x%08x\n",hres);
717 fd.sWeight = 0;
718 IFont_Release(ifnt2);
719
720 /* Test charset */
721 fd.sCharset = 1;
722 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt2);
723 hres = IFont_IsEqual(ifnt,ifnt2);
724 ok(hres == S_FALSE,
725 "IFont_IsEqual: (Charset) Expected S_FALSE but got 0x%08x\n",hres);
726 fd.sCharset = 0;
727 IFont_Release(ifnt2);
728
729 /* Test italic setting */
730 fd.fItalic = TRUE;
731 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt2);
732 hres = IFont_IsEqual(ifnt,ifnt2);
733 ok(hres == S_FALSE,
734 "IFont_IsEqual: (Italic) Expected S_FALSE but got 0x%08x\n",hres);
735 fd.fItalic = FALSE;
736 IFont_Release(ifnt2);
737
738 /* Test underline setting */
739 fd.fUnderline = TRUE;
740 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt2);
741 hres = IFont_IsEqual(ifnt,ifnt2);
742 ok(hres == S_FALSE,
743 "IFont_IsEqual: (Underline) Expected S_FALSE but got 0x%08x\n",hres);
744 fd.fUnderline = FALSE;
745 IFont_Release(ifnt2);
746
747 /* Test strikethrough setting */
748 fd.fStrikethrough = TRUE;
749 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt2);
750 hres = IFont_IsEqual(ifnt,ifnt2);
751 ok(hres == S_FALSE,
752 "IFont_IsEqual: (Strikethrough) Expected S_FALSE but got 0x%08x\n",hres);
753 fd.fStrikethrough = FALSE;
754 IFont_Release(ifnt2);
755
756 /* Free IFont. */
757 IFont_Release(ifnt);
758 }
759
760 static void test_ReleaseHfont(void)
761 {
762 FONTDESC fd;
763 LPVOID pvObj1 = NULL;
764 LPVOID pvObj2 = NULL;
765 IFont* ifnt1 = NULL;
766 IFont* ifnt2 = NULL;
767 HFONT hfnt1 = 0;
768 HFONT hfnt2 = 0;
769 HRESULT hres;
770
771 /* Basic font description */
772 fd.cbSizeofstruct = sizeof(FONTDESC);
773 fd.lpstrName = system_font;
774 S(fd.cySize).Lo = 100;
775 S(fd.cySize).Hi = 100;
776 fd.sWeight = 0;
777 fd.sCharset = 0;
778 fd.fItalic = FALSE;
779 fd.fUnderline = FALSE;
780 fd.fStrikethrough = FALSE;
781
782 /* Create HFONTs and IFONTs */
783 pOleCreateFontIndirect(&fd, &IID_IFont, &pvObj1);
784 ifnt1 = pvObj1;
785 IFont_get_hFont(ifnt1,&hfnt1);
786 fd.lpstrName = arial_font;
787 pOleCreateFontIndirect(&fd, &IID_IFont, &pvObj2);
788 ifnt2 = pvObj2;
789 IFont_get_hFont(ifnt2,&hfnt2);
790
791 /* Try invalid HFONT */
792 hres = IFont_ReleaseHfont(ifnt1,NULL);
793 ok(hres == E_INVALIDARG,
794 "IFont_ReleaseHfont: (Bad HFONT) Expected E_INVALIDARG but got 0x%08x\n",
795 hres);
796
797 /* Try to add a bad HFONT */
798 hres = IFont_ReleaseHfont(ifnt1,(HFONT)32);
799 ok(hres == S_FALSE,
800 "IFont_ReleaseHfont: (Bad HFONT) Expected S_FALSE but got 0x%08x\n",
801 hres);
802
803 /* Release all refs */
804 hres = IFont_ReleaseHfont(ifnt1,hfnt1);
805 ok(hres == S_OK,
806 "IFont_AddRefHfont: (Release ref) Expected S_OK but got 0x%08x\n",
807 hres);
808
809 hres = IFont_ReleaseHfont(ifnt2,hfnt2);
810 ok(hres == S_OK,
811 "IFont_AddRefHfont: (Release ref) Expected S_OK but got 0x%08x\n",
812 hres);
813
814 /* Check that both lists are empty */
815 hres = IFont_ReleaseHfont(ifnt1,hfnt1);
816 ok(hres == S_FALSE,
817 "IFont_AddRefHfont: (Release ref) Expected S_FALSE but got 0x%08x\n",
818 hres);
819
820 /* The list should be empty */
821 hres = IFont_ReleaseHfont(ifnt2,hfnt2);
822 ok(hres == S_FALSE,
823 "IFont_AddRefHfont: (Release ref) Expected S_FALSE but got 0x%08x\n",
824 hres);
825
826 IFont_Release(ifnt1);
827 IFont_Release(ifnt2);
828 }
829
830 static void test_AddRefHfont(void)
831 {
832 FONTDESC fd;
833 IFont* ifnt1 = NULL;
834 IFont* ifnt2 = NULL;
835 IFont* ifnt3 = NULL;
836 HFONT hfnt1 = 0;
837 HFONT hfnt2 = 0;
838 HFONT hfnt3 = 0;
839 HRESULT hres;
840
841 /* Basic font description */
842 fd.cbSizeofstruct = sizeof(FONTDESC);
843 fd.lpstrName = system_font;
844 S(fd.cySize).Lo = 100;
845 S(fd.cySize).Hi = 100;
846 fd.sWeight = 0;
847 fd.sCharset = 0;
848 fd.fItalic = FALSE;
849 fd.fUnderline = FALSE;
850 fd.fStrikethrough = FALSE;
851
852 /* Create HFONTs and IFONTs */
853 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt1);
854 IFont_get_hFont(ifnt1,&hfnt1);
855 fd.lpstrName = arial_font;
856 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt2);
857 IFont_get_hFont(ifnt2,&hfnt2);
858
859 /* Try invalid HFONT */
860 hres = IFont_AddRefHfont(ifnt1,NULL);
861 ok(hres == E_INVALIDARG,
862 "IFont_AddRefHfont: (Bad HFONT) Expected E_INVALIDARG but got 0x%08x\n",
863 hres);
864
865 /* Try to add a bad HFONT */
866 hres = IFont_AddRefHfont(ifnt1,(HFONT)32);
867 ok(hres == S_FALSE,
868 "IFont_AddRefHfont: (Bad HFONT) Expected S_FALSE but got 0x%08x\n",
869 hres);
870
871 /* Add simple IFONT HFONT pair */
872 hres = IFont_AddRefHfont(ifnt1,hfnt1);
873 ok(hres == S_OK,
874 "IFont_AddRefHfont: (Add ref) Expected S_OK but got 0x%08x\n",
875 hres);
876
877 /* IFONT and HFONT do not have to be the same (always looks at HFONT) */
878 hres = IFont_AddRefHfont(ifnt2,hfnt1);
879 ok(hres == S_OK,
880 "IFont_AddRefHfont: (Add ref) Expected S_OK but got 0x%08x\n",
881 hres);
882
883 /* Release all hfnt1 refs */
884 hres = IFont_ReleaseHfont(ifnt1,hfnt1);
885 ok(hres == S_OK,
886 "IFont_AddRefHfont: (Release ref) Expected S_OK but got 0x%08x\n",
887 hres);
888
889 hres = IFont_ReleaseHfont(ifnt1,hfnt1);
890 ok(hres == S_OK,
891 "IFont_AddRefHfont: (Release ref) Expected S_OK but got 0x%08x\n",
892 hres);
893
894 hres = IFont_ReleaseHfont(ifnt1,hfnt1);
895 ok(hres == S_OK,
896 "IFont_AddRefHfont: (Release ref) Expected S_OK but got 0x%08x\n",
897 hres);
898
899 /* Check if hfnt1 is empty */
900 hres = IFont_ReleaseHfont(ifnt1,hfnt1);
901 ok(hres == S_FALSE,
902 "IFont_AddRefHfont: (Release ref) Expected S_FALSE but got 0x%08x\n",
903 hres);
904
905 /* Release all hfnt2 refs */
906 hres = IFont_ReleaseHfont(ifnt2,hfnt2);
907 ok(hres == S_OK,
908 "IFont_AddRefHfont: (Release ref) Expected S_OK but got 0x%08x\n",
909 hres);
910
911 /* Check if hfnt2 is empty */
912 hres = IFont_ReleaseHfont(ifnt2,hfnt2);
913 ok(hres == S_FALSE,
914 "IFont_AddRefHfont: (Release ref) Expected S_FALSE but got 0x%08x\n",
915 hres);
916
917 /* Show that releasing an IFONT does not always release it from the HFONT cache. */
918
919 IFont_Release(ifnt1);
920
921 /* Add a reference for destroyed hfnt1 */
922 hres = IFont_AddRefHfont(ifnt2,hfnt1);
923 ok(hres == S_OK,
924 "IFont_AddRefHfont: (Add ref) Expected S_OK but got 0x%08x\n",
925 hres);
926
927 /* Decrement reference for destroyed hfnt1 */
928 hres = IFont_ReleaseHfont(ifnt2,hfnt1);
929 ok(hres == S_OK ||
930 hres == S_FALSE, /* <= win2k */
931 "IFont_AddRefHfont: (Release ref) Expected S_OK or S_FALSE but got 0x%08x\n",
932 hres);
933
934 /* Shows that releasing all IFONT's does clear the HFONT cache. */
935
936 IFont_Release(ifnt2);
937
938 /* Need to make a new IFONT for testing */
939 fd.fUnderline = TRUE;
940 pOleCreateFontIndirect(&fd, &IID_IFont, (void **)&ifnt3);
941 IFont_get_hFont(ifnt3,&hfnt3);
942
943 /* Add a reference for destroyed hfnt1 */
944 hres = IFont_AddRefHfont(ifnt3,hfnt1);
945 ok(hres == S_FALSE,
946 "IFont_AddRefHfont: (Add ref) Expected S_OK but got 0x%08x\n",
947 hres);
948
949 /* Decrement reference for destroyed hfnt1 */
950 hres = IFont_ReleaseHfont(ifnt3,hfnt1);
951 ok(hres == S_FALSE,
952 "IFont_AddRefHfont: (Release ref) Expected S_OK but got 0x%08x\n",
953 hres);
954
955 IFont_Release(ifnt3);
956 }
957
958 static void test_returns(void)
959 {
960 IFont *pFont;
961 FONTDESC fontdesc;
962 HRESULT hr;
963
964 fontdesc.cbSizeofstruct = sizeof(fontdesc);
965 fontdesc.lpstrName = MSSansSerif_font;
966 fontdesc.cySize.int64 = 12 * 10000; /* 12 pt */
967 fontdesc.sWeight = FW_NORMAL;
968 fontdesc.sCharset = 0;
969 fontdesc.fItalic = FALSE;
970 fontdesc.fUnderline = FALSE;
971 fontdesc.fStrikethrough = FALSE;
972
973 hr = pOleCreateFontIndirect(&fontdesc, &IID_IFont, (void **)&pFont);
974 EXPECT_HR(hr, S_OK);
975
976 hr = IFont_put_Name(pFont, NULL);
977 EXPECT_HR(hr, CTL_E_INVALIDPROPERTYVALUE);
978
979 hr = IFont_get_Name(pFont, NULL);
980 EXPECT_HR(hr, E_POINTER);
981
982 hr = IFont_get_Size(pFont, NULL);
983 EXPECT_HR(hr, E_POINTER);
984
985 hr = IFont_get_Bold(pFont, NULL);
986 EXPECT_HR(hr, E_POINTER);
987
988 IFont_Release(pFont);
989 }
990
991 static void test_hfont_lifetime(void)
992 {
993 IFont *font, *font2;
994 FONTDESC fontdesc;
995 HRESULT hr;
996 HFONT hfont, first_hfont = NULL;
997 CY size;
998 DWORD obj_type;
999 int i;
1000
1001 fontdesc.cbSizeofstruct = sizeof(fontdesc);
1002 fontdesc.lpstrName = arial_font;
1003 fontdesc.cySize.int64 = 12 * 10000; /* 12 pt */
1004 fontdesc.sWeight = FW_NORMAL;
1005 fontdesc.sCharset = ANSI_CHARSET;
1006 fontdesc.fItalic = FALSE;
1007 fontdesc.fUnderline = FALSE;
1008 fontdesc.fStrikethrough = FALSE;
1009
1010 hr = pOleCreateFontIndirect(&fontdesc, &IID_IFont, (void **)&font);
1011 EXPECT_HR(hr, S_OK);
1012
1013 hr = IFont_get_hFont(font, &hfont);
1014 EXPECT_HR(hr, S_OK);
1015
1016 /* show that if the font is updated the old hfont is deleted when the
1017 new font is realized */
1018 for(i = 0; i < 100; i++)
1019 {
1020 HFONT last_hfont = hfont;
1021
1022 size.int64 = (i + 10) * 20000;
1023
1024 obj_type = GetObjectType(hfont);
1025 ok(obj_type == OBJ_FONT, "got obj type %d\n", obj_type);
1026
1027 hr = IFont_put_Size(font, size);
1028 EXPECT_HR(hr, S_OK);
1029
1030 /* put_Size doesn't cause the new font to be realized */
1031 obj_type = GetObjectType(last_hfont);
1032 ok(obj_type == OBJ_FONT, "got obj type %d\n", obj_type);
1033
1034 hr = IFont_get_hFont(font, &hfont);
1035 EXPECT_HR(hr, S_OK);
1036
1037 obj_type = GetObjectType(last_hfont);
1038 ok(obj_type == 0, "%d: got obj type %d\n", i, obj_type);
1039 }
1040
1041 /* now show that if we take a reference on the hfont, it persists
1042 until the font object is released */
1043 for(i = 0; i < 100; i++)
1044 {
1045 size.int64 = (i + 10) * 20000;
1046
1047 obj_type = GetObjectType(hfont);
1048 ok(obj_type == OBJ_FONT, "got obj type %d\n", obj_type);
1049
1050 hr = IFont_put_Size(font, size);
1051 EXPECT_HR(hr, S_OK);
1052
1053 hr = IFont_get_hFont(font, &hfont);
1054 EXPECT_HR(hr, S_OK);
1055
1056 hr = IFont_AddRefHfont(font, hfont);
1057 EXPECT_HR(hr, S_OK);
1058
1059 if(i == 0) first_hfont = hfont;
1060 obj_type = GetObjectType(first_hfont);
1061 ok(obj_type == OBJ_FONT, "got obj type %d\n", obj_type);
1062 }
1063
1064 IFont_Release(font);
1065
1066 obj_type = GetObjectType(first_hfont);
1067 ok(obj_type == 0, "got obj type %d\n", obj_type);
1068
1069 /* An AddRefHfont followed by a ReleaseHfont means the font doesn't not persist
1070 through re-realization */
1071
1072 hr = pOleCreateFontIndirect(&fontdesc, &IID_IFont, (void **)&font);
1073 EXPECT_HR(hr, S_OK);
1074
1075 hr = IFont_get_hFont(font, &hfont);
1076 EXPECT_HR(hr, S_OK);
1077
1078 for(i = 0; i < 100; i++)
1079 {
1080 HFONT last_hfont = hfont;
1081
1082 size.int64 = (i + 10) * 20000;
1083
1084 obj_type = GetObjectType(hfont);
1085 ok(obj_type == OBJ_FONT, "got obj type %d\n", obj_type);
1086
1087 hr = IFont_put_Size(font, size);
1088 EXPECT_HR(hr, S_OK);
1089
1090 /* put_Size doesn't cause the new font to be realized */
1091 obj_type = GetObjectType(last_hfont);
1092 ok(obj_type == OBJ_FONT, "got obj type %d\n", obj_type);
1093
1094 hr = IFont_get_hFont(font, &hfont);
1095 EXPECT_HR(hr, S_OK);
1096
1097 hr = IFont_AddRefHfont(font, hfont);
1098 EXPECT_HR(hr, S_OK);
1099
1100 hr = IFont_ReleaseHfont(font, hfont);
1101 EXPECT_HR(hr, S_OK);
1102
1103 obj_type = GetObjectType(last_hfont);
1104 ok(obj_type == 0, "%d: got obj type %d\n", i, obj_type);
1105 }
1106
1107 /* Interestingly if we release a nonexistent reference on the hfont,
1108 * it persists until the font object is released
1109 */
1110 for(i = 0; i < 100; i++)
1111 {
1112 size.int64 = (i + 10) * 20000;
1113
1114 obj_type = GetObjectType(hfont);
1115 ok(obj_type == OBJ_FONT, "got obj type %d\n", obj_type);
1116
1117 hr = IFont_put_Size(font, size);
1118 EXPECT_HR(hr, S_OK);
1119
1120 hr = IFont_get_hFont(font, &hfont);
1121 EXPECT_HR(hr, S_OK);
1122
1123 hr = IFont_ReleaseHfont(font, hfont);
1124 EXPECT_HR(hr, S_OK);
1125
1126 if(i == 0) first_hfont = hfont;
1127 obj_type = GetObjectType(first_hfont);
1128 ok(obj_type == OBJ_FONT, "got obj type %d\n", obj_type);
1129 }
1130
1131 IFont_Release(font);
1132
1133 obj_type = GetObjectType(first_hfont);
1134 ok(obj_type == 0, "got obj type %d\n", obj_type);
1135
1136 /* If we take two internal references on a hfont then we can release
1137 it twice. So it looks like there's a total reference count
1138 that includes internal and external references */
1139
1140 hr = pOleCreateFontIndirect(&fontdesc, &IID_IFont, (void **)&font);
1141 EXPECT_HR(hr, S_OK);
1142 hr = pOleCreateFontIndirect(&fontdesc, &IID_IFont, (void **)&font2);
1143 EXPECT_HR(hr, S_OK);
1144
1145 hr = IFont_get_hFont(font, &hfont);
1146 EXPECT_HR(hr, S_OK);
1147 hr = IFont_get_hFont(font2, &first_hfont);
1148 EXPECT_HR(hr, S_OK);
1149 todo_wine
1150 ok(hfont == first_hfont, "fonts differ\n");
1151 hr = IFont_ReleaseHfont(font, hfont);
1152 EXPECT_HR(hr, S_OK);
1153 hr = IFont_ReleaseHfont(font, hfont);
1154 todo_wine
1155 EXPECT_HR(hr, S_OK);
1156 hr = IFont_ReleaseHfont(font, hfont);
1157 EXPECT_HR(hr, S_FALSE);
1158
1159 obj_type = GetObjectType(hfont);
1160 ok(obj_type == OBJ_FONT, "got obj type %d\n", obj_type);
1161
1162 IFont_Release(font);
1163
1164 obj_type = GetObjectType(hfont);
1165 ok(obj_type == OBJ_FONT, "got obj type %d\n", obj_type);
1166
1167 IFont_Release(font2);
1168
1169 obj_type = GetObjectType(hfont);
1170 ok(obj_type == 0, "got obj type %d\n", obj_type);
1171 }
1172
1173 static void test_realization(void)
1174 {
1175 IFont *font;
1176 FONTDESC fontdesc;
1177 HRESULT hr;
1178 BSTR name;
1179 SHORT cs;
1180
1181 /* Try to create a symbol only font (marlett) with charset
1182 set to ANSI. This will result in another, ANSI, font
1183 being selected */
1184 fontdesc.cbSizeofstruct = sizeof(fontdesc);
1185 fontdesc.lpstrName = marlett_font;
1186 fontdesc.cySize.int64 = 12 * 10000; /* 12 pt */
1187 fontdesc.sWeight = FW_NORMAL;
1188 fontdesc.sCharset = ANSI_CHARSET;
1189 fontdesc.fItalic = FALSE;
1190 fontdesc.fUnderline = FALSE;
1191 fontdesc.fStrikethrough = FALSE;
1192
1193 hr = pOleCreateFontIndirect(&fontdesc, &IID_IFont, (void **)&font);
1194 EXPECT_HR(hr, S_OK);
1195
1196 hr = IFont_get_Charset(font, &cs);
1197 EXPECT_HR(hr, S_OK);
1198 ok(cs == ANSI_CHARSET, "got charset %d\n", cs);
1199
1200 IFont_Release(font);
1201
1202 /* Now create an ANSI font and change the name to marlett */
1203
1204 fontdesc.lpstrName = arial_font;
1205
1206 hr = pOleCreateFontIndirect(&fontdesc, &IID_IFont, (void **)&font);
1207 EXPECT_HR(hr, S_OK);
1208
1209 hr = IFont_get_Charset(font, &cs);
1210 EXPECT_HR(hr, S_OK);
1211 ok(cs == ANSI_CHARSET, "got charset %d\n", cs);
1212
1213 name = SysAllocString(marlett_font);
1214 hr = IFont_put_Name(font, name);
1215 EXPECT_HR(hr, S_OK);
1216 SysFreeString(name);
1217
1218 hr = IFont_get_Name(font, &name);
1219 EXPECT_HR(hr, S_OK);
1220 ok(!lstrcmpiW(name, marlett_font), "got name %s\n", wine_dbgstr_w(name));
1221 SysFreeString(name);
1222
1223 hr = IFont_get_Charset(font, &cs);
1224 EXPECT_HR(hr, S_OK);
1225 ok(cs == SYMBOL_CHARSET, "got charset %d\n", cs);
1226
1227 IFont_Release(font);
1228 }
1229
1230 static void test_OleCreateFontIndirect(void)
1231 {
1232 FONTDESC fontdesc;
1233 IUnknown *unk, *unk2;
1234 IFont *font;
1235 HRESULT hr;
1236
1237 fontdesc.cbSizeofstruct = sizeof(fontdesc);
1238 fontdesc.lpstrName = arial_font;
1239 fontdesc.cySize.int64 = 12 * 10000; /* 12 pt */
1240 fontdesc.sWeight = FW_NORMAL;
1241 fontdesc.sCharset = ANSI_CHARSET;
1242 fontdesc.fItalic = FALSE;
1243 fontdesc.fUnderline = FALSE;
1244 fontdesc.fStrikethrough = FALSE;
1245
1246 hr = pOleCreateFontIndirect(&fontdesc, &IID_IFont, (void**)&font);
1247 EXPECT_HR(hr, S_OK);
1248 IFont_Release(font);
1249
1250 /* play with cbSizeofstruct value */
1251 fontdesc.cbSizeofstruct = sizeof(fontdesc)-1;
1252 hr = pOleCreateFontIndirect(&fontdesc, &IID_IFont, (void**)&font);
1253 EXPECT_HR(hr, S_OK);
1254 IFont_Release(font);
1255
1256 fontdesc.cbSizeofstruct = sizeof(fontdesc)+1;
1257 hr = pOleCreateFontIndirect(&fontdesc, &IID_IFont, (void**)&font);
1258 EXPECT_HR(hr, S_OK);
1259 IFont_Release(font);
1260
1261 fontdesc.cbSizeofstruct = 0;
1262 hr = pOleCreateFontIndirect(&fontdesc, &IID_IFont, (void**)&font);
1263 EXPECT_HR(hr, S_OK);
1264 IFont_Release(font);
1265
1266 hr = OleInitialize(NULL);
1267 ok(hr == S_OK, "got 0x%08x\n", hr);
1268
1269 hr = CoGetClassObject(&CLSID_StdFont, CLSCTX_INPROC_SERVER, NULL, &IID_IClassFactory, (void**)&unk);
1270 ok(hr == S_OK, "got 0x%08x\n", hr);
1271
1272 hr = IUnknown_QueryInterface(unk, &IID_IUnknown, (void**)&unk2);
1273 ok(hr == S_OK, "got 0x%08x\n", hr);
1274
1275 IUnknown_Release(unk);
1276 IUnknown_Release(unk2);
1277
1278 OleUninitialize();
1279 }
1280
1281 START_TEST(olefont)
1282 {
1283 hOleaut32 = GetModuleHandleA("oleaut32.dll");
1284 pOleCreateFontIndirect = (void*)GetProcAddress(hOleaut32, "OleCreateFontIndirect");
1285 if (!pOleCreateFontIndirect)
1286 {
1287 win_skip("OleCreateFontIndirect not available\n");
1288 return;
1289 }
1290
1291 test_QueryInterface();
1292 test_type_info();
1293 test_ifont_sizes();
1294 test_font_events_disp();
1295 test_GetIDsOfNames();
1296 test_Invoke();
1297 test_IsEqual();
1298 test_ReleaseHfont();
1299 test_AddRefHfont();
1300 test_returns();
1301 test_hfont_lifetime();
1302 test_realization();
1303 test_OleCreateFontIndirect();
1304 }