4 * Copyright 2012 Nikolay Sivov for CodeWeavers
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
32 #include "dwrite_private.h"
33 #include "wine/debug.h"
35 WINE_DEFAULT_DEBUG_CHANNEL(dwrite
);
37 static IDWriteFactory5
*shared_factory
;
38 static void release_shared_factory(IDWriteFactory5
*);
40 BOOL WINAPI
DllMain(HINSTANCE hinstDLL
, DWORD reason
, LPVOID reserved
)
44 case DLL_PROCESS_ATTACH
:
45 DisableThreadLibraryCalls( hinstDLL
);
47 init_local_fontfile_loader();
49 case DLL_PROCESS_DETACH
:
51 release_shared_factory(shared_factory
);
57 struct renderingparams
{
58 IDWriteRenderingParams3 IDWriteRenderingParams3_iface
;
63 FLOAT grayscalecontrast
;
64 FLOAT cleartype_level
;
65 DWRITE_PIXEL_GEOMETRY geometry
;
66 DWRITE_RENDERING_MODE1 mode
;
67 DWRITE_GRID_FIT_MODE gridfit
;
70 static inline struct renderingparams
*impl_from_IDWriteRenderingParams3(IDWriteRenderingParams3
*iface
)
72 return CONTAINING_RECORD(iface
, struct renderingparams
, IDWriteRenderingParams3_iface
);
75 static HRESULT WINAPI
renderingparams_QueryInterface(IDWriteRenderingParams3
*iface
, REFIID riid
, void **obj
)
77 struct renderingparams
*This
= impl_from_IDWriteRenderingParams3(iface
);
79 TRACE("(%p)->(%s %p)\n", This
, debugstr_guid(riid
), obj
);
81 if (IsEqualIID(riid
, &IID_IDWriteRenderingParams3
) ||
82 IsEqualIID(riid
, &IID_IDWriteRenderingParams2
) ||
83 IsEqualIID(riid
, &IID_IDWriteRenderingParams1
) ||
84 IsEqualIID(riid
, &IID_IDWriteRenderingParams
) ||
85 IsEqualIID(riid
, &IID_IUnknown
))
88 IDWriteRenderingParams3_AddRef(iface
);
97 static ULONG WINAPI
renderingparams_AddRef(IDWriteRenderingParams3
*iface
)
99 struct renderingparams
*This
= impl_from_IDWriteRenderingParams3(iface
);
100 ULONG ref
= InterlockedIncrement(&This
->ref
);
101 TRACE("(%p)->(%d)\n", This
, ref
);
105 static ULONG WINAPI
renderingparams_Release(IDWriteRenderingParams3
*iface
)
107 struct renderingparams
*This
= impl_from_IDWriteRenderingParams3(iface
);
108 ULONG ref
= InterlockedDecrement(&This
->ref
);
110 TRACE("(%p)->(%d)\n", This
, ref
);
118 static FLOAT WINAPI
renderingparams_GetGamma(IDWriteRenderingParams3
*iface
)
120 struct renderingparams
*This
= impl_from_IDWriteRenderingParams3(iface
);
121 TRACE("(%p)\n", This
);
125 static FLOAT WINAPI
renderingparams_GetEnhancedContrast(IDWriteRenderingParams3
*iface
)
127 struct renderingparams
*This
= impl_from_IDWriteRenderingParams3(iface
);
128 TRACE("(%p)\n", This
);
129 return This
->contrast
;
132 static FLOAT WINAPI
renderingparams_GetClearTypeLevel(IDWriteRenderingParams3
*iface
)
134 struct renderingparams
*This
= impl_from_IDWriteRenderingParams3(iface
);
135 TRACE("(%p)\n", This
);
136 return This
->cleartype_level
;
139 static DWRITE_PIXEL_GEOMETRY WINAPI
renderingparams_GetPixelGeometry(IDWriteRenderingParams3
*iface
)
141 struct renderingparams
*This
= impl_from_IDWriteRenderingParams3(iface
);
142 TRACE("(%p)\n", This
);
143 return This
->geometry
;
146 static DWRITE_RENDERING_MODE
rendering_mode_from_mode1(DWRITE_RENDERING_MODE1 mode
)
148 static const DWRITE_RENDERING_MODE rendering_modes
[] = {
149 DWRITE_RENDERING_MODE_DEFAULT
, /* DWRITE_RENDERING_MODE1_DEFAULT */
150 DWRITE_RENDERING_MODE_ALIASED
, /* DWRITE_RENDERING_MODE1_ALIASED */
151 DWRITE_RENDERING_MODE_GDI_CLASSIC
, /* DWRITE_RENDERING_MODE1_GDI_CLASSIC */
152 DWRITE_RENDERING_MODE_GDI_NATURAL
, /* DWRITE_RENDERING_MODE1_GDI_NATURAL */
153 DWRITE_RENDERING_MODE_NATURAL
, /* DWRITE_RENDERING_MODE1_NATURAL */
154 DWRITE_RENDERING_MODE_NATURAL_SYMMETRIC
, /* DWRITE_RENDERING_MODE1_NATURAL_SYMMETRIC */
155 DWRITE_RENDERING_MODE_OUTLINE
, /* DWRITE_RENDERING_MODE1_OUTLINE */
156 DWRITE_RENDERING_MODE_NATURAL_SYMMETRIC
, /* DWRITE_RENDERING_MODE1_NATURAL_SYMMETRIC_DOWNSAMPLED */
159 return rendering_modes
[mode
];
162 static DWRITE_RENDERING_MODE WINAPI
renderingparams_GetRenderingMode(IDWriteRenderingParams3
*iface
)
164 struct renderingparams
*This
= impl_from_IDWriteRenderingParams3(iface
);
166 TRACE("(%p)\n", This
);
168 return rendering_mode_from_mode1(This
->mode
);
171 static FLOAT WINAPI
renderingparams1_GetGrayscaleEnhancedContrast(IDWriteRenderingParams3
*iface
)
173 struct renderingparams
*This
= impl_from_IDWriteRenderingParams3(iface
);
174 TRACE("(%p)\n", This
);
175 return This
->grayscalecontrast
;
178 static DWRITE_GRID_FIT_MODE WINAPI
renderingparams2_GetGridFitMode(IDWriteRenderingParams3
*iface
)
180 struct renderingparams
*This
= impl_from_IDWriteRenderingParams3(iface
);
181 TRACE("(%p)\n", This
);
182 return This
->gridfit
;
185 static DWRITE_RENDERING_MODE1 WINAPI
renderingparams3_GetRenderingMode1(IDWriteRenderingParams3
*iface
)
187 struct renderingparams
*This
= impl_from_IDWriteRenderingParams3(iface
);
188 TRACE("(%p)\n", This
);
192 static const struct IDWriteRenderingParams3Vtbl renderingparamsvtbl
= {
193 renderingparams_QueryInterface
,
194 renderingparams_AddRef
,
195 renderingparams_Release
,
196 renderingparams_GetGamma
,
197 renderingparams_GetEnhancedContrast
,
198 renderingparams_GetClearTypeLevel
,
199 renderingparams_GetPixelGeometry
,
200 renderingparams_GetRenderingMode
,
201 renderingparams1_GetGrayscaleEnhancedContrast
,
202 renderingparams2_GetGridFitMode
,
203 renderingparams3_GetRenderingMode1
206 static HRESULT
create_renderingparams(FLOAT gamma
, FLOAT contrast
, FLOAT grayscalecontrast
, FLOAT cleartype_level
,
207 DWRITE_PIXEL_GEOMETRY geometry
, DWRITE_RENDERING_MODE1 mode
, DWRITE_GRID_FIT_MODE gridfit
,
208 IDWriteRenderingParams3
**params
)
210 struct renderingparams
*This
;
214 if (gamma
<= 0.0f
|| contrast
< 0.0f
|| grayscalecontrast
< 0.0f
|| cleartype_level
< 0.0f
)
217 if ((UINT32
)gridfit
> DWRITE_GRID_FIT_MODE_ENABLED
|| (UINT32
)geometry
> DWRITE_PIXEL_GEOMETRY_BGR
)
220 This
= heap_alloc(sizeof(struct renderingparams
));
221 if (!This
) return E_OUTOFMEMORY
;
223 This
->IDWriteRenderingParams3_iface
.lpVtbl
= &renderingparamsvtbl
;
227 This
->contrast
= contrast
;
228 This
->grayscalecontrast
= grayscalecontrast
;
229 This
->cleartype_level
= cleartype_level
;
230 This
->geometry
= geometry
;
232 This
->gridfit
= gridfit
;
234 *params
= &This
->IDWriteRenderingParams3_iface
;
239 struct localizedpair
{
244 struct localizedstrings
{
245 IDWriteLocalizedStrings IDWriteLocalizedStrings_iface
;
248 struct localizedpair
*data
;
253 static inline struct localizedstrings
*impl_from_IDWriteLocalizedStrings(IDWriteLocalizedStrings
*iface
)
255 return CONTAINING_RECORD(iface
, struct localizedstrings
, IDWriteLocalizedStrings_iface
);
258 static HRESULT WINAPI
localizedstrings_QueryInterface(IDWriteLocalizedStrings
*iface
, REFIID riid
, void **obj
)
260 struct localizedstrings
*This
= impl_from_IDWriteLocalizedStrings(iface
);
262 TRACE("(%p)->(%s %p)\n", This
, debugstr_guid(riid
), obj
);
264 if (IsEqualIID(riid
, &IID_IUnknown
) || IsEqualIID(riid
, &IID_IDWriteLocalizedStrings
))
267 IDWriteLocalizedStrings_AddRef(iface
);
273 return E_NOINTERFACE
;
276 static ULONG WINAPI
localizedstrings_AddRef(IDWriteLocalizedStrings
*iface
)
278 struct localizedstrings
*This
= impl_from_IDWriteLocalizedStrings(iface
);
279 ULONG ref
= InterlockedIncrement(&This
->ref
);
280 TRACE("(%p)->(%d)\n", This
, ref
);
284 static ULONG WINAPI
localizedstrings_Release(IDWriteLocalizedStrings
*iface
)
286 struct localizedstrings
*This
= impl_from_IDWriteLocalizedStrings(iface
);
287 ULONG ref
= InterlockedDecrement(&This
->ref
);
289 TRACE("(%p)->(%d)\n", This
, ref
);
294 for (i
= 0; i
< This
->count
; i
++) {
295 heap_free(This
->data
[i
].locale
);
296 heap_free(This
->data
[i
].string
);
299 heap_free(This
->data
);
306 static UINT32 WINAPI
localizedstrings_GetCount(IDWriteLocalizedStrings
*iface
)
308 struct localizedstrings
*This
= impl_from_IDWriteLocalizedStrings(iface
);
309 TRACE("(%p)\n", This
);
313 static HRESULT WINAPI
localizedstrings_FindLocaleName(IDWriteLocalizedStrings
*iface
,
314 WCHAR
const *locale_name
, UINT32
*index
, BOOL
*exists
)
316 struct localizedstrings
*This
= impl_from_IDWriteLocalizedStrings(iface
);
319 TRACE("(%p)->(%s %p %p)\n", This
, debugstr_w(locale_name
), index
, exists
);
324 for (i
= 0; i
< This
->count
; i
++) {
325 if (!strcmpiW(This
->data
[i
].locale
, locale_name
)) {
335 static HRESULT WINAPI
localizedstrings_GetLocaleNameLength(IDWriteLocalizedStrings
*iface
, UINT32 index
, UINT32
*length
)
337 struct localizedstrings
*This
= impl_from_IDWriteLocalizedStrings(iface
);
339 TRACE("(%p)->(%u %p)\n", This
, index
, length
);
341 if (index
>= This
->count
) {
342 *length
= (UINT32
)-1;
346 *length
= strlenW(This
->data
[index
].locale
);
350 static HRESULT WINAPI
localizedstrings_GetLocaleName(IDWriteLocalizedStrings
*iface
, UINT32 index
, WCHAR
*buffer
, UINT32 size
)
352 struct localizedstrings
*This
= impl_from_IDWriteLocalizedStrings(iface
);
354 TRACE("(%p)->(%u %p %u)\n", This
, index
, buffer
, size
);
356 if (index
>= This
->count
) {
357 if (buffer
) *buffer
= 0;
361 if (size
< strlenW(This
->data
[index
].locale
)+1) {
362 if (buffer
) *buffer
= 0;
363 return E_NOT_SUFFICIENT_BUFFER
;
366 strcpyW(buffer
, This
->data
[index
].locale
);
370 static HRESULT WINAPI
localizedstrings_GetStringLength(IDWriteLocalizedStrings
*iface
, UINT32 index
, UINT32
*length
)
372 struct localizedstrings
*This
= impl_from_IDWriteLocalizedStrings(iface
);
374 TRACE("(%p)->(%u %p)\n", This
, index
, length
);
376 if (index
>= This
->count
) {
377 *length
= (UINT32
)-1;
381 *length
= strlenW(This
->data
[index
].string
);
385 static HRESULT WINAPI
localizedstrings_GetString(IDWriteLocalizedStrings
*iface
, UINT32 index
, WCHAR
*buffer
, UINT32 size
)
387 struct localizedstrings
*This
= impl_from_IDWriteLocalizedStrings(iface
);
389 TRACE("(%p)->(%u %p %u)\n", This
, index
, buffer
, size
);
391 if (index
>= This
->count
) {
392 if (buffer
) *buffer
= 0;
396 if (size
< strlenW(This
->data
[index
].string
)+1) {
397 if (buffer
) *buffer
= 0;
398 return E_NOT_SUFFICIENT_BUFFER
;
401 strcpyW(buffer
, This
->data
[index
].string
);
405 static const IDWriteLocalizedStringsVtbl localizedstringsvtbl
= {
406 localizedstrings_QueryInterface
,
407 localizedstrings_AddRef
,
408 localizedstrings_Release
,
409 localizedstrings_GetCount
,
410 localizedstrings_FindLocaleName
,
411 localizedstrings_GetLocaleNameLength
,
412 localizedstrings_GetLocaleName
,
413 localizedstrings_GetStringLength
,
414 localizedstrings_GetString
417 HRESULT
create_localizedstrings(IDWriteLocalizedStrings
**strings
)
419 struct localizedstrings
*This
;
423 This
= heap_alloc(sizeof(struct localizedstrings
));
424 if (!This
) return E_OUTOFMEMORY
;
426 This
->IDWriteLocalizedStrings_iface
.lpVtbl
= &localizedstringsvtbl
;
429 This
->data
= heap_alloc_zero(sizeof(struct localizedpair
));
432 return E_OUTOFMEMORY
;
436 *strings
= &This
->IDWriteLocalizedStrings_iface
;
441 HRESULT
add_localizedstring(IDWriteLocalizedStrings
*iface
, const WCHAR
*locale
, const WCHAR
*string
)
443 struct localizedstrings
*This
= impl_from_IDWriteLocalizedStrings(iface
);
446 /* make sure there's no duplicates */
447 for (i
= 0; i
< This
->count
; i
++)
448 if (!strcmpW(This
->data
[i
].locale
, locale
))
451 if (This
->count
== This
->alloc
) {
454 ptr
= heap_realloc(This
->data
, 2*This
->alloc
*sizeof(struct localizedpair
));
456 return E_OUTOFMEMORY
;
462 This
->data
[This
->count
].locale
= heap_strdupW(locale
);
463 This
->data
[This
->count
].string
= heap_strdupW(string
);
464 if (!This
->data
[This
->count
].locale
|| !This
->data
[This
->count
].string
) {
465 heap_free(This
->data
[This
->count
].locale
);
466 heap_free(This
->data
[This
->count
].string
);
467 return E_OUTOFMEMORY
;
475 HRESULT
clone_localizedstring(IDWriteLocalizedStrings
*iface
, IDWriteLocalizedStrings
**ret
)
477 struct localizedstrings
*strings
, *strings_clone
;
485 strings
= impl_from_IDWriteLocalizedStrings(iface
);
486 strings_clone
= heap_alloc(sizeof(struct localizedstrings
));
487 if (!strings_clone
) return E_OUTOFMEMORY
;
489 strings_clone
->IDWriteLocalizedStrings_iface
.lpVtbl
= &localizedstringsvtbl
;
490 strings_clone
->ref
= 1;
491 strings_clone
->count
= strings
->count
;
492 strings_clone
->data
= heap_alloc(sizeof(struct localizedpair
) * strings_clone
->count
);
493 if (!strings_clone
->data
) {
494 heap_free(strings_clone
);
495 return E_OUTOFMEMORY
;
497 for (i
= 0; i
< strings_clone
->count
; i
++)
499 strings_clone
->data
[i
].locale
= heap_strdupW(strings
->data
[i
].locale
);
500 strings_clone
->data
[i
].string
= heap_strdupW(strings
->data
[i
].string
);
502 strings_clone
->alloc
= strings_clone
->count
;
504 *ret
= &strings_clone
->IDWriteLocalizedStrings_iface
;
509 void set_en_localizedstring(IDWriteLocalizedStrings
*iface
, const WCHAR
*string
)
511 static const WCHAR enusW
[] = {'e','n','-','U','S',0};
512 struct localizedstrings
*This
= impl_from_IDWriteLocalizedStrings(iface
);
515 for (i
= 0; i
< This
->count
; i
++) {
516 if (!strcmpiW(This
->data
[i
].locale
, enusW
)) {
517 heap_free(This
->data
[i
].string
);
518 This
->data
[i
].string
= heap_strdupW(string
);
524 struct collectionloader
527 IDWriteFontCollectionLoader
*loader
;
533 struct list fontfaces
;
534 IDWriteFontFileLoader
*loader
;
537 struct dwritefactory
{
538 IDWriteFactory5 IDWriteFactory5_iface
;
541 IDWriteFontCollection1
*system_collection
;
542 IDWriteFontCollection1
*eudc_collection
;
543 IDWriteGdiInterop1
*gdiinterop
;
544 IDWriteFontFallback
*fallback
;
546 IDWriteFontFileLoader
*localfontfileloader
;
547 struct list localfontfaces
;
549 struct list collection_loaders
;
550 struct list file_loaders
;
555 static inline struct dwritefactory
*impl_from_IDWriteFactory5(IDWriteFactory5
*iface
)
557 return CONTAINING_RECORD(iface
, struct dwritefactory
, IDWriteFactory5_iface
);
560 static void release_fontface_cache(struct list
*fontfaces
)
562 struct fontfacecached
*fontface
, *fontface2
;
564 LIST_FOR_EACH_ENTRY_SAFE(fontface
, fontface2
, fontfaces
, struct fontfacecached
, entry
) {
565 list_remove(&fontface
->entry
);
566 fontface_detach_from_cache(fontface
->fontface
);
571 static void release_fileloader(struct fileloader
*fileloader
)
573 list_remove(&fileloader
->entry
);
574 release_fontface_cache(&fileloader
->fontfaces
);
575 IDWriteFontFileLoader_Release(fileloader
->loader
);
576 heap_free(fileloader
);
579 static void release_dwritefactory(struct dwritefactory
*factory
)
581 struct fileloader
*fileloader
, *fileloader2
;
582 struct collectionloader
*loader
, *loader2
;
584 EnterCriticalSection(&factory
->cs
);
585 release_fontface_cache(&factory
->localfontfaces
);
586 LeaveCriticalSection(&factory
->cs
);
588 LIST_FOR_EACH_ENTRY_SAFE(loader
, loader2
, &factory
->collection_loaders
, struct collectionloader
, entry
) {
589 list_remove(&loader
->entry
);
590 IDWriteFontCollectionLoader_Release(loader
->loader
);
594 LIST_FOR_EACH_ENTRY_SAFE(fileloader
, fileloader2
, &factory
->file_loaders
, struct fileloader
, entry
)
595 release_fileloader(fileloader
);
597 if (factory
->system_collection
)
598 IDWriteFontCollection1_Release(factory
->system_collection
);
599 if (factory
->eudc_collection
)
600 IDWriteFontCollection1_Release(factory
->eudc_collection
);
601 if (factory
->fallback
)
602 release_system_fontfallback(factory
->fallback
);
604 factory
->cs
.DebugInfo
->Spare
[0] = 0;
605 DeleteCriticalSection(&factory
->cs
);
609 static void release_shared_factory(IDWriteFactory5
*iface
)
611 struct dwritefactory
*factory
;
613 factory
= impl_from_IDWriteFactory5(iface
);
614 release_dwritefactory(factory
);
617 static struct fileloader
*factory_get_file_loader(struct dwritefactory
*factory
, IDWriteFontFileLoader
*loader
)
619 struct fileloader
*entry
, *found
= NULL
;
621 LIST_FOR_EACH_ENTRY(entry
, &factory
->file_loaders
, struct fileloader
, entry
) {
622 if (entry
->loader
== loader
) {
631 static struct collectionloader
*factory_get_collection_loader(struct dwritefactory
*factory
,
632 IDWriteFontCollectionLoader
*loader
)
634 struct collectionloader
*entry
, *found
= NULL
;
636 LIST_FOR_EACH_ENTRY(entry
, &factory
->collection_loaders
, struct collectionloader
, entry
) {
637 if (entry
->loader
== loader
) {
646 static IDWriteFontCollection1
*factory_get_system_collection(struct dwritefactory
*factory
)
648 IDWriteFontCollection1
*collection
;
651 if (factory
->system_collection
) {
652 IDWriteFontCollection1_AddRef(factory
->system_collection
);
653 return factory
->system_collection
;
656 if (FAILED(hr
= get_system_fontcollection(&factory
->IDWriteFactory5_iface
, &collection
))) {
657 WARN("Failed to create system font collection, hr %#x.\n", hr
);
661 if (InterlockedCompareExchangePointer((void **)&factory
->system_collection
, collection
, NULL
))
662 IDWriteFontCollection1_Release(collection
);
664 return factory
->system_collection
;
667 static HRESULT WINAPI
dwritefactory_QueryInterface(IDWriteFactory5
*iface
, REFIID riid
, void **obj
)
669 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
671 TRACE("(%p)->(%s %p)\n", This
, debugstr_guid(riid
), obj
);
673 if (IsEqualIID(riid
, &IID_IDWriteFactory5
) ||
674 IsEqualIID(riid
, &IID_IDWriteFactory4
) ||
675 IsEqualIID(riid
, &IID_IDWriteFactory3
) ||
676 IsEqualIID(riid
, &IID_IDWriteFactory2
) ||
677 IsEqualIID(riid
, &IID_IDWriteFactory1
) ||
678 IsEqualIID(riid
, &IID_IDWriteFactory
) ||
679 IsEqualIID(riid
, &IID_IUnknown
))
682 IDWriteFactory5_AddRef(iface
);
688 return E_NOINTERFACE
;
691 static ULONG WINAPI
dwritefactory_AddRef(IDWriteFactory5
*iface
)
693 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
694 ULONG ref
= InterlockedIncrement(&This
->ref
);
695 TRACE("(%p)->(%d)\n", This
, ref
);
699 static ULONG WINAPI
dwritefactory_Release(IDWriteFactory5
*iface
)
701 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
702 ULONG ref
= InterlockedDecrement(&This
->ref
);
704 TRACE("(%p)->(%d)\n", This
, ref
);
707 release_dwritefactory(This
);
712 static HRESULT WINAPI
dwritefactory_GetSystemFontCollection(IDWriteFactory5
*iface
,
713 IDWriteFontCollection
**collection
, BOOL check_for_updates
)
715 return IDWriteFactory5_GetSystemFontCollection(iface
, FALSE
, (IDWriteFontCollection1
**)collection
,
719 static HRESULT WINAPI
dwritefactory_CreateCustomFontCollection(IDWriteFactory5
*iface
,
720 IDWriteFontCollectionLoader
*loader
, void const *key
, UINT32 key_size
, IDWriteFontCollection
**collection
)
722 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
723 IDWriteFontFileEnumerator
*enumerator
;
724 struct collectionloader
*found
;
727 TRACE("(%p)->(%p %p %u %p)\n", This
, loader
, key
, key_size
, collection
);
734 found
= factory_get_collection_loader(This
, loader
);
738 hr
= IDWriteFontCollectionLoader_CreateEnumeratorFromKey(found
->loader
, (IDWriteFactory
*)iface
,
739 key
, key_size
, &enumerator
);
743 hr
= create_font_collection(iface
, enumerator
, FALSE
, (IDWriteFontCollection1
**)collection
);
744 IDWriteFontFileEnumerator_Release(enumerator
);
748 static HRESULT WINAPI
dwritefactory_RegisterFontCollectionLoader(IDWriteFactory5
*iface
,
749 IDWriteFontCollectionLoader
*loader
)
751 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
752 struct collectionloader
*entry
;
754 TRACE("(%p)->(%p)\n", This
, loader
);
759 if (factory_get_collection_loader(This
, loader
))
760 return DWRITE_E_ALREADYREGISTERED
;
762 entry
= heap_alloc(sizeof(*entry
));
764 return E_OUTOFMEMORY
;
766 entry
->loader
= loader
;
767 IDWriteFontCollectionLoader_AddRef(loader
);
768 list_add_tail(&This
->collection_loaders
, &entry
->entry
);
773 static HRESULT WINAPI
dwritefactory_UnregisterFontCollectionLoader(IDWriteFactory5
*iface
,
774 IDWriteFontCollectionLoader
*loader
)
776 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
777 struct collectionloader
*found
;
779 TRACE("(%p)->(%p)\n", This
, loader
);
784 found
= factory_get_collection_loader(This
, loader
);
788 IDWriteFontCollectionLoader_Release(found
->loader
);
789 list_remove(&found
->entry
);
795 static HRESULT WINAPI
dwritefactory_CreateFontFileReference(IDWriteFactory5
*iface
,
796 WCHAR
const *path
, FILETIME
const *writetime
, IDWriteFontFile
**font_file
)
798 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
803 TRACE("(%p)->(%s %p %p)\n", This
, debugstr_w(path
), writetime
, font_file
);
807 /* Get a reference key in local file loader format. */
808 hr
= get_local_refkey(path
, writetime
, &key
, &key_size
);
812 hr
= create_font_file(This
->localfontfileloader
, key
, key_size
, font_file
);
818 static HRESULT WINAPI
dwritefactory_CreateCustomFontFileReference(IDWriteFactory5
*iface
,
819 void const *reference_key
, UINT32 key_size
, IDWriteFontFileLoader
*loader
, IDWriteFontFile
**font_file
)
821 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
823 TRACE("(%p)->(%p %u %p %p)\n", This
, reference_key
, key_size
, loader
, font_file
);
827 if (!loader
|| !(factory_get_file_loader(This
, loader
) || This
->localfontfileloader
== loader
))
830 return create_font_file(loader
, reference_key
, key_size
, font_file
);
833 void factory_lock(IDWriteFactory5
*iface
)
835 struct dwritefactory
*factory
= impl_from_IDWriteFactory5(iface
);
836 EnterCriticalSection(&factory
->cs
);
839 void factory_unlock(IDWriteFactory5
*iface
)
841 struct dwritefactory
*factory
= impl_from_IDWriteFactory5(iface
);
842 LeaveCriticalSection(&factory
->cs
);
845 HRESULT
factory_get_cached_fontface(IDWriteFactory5
*iface
, IDWriteFontFile
* const *font_files
, UINT32 index
,
846 DWRITE_FONT_SIMULATIONS simulations
, struct list
**cached_list
, REFIID riid
, void **obj
)
848 struct dwritefactory
*factory
= impl_from_IDWriteFactory5(iface
);
849 struct fontfacecached
*cached
;
850 IDWriteFontFileLoader
*loader
;
851 struct list
*fontfaces
;
859 hr
= IDWriteFontFile_GetReferenceKey(*font_files
, &key
, &key_size
);
863 hr
= IDWriteFontFile_GetLoader(*font_files
, &loader
);
867 if (loader
== factory
->localfontfileloader
) {
868 fontfaces
= &factory
->localfontfaces
;
869 IDWriteFontFileLoader_Release(loader
);
872 struct fileloader
*fileloader
= factory_get_file_loader(factory
, loader
);
873 IDWriteFontFileLoader_Release(loader
);
876 fontfaces
= &fileloader
->fontfaces
;
879 *cached_list
= fontfaces
;
881 EnterCriticalSection(&factory
->cs
);
883 /* search through cache list */
884 LIST_FOR_EACH_ENTRY(cached
, fontfaces
, struct fontfacecached
, entry
) {
885 UINT32 cached_key_size
, count
= 1, cached_face_index
;
886 DWRITE_FONT_SIMULATIONS cached_simulations
;
887 const void *cached_key
;
888 IDWriteFontFile
*file
;
890 cached_face_index
= IDWriteFontFace4_GetIndex(cached
->fontface
);
891 cached_simulations
= IDWriteFontFace4_GetSimulations(cached
->fontface
);
894 if (cached_face_index
!= index
|| cached_simulations
!= simulations
)
897 hr
= IDWriteFontFace4_GetFiles(cached
->fontface
, &count
, &file
);
901 hr
= IDWriteFontFile_GetReferenceKey(file
, &cached_key
, &cached_key_size
);
902 IDWriteFontFile_Release(file
);
906 if (cached_key_size
== key_size
&& !memcmp(cached_key
, key
, key_size
)) {
907 if (FAILED(hr
= IDWriteFontFace4_QueryInterface(cached
->fontface
, riid
, obj
)))
908 WARN("Failed to get %s from fontface, hr %#x.\n", debugstr_guid(riid
), hr
);
910 TRACE("returning cached fontface %p\n", cached
->fontface
);
915 LeaveCriticalSection(&factory
->cs
);
917 return *obj
? S_OK
: S_FALSE
;
920 struct fontfacecached
*factory_cache_fontface(IDWriteFactory5
*iface
, struct list
*fontfaces
,
921 IDWriteFontFace4
*fontface
)
923 struct dwritefactory
*factory
= impl_from_IDWriteFactory5(iface
);
924 struct fontfacecached
*cached
;
926 /* new cache entry */
927 cached
= heap_alloc(sizeof(*cached
));
931 cached
->fontface
= fontface
;
932 EnterCriticalSection(&factory
->cs
);
933 list_add_tail(fontfaces
, &cached
->entry
);
934 LeaveCriticalSection(&factory
->cs
);
939 static HRESULT WINAPI
dwritefactory_CreateFontFace(IDWriteFactory5
*iface
, DWRITE_FONT_FACE_TYPE req_facetype
,
940 UINT32 files_number
, IDWriteFontFile
* const* font_files
, UINT32 index
, DWRITE_FONT_SIMULATIONS simulations
,
941 IDWriteFontFace
**fontface
)
943 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
944 DWRITE_FONT_FILE_TYPE file_type
;
945 DWRITE_FONT_FACE_TYPE face_type
;
946 IDWriteFontFileStream
*stream
;
947 struct fontface_desc desc
;
948 struct list
*fontfaces
;
953 TRACE("(%p)->(%d %u %p %u 0x%x %p)\n", This
, req_facetype
, files_number
, font_files
, index
,
954 simulations
, fontface
);
958 if (!is_face_type_supported(req_facetype
))
961 if (req_facetype
!= DWRITE_FONT_FACE_TYPE_OPENTYPE_COLLECTION
&& index
)
964 if (!is_simulation_valid(simulations
))
967 if (FAILED(hr
= get_filestream_from_file(*font_files
, &stream
)))
970 /* check actual file/face type */
971 is_supported
= FALSE
;
972 face_type
= DWRITE_FONT_FACE_TYPE_UNKNOWN
;
973 hr
= opentype_analyze_font(stream
, &is_supported
, &file_type
, &face_type
, &face_count
);
982 if (face_type
!= req_facetype
) {
983 hr
= DWRITE_E_FILEFORMAT
;
987 hr
= factory_get_cached_fontface(iface
, font_files
, index
, simulations
, &fontfaces
,
988 &IID_IDWriteFontFace
, (void **)fontface
);
992 desc
.factory
= iface
;
993 desc
.face_type
= req_facetype
;
994 desc
.files
= font_files
;
995 desc
.stream
= stream
;
996 desc
.files_number
= files_number
;
998 desc
.simulations
= simulations
;
999 desc
.font_data
= NULL
;
1000 hr
= create_fontface(&desc
, fontfaces
, (IDWriteFontFace4
**)fontface
);
1003 IDWriteFontFileStream_Release(stream
);
1007 static HRESULT WINAPI
dwritefactory_CreateRenderingParams(IDWriteFactory5
*iface
, IDWriteRenderingParams
**params
)
1009 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1013 TRACE("(%p)->(%p)\n", This
, params
);
1016 monitor
= MonitorFromPoint(pt
, MONITOR_DEFAULTTOPRIMARY
);
1017 return IDWriteFactory5_CreateMonitorRenderingParams(iface
, monitor
, params
);
1020 static HRESULT WINAPI
dwritefactory_CreateMonitorRenderingParams(IDWriteFactory5
*iface
, HMONITOR monitor
,
1021 IDWriteRenderingParams
**params
)
1023 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1024 IDWriteRenderingParams3
*params3
;
1025 static int fixme_once
= 0;
1028 TRACE("(%p)->(%p %p)\n", This
, monitor
, params
);
1031 FIXME("(%p): monitor setting ignored\n", monitor
);
1033 /* FIXME: use actual per-monitor gamma factor */
1034 hr
= IDWriteFactory5_CreateCustomRenderingParams(iface
, 2.0f
, 0.0f
, 1.0f
, 0.0f
, DWRITE_PIXEL_GEOMETRY_FLAT
,
1035 DWRITE_RENDERING_MODE1_DEFAULT
, DWRITE_GRID_FIT_MODE_DEFAULT
, ¶ms3
);
1036 *params
= (IDWriteRenderingParams
*)params3
;
1040 static HRESULT WINAPI
dwritefactory_CreateCustomRenderingParams(IDWriteFactory5
*iface
, FLOAT gamma
,
1041 FLOAT enhancedContrast
, FLOAT cleartype_level
, DWRITE_PIXEL_GEOMETRY geometry
, DWRITE_RENDERING_MODE mode
,
1042 IDWriteRenderingParams
**params
)
1044 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1045 IDWriteRenderingParams3
*params3
;
1048 TRACE("(%p)->(%f %f %f %d %d %p)\n", This
, gamma
, enhancedContrast
, cleartype_level
, geometry
, mode
, params
);
1050 if ((UINT32
)mode
> DWRITE_RENDERING_MODE_OUTLINE
) {
1052 return E_INVALIDARG
;
1055 hr
= IDWriteFactory5_CreateCustomRenderingParams(iface
, gamma
, enhancedContrast
, 1.0f
, cleartype_level
, geometry
,
1056 (DWRITE_RENDERING_MODE1
)mode
, DWRITE_GRID_FIT_MODE_DEFAULT
, ¶ms3
);
1057 *params
= (IDWriteRenderingParams
*)params3
;
1061 static HRESULT WINAPI
dwritefactory_RegisterFontFileLoader(IDWriteFactory5
*iface
, IDWriteFontFileLoader
*loader
)
1063 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1064 struct fileloader
*entry
;
1066 TRACE("(%p)->(%p)\n", This
, loader
);
1069 return E_INVALIDARG
;
1071 if (factory_get_file_loader(This
, loader
))
1072 return DWRITE_E_ALREADYREGISTERED
;
1074 entry
= heap_alloc(sizeof(*entry
));
1076 return E_OUTOFMEMORY
;
1078 entry
->loader
= loader
;
1079 list_init(&entry
->fontfaces
);
1080 IDWriteFontFileLoader_AddRef(loader
);
1081 list_add_tail(&This
->file_loaders
, &entry
->entry
);
1086 static HRESULT WINAPI
dwritefactory_UnregisterFontFileLoader(IDWriteFactory5
*iface
, IDWriteFontFileLoader
*loader
)
1088 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1089 struct fileloader
*found
;
1091 TRACE("(%p)->(%p)\n", This
, loader
);
1094 return E_INVALIDARG
;
1096 found
= factory_get_file_loader(This
, loader
);
1098 return E_INVALIDARG
;
1100 release_fileloader(found
);
1104 static HRESULT WINAPI
dwritefactory_CreateTextFormat(IDWriteFactory5
*iface
, WCHAR
const* family_name
,
1105 IDWriteFontCollection
*collection
, DWRITE_FONT_WEIGHT weight
, DWRITE_FONT_STYLE style
,
1106 DWRITE_FONT_STRETCH stretch
, FLOAT size
, WCHAR
const *locale
, IDWriteTextFormat
**format
)
1108 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1111 TRACE("(%p)->(%s %p %d %d %d %f %s %p)\n", This
, debugstr_w(family_name
), collection
, weight
, style
, stretch
,
1112 size
, debugstr_w(locale
), format
);
1115 IDWriteFontCollection_AddRef(collection
);
1117 collection
= (IDWriteFontCollection
*)factory_get_system_collection(This
);
1124 hr
= create_textformat(family_name
, collection
, weight
, style
, stretch
, size
, locale
, format
);
1125 IDWriteFontCollection_Release(collection
);
1129 static HRESULT WINAPI
dwritefactory_CreateTypography(IDWriteFactory5
*iface
, IDWriteTypography
**typography
)
1131 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1132 TRACE("(%p)->(%p)\n", This
, typography
);
1133 return create_typography(typography
);
1136 static HRESULT WINAPI
dwritefactory_GetGdiInterop(IDWriteFactory5
*iface
, IDWriteGdiInterop
**gdi_interop
)
1138 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1141 TRACE("(%p)->(%p)\n", This
, gdi_interop
);
1143 if (This
->gdiinterop
)
1144 IDWriteGdiInterop1_AddRef(This
->gdiinterop
);
1146 hr
= create_gdiinterop(iface
, &This
->gdiinterop
);
1148 *gdi_interop
= (IDWriteGdiInterop
*)This
->gdiinterop
;
1153 static HRESULT WINAPI
dwritefactory_CreateTextLayout(IDWriteFactory5
*iface
, WCHAR
const* string
,
1154 UINT32 length
, IDWriteTextFormat
*format
, FLOAT max_width
, FLOAT max_height
, IDWriteTextLayout
**layout
)
1156 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1157 struct textlayout_desc desc
;
1159 TRACE("(%p)->(%s:%u %p %f %f %p)\n", This
, debugstr_wn(string
, length
), length
, format
, max_width
, max_height
, layout
);
1161 desc
.factory
= iface
;
1162 desc
.string
= string
;
1163 desc
.length
= length
;
1164 desc
.format
= format
;
1165 desc
.max_width
= max_width
;
1166 desc
.max_height
= max_height
;
1167 desc
.is_gdi_compatible
= FALSE
;
1169 desc
.transform
= NULL
;
1170 desc
.use_gdi_natural
= FALSE
;
1171 return create_textlayout(&desc
, layout
);
1174 static HRESULT WINAPI
dwritefactory_CreateGdiCompatibleTextLayout(IDWriteFactory5
*iface
, WCHAR
const* string
,
1175 UINT32 length
, IDWriteTextFormat
*format
, FLOAT max_width
, FLOAT max_height
, FLOAT pixels_per_dip
,
1176 DWRITE_MATRIX
const* transform
, BOOL use_gdi_natural
, IDWriteTextLayout
**layout
)
1178 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1179 struct textlayout_desc desc
;
1181 TRACE("(%p)->(%s:%u %p %f %f %f %p %d %p)\n", This
, debugstr_wn(string
, length
), length
, format
, max_width
,
1182 max_height
, pixels_per_dip
, transform
, use_gdi_natural
, layout
);
1184 desc
.factory
= iface
;
1185 desc
.string
= string
;
1186 desc
.length
= length
;
1187 desc
.format
= format
;
1188 desc
.max_width
= max_width
;
1189 desc
.max_height
= max_height
;
1190 desc
.is_gdi_compatible
= TRUE
;
1191 desc
.ppdip
= pixels_per_dip
;
1192 desc
.transform
= transform
;
1193 desc
.use_gdi_natural
= use_gdi_natural
;
1194 return create_textlayout(&desc
, layout
);
1197 static HRESULT WINAPI
dwritefactory_CreateEllipsisTrimmingSign(IDWriteFactory5
*iface
, IDWriteTextFormat
*format
,
1198 IDWriteInlineObject
**trimming_sign
)
1200 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1201 TRACE("(%p)->(%p %p)\n", This
, format
, trimming_sign
);
1202 return create_trimmingsign(iface
, format
, trimming_sign
);
1205 static HRESULT WINAPI
dwritefactory_CreateTextAnalyzer(IDWriteFactory5
*iface
, IDWriteTextAnalyzer
**analyzer
)
1207 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1209 TRACE("(%p)->(%p)\n", This
, analyzer
);
1211 *analyzer
= get_text_analyzer();
1216 static HRESULT WINAPI
dwritefactory_CreateNumberSubstitution(IDWriteFactory5
*iface
,
1217 DWRITE_NUMBER_SUBSTITUTION_METHOD method
, WCHAR
const* locale
, BOOL ignore_user_override
,
1218 IDWriteNumberSubstitution
**substitution
)
1220 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1221 TRACE("(%p)->(%d %s %d %p)\n", This
, method
, debugstr_w(locale
), ignore_user_override
, substitution
);
1222 return create_numbersubstitution(method
, locale
, ignore_user_override
, substitution
);
1225 static HRESULT WINAPI
dwritefactory_CreateGlyphRunAnalysis(IDWriteFactory5
*iface
, DWRITE_GLYPH_RUN
const *run
,
1226 FLOAT ppdip
, DWRITE_MATRIX
const* transform
, DWRITE_RENDERING_MODE rendering_mode
,
1227 DWRITE_MEASURING_MODE measuring_mode
, FLOAT originX
, FLOAT originY
, IDWriteGlyphRunAnalysis
**analysis
)
1229 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1230 struct glyphrunanalysis_desc desc
;
1232 TRACE("(%p)->(%p %.2f %p %d %d %.2f %.2f %p)\n", This
, run
, ppdip
, transform
, rendering_mode
,
1233 measuring_mode
, originX
, originY
, analysis
);
1235 if (ppdip
<= 0.0f
) {
1237 return E_INVALIDARG
;
1242 desc
.transform
= transform
;
1243 desc
.rendering_mode
= (DWRITE_RENDERING_MODE1
)rendering_mode
;
1244 desc
.measuring_mode
= measuring_mode
;
1245 desc
.gridfit_mode
= DWRITE_GRID_FIT_MODE_DEFAULT
;
1246 desc
.aa_mode
= DWRITE_TEXT_ANTIALIAS_MODE_CLEARTYPE
;
1247 desc
.origin_x
= originX
;
1248 desc
.origin_y
= originY
;
1249 return create_glyphrunanalysis(&desc
, analysis
);
1252 static HRESULT WINAPI
dwritefactory1_GetEudcFontCollection(IDWriteFactory5
*iface
, IDWriteFontCollection
**collection
,
1253 BOOL check_for_updates
)
1255 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1258 TRACE("(%p)->(%p %d)\n", This
, collection
, check_for_updates
);
1260 if (check_for_updates
)
1261 FIXME("checking for eudc updates not implemented\n");
1263 if (This
->eudc_collection
)
1264 IDWriteFontCollection1_AddRef(This
->eudc_collection
);
1266 IDWriteFontCollection1
*eudc_collection
;
1268 if (FAILED(hr
= get_eudc_fontcollection(iface
, &eudc_collection
))) {
1270 WARN("Failed to get EUDC collection, hr %#x.\n", hr
);
1274 if (InterlockedCompareExchangePointer((void **)&This
->eudc_collection
, eudc_collection
, NULL
))
1275 IDWriteFontCollection1_Release(eudc_collection
);
1278 *collection
= (IDWriteFontCollection
*)This
->eudc_collection
;
1283 static HRESULT WINAPI
dwritefactory1_CreateCustomRenderingParams(IDWriteFactory5
*iface
, FLOAT gamma
,
1284 FLOAT enhcontrast
, FLOAT enhcontrast_grayscale
, FLOAT cleartype_level
, DWRITE_PIXEL_GEOMETRY geometry
,
1285 DWRITE_RENDERING_MODE mode
, IDWriteRenderingParams1
** params
)
1287 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1288 IDWriteRenderingParams3
*params3
;
1291 TRACE("(%p)->(%.2f %.2f %.2f %.2f %d %d %p)\n", This
, gamma
, enhcontrast
, enhcontrast_grayscale
,
1292 cleartype_level
, geometry
, mode
, params
);
1294 if ((UINT32
)mode
> DWRITE_RENDERING_MODE_OUTLINE
) {
1296 return E_INVALIDARG
;
1299 hr
= IDWriteFactory5_CreateCustomRenderingParams(iface
, gamma
, enhcontrast
, enhcontrast_grayscale
,
1300 cleartype_level
, geometry
, (DWRITE_RENDERING_MODE1
)mode
, DWRITE_GRID_FIT_MODE_DEFAULT
, ¶ms3
);
1301 *params
= (IDWriteRenderingParams1
*)params3
;
1305 static HRESULT WINAPI
dwritefactory2_GetSystemFontFallback(IDWriteFactory5
*iface
, IDWriteFontFallback
**fallback
)
1307 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1309 TRACE("(%p)->(%p)\n", This
, fallback
);
1313 if (!This
->fallback
) {
1314 HRESULT hr
= create_system_fontfallback(iface
, &This
->fallback
);
1319 *fallback
= This
->fallback
;
1320 IDWriteFontFallback_AddRef(*fallback
);
1324 static HRESULT WINAPI
dwritefactory2_CreateFontFallbackBuilder(IDWriteFactory5
*iface
,
1325 IDWriteFontFallbackBuilder
**fallbackbuilder
)
1327 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1329 TRACE("(%p)->(%p)\n", This
, fallbackbuilder
);
1331 return create_fontfallback_builder(iface
, fallbackbuilder
);
1334 static HRESULT WINAPI
dwritefactory2_TranslateColorGlyphRun(IDWriteFactory5
*iface
, FLOAT originX
, FLOAT originY
,
1335 const DWRITE_GLYPH_RUN
*run
, const DWRITE_GLYPH_RUN_DESCRIPTION
*rundescr
, DWRITE_MEASURING_MODE mode
,
1336 const DWRITE_MATRIX
*transform
, UINT32 palette
, IDWriteColorGlyphRunEnumerator
**colorlayers
)
1338 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1339 TRACE("(%p)->(%.2f %.2f %p %p %d %p %u %p)\n", This
, originX
, originY
, run
, rundescr
, mode
,
1340 transform
, palette
, colorlayers
);
1341 return create_colorglyphenum(originX
, originY
, run
, rundescr
, mode
, transform
, palette
, colorlayers
);
1344 static HRESULT WINAPI
dwritefactory2_CreateCustomRenderingParams(IDWriteFactory5
*iface
, FLOAT gamma
, FLOAT contrast
,
1345 FLOAT grayscalecontrast
, FLOAT cleartype_level
, DWRITE_PIXEL_GEOMETRY geometry
, DWRITE_RENDERING_MODE mode
,
1346 DWRITE_GRID_FIT_MODE gridfit
, IDWriteRenderingParams2
**params
)
1348 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1349 IDWriteRenderingParams3
*params3
;
1352 TRACE("(%p)->(%.2f %.2f %.2f %.2f %d %d %d %p)\n", This
, gamma
, contrast
, grayscalecontrast
, cleartype_level
,
1353 geometry
, mode
, gridfit
, params
);
1355 if ((UINT32
)mode
> DWRITE_RENDERING_MODE_OUTLINE
) {
1357 return E_INVALIDARG
;
1360 hr
= IDWriteFactory5_CreateCustomRenderingParams(iface
, gamma
, contrast
, grayscalecontrast
,
1361 cleartype_level
, geometry
, (DWRITE_RENDERING_MODE1
)mode
, DWRITE_GRID_FIT_MODE_DEFAULT
, ¶ms3
);
1362 *params
= (IDWriteRenderingParams2
*)params3
;
1366 static HRESULT WINAPI
dwritefactory2_CreateGlyphRunAnalysis(IDWriteFactory5
*iface
, const DWRITE_GLYPH_RUN
*run
,
1367 const DWRITE_MATRIX
*transform
, DWRITE_RENDERING_MODE rendering_mode
, DWRITE_MEASURING_MODE measuring_mode
,
1368 DWRITE_GRID_FIT_MODE gridfit_mode
, DWRITE_TEXT_ANTIALIAS_MODE aa_mode
, FLOAT originX
, FLOAT originY
,
1369 IDWriteGlyphRunAnalysis
**analysis
)
1371 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1372 struct glyphrunanalysis_desc desc
;
1374 TRACE("(%p)->(%p %p %d %d %d %d %.2f %.2f %p)\n", This
, run
, transform
, rendering_mode
, measuring_mode
,
1375 gridfit_mode
, aa_mode
, originX
, originY
, analysis
);
1379 desc
.transform
= transform
;
1380 desc
.rendering_mode
= (DWRITE_RENDERING_MODE1
)rendering_mode
;
1381 desc
.measuring_mode
= measuring_mode
;
1382 desc
.gridfit_mode
= gridfit_mode
;
1383 desc
.aa_mode
= aa_mode
;
1384 desc
.origin_x
= originX
;
1385 desc
.origin_y
= originY
;
1386 return create_glyphrunanalysis(&desc
, analysis
);
1389 static HRESULT WINAPI
dwritefactory3_CreateGlyphRunAnalysis(IDWriteFactory5
*iface
, DWRITE_GLYPH_RUN
const *run
,
1390 DWRITE_MATRIX
const *transform
, DWRITE_RENDERING_MODE1 rendering_mode
, DWRITE_MEASURING_MODE measuring_mode
,
1391 DWRITE_GRID_FIT_MODE gridfit_mode
, DWRITE_TEXT_ANTIALIAS_MODE aa_mode
, FLOAT originX
, FLOAT originY
,
1392 IDWriteGlyphRunAnalysis
**analysis
)
1394 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1395 struct glyphrunanalysis_desc desc
;
1397 TRACE("(%p)->(%p %p %d %d %d %d %.2f %.2f %p)\n", This
, run
, transform
, rendering_mode
, measuring_mode
,
1398 gridfit_mode
, aa_mode
, originX
, originY
, analysis
);
1402 desc
.transform
= transform
;
1403 desc
.rendering_mode
= rendering_mode
;
1404 desc
.measuring_mode
= measuring_mode
;
1405 desc
.gridfit_mode
= gridfit_mode
;
1406 desc
.aa_mode
= aa_mode
;
1407 desc
.origin_x
= originX
;
1408 desc
.origin_y
= originY
;
1409 return create_glyphrunanalysis(&desc
, analysis
);
1412 static HRESULT WINAPI
dwritefactory3_CreateCustomRenderingParams(IDWriteFactory5
*iface
, FLOAT gamma
, FLOAT contrast
,
1413 FLOAT grayscale_contrast
, FLOAT cleartype_level
, DWRITE_PIXEL_GEOMETRY pixel_geometry
,
1414 DWRITE_RENDERING_MODE1 rendering_mode
, DWRITE_GRID_FIT_MODE gridfit_mode
, IDWriteRenderingParams3
**params
)
1416 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1418 TRACE("(%p)->(%.2f %.2f %.2f %.2f %d %d %d %p)\n", This
, gamma
, contrast
, grayscale_contrast
, cleartype_level
,
1419 pixel_geometry
, rendering_mode
, gridfit_mode
, params
);
1421 return create_renderingparams(gamma
, contrast
, grayscale_contrast
, cleartype_level
, pixel_geometry
, rendering_mode
,
1422 gridfit_mode
, params
);
1425 static HRESULT WINAPI
dwritefactory3_CreateFontFaceReference_(IDWriteFactory5
*iface
, IDWriteFontFile
*file
,
1426 UINT32 index
, DWRITE_FONT_SIMULATIONS simulations
, IDWriteFontFaceReference
**reference
)
1428 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1430 TRACE("(%p)->(%p %u %x %p)\n", This
, file
, index
, simulations
, reference
);
1432 return create_fontfacereference(iface
, file
, index
, simulations
, reference
);
1435 static HRESULT WINAPI
dwritefactory3_CreateFontFaceReference(IDWriteFactory5
*iface
, WCHAR
const *path
,
1436 FILETIME
const *writetime
, UINT32 index
, DWRITE_FONT_SIMULATIONS simulations
,
1437 IDWriteFontFaceReference
**reference
)
1439 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1440 IDWriteFontFile
*file
;
1443 TRACE("(%p)->(%s %p %u %x, %p)\n", This
, debugstr_w(path
), writetime
, index
, simulations
, reference
);
1445 hr
= IDWriteFactory5_CreateFontFileReference(iface
, path
, writetime
, &file
);
1451 hr
= IDWriteFactory5_CreateFontFaceReference_(iface
, file
, index
, simulations
, reference
);
1452 IDWriteFontFile_Release(file
);
1456 static HRESULT WINAPI
dwritefactory3_GetSystemFontSet(IDWriteFactory5
*iface
, IDWriteFontSet
**fontset
)
1458 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1460 FIXME("(%p)->(%p): stub\n", This
, fontset
);
1465 static HRESULT WINAPI
dwritefactory3_CreateFontSetBuilder(IDWriteFactory5
*iface
, IDWriteFontSetBuilder
**builder
)
1467 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1469 FIXME("(%p)->(%p): stub\n", This
, builder
);
1474 static HRESULT WINAPI
dwritefactory3_CreateFontCollectionFromFontSet(IDWriteFactory5
*iface
, IDWriteFontSet
*fontset
,
1475 IDWriteFontCollection1
**collection
)
1477 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1479 FIXME("(%p)->(%p %p): stub\n", This
, fontset
, collection
);
1484 static HRESULT WINAPI
dwritefactory3_GetSystemFontCollection(IDWriteFactory5
*iface
, BOOL include_downloadable
,
1485 IDWriteFontCollection1
**collection
, BOOL check_for_updates
)
1487 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1489 TRACE("(%p)->(%d %p %d)\n", This
, include_downloadable
, collection
, check_for_updates
);
1491 if (include_downloadable
)
1492 FIXME("remote fonts are not supported\n");
1494 if (check_for_updates
)
1495 FIXME("checking for system font updates not implemented\n");
1497 *collection
= factory_get_system_collection(This
);
1499 return *collection
? S_OK
: E_FAIL
;
1502 static HRESULT WINAPI
dwritefactory3_GetFontDownloadQueue(IDWriteFactory5
*iface
, IDWriteFontDownloadQueue
**queue
)
1504 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1506 FIXME("(%p)->(%p): stub\n", This
, queue
);
1511 static HRESULT WINAPI
dwritefactory4_TranslateColorGlyphRun(IDWriteFactory5
*iface
, D2D1_POINT_2F baseline_origin
,
1512 DWRITE_GLYPH_RUN
const *run
, DWRITE_GLYPH_RUN_DESCRIPTION
const *run_desc
,
1513 DWRITE_GLYPH_IMAGE_FORMATS desired_formats
, DWRITE_MEASURING_MODE measuring_mode
, DWRITE_MATRIX
const *transform
,
1514 UINT32 palette
, IDWriteColorGlyphRunEnumerator1
**layers
)
1516 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1518 FIXME("(%p)->(%p %p %u %d %p %u %p): stub\n", This
, run
, run_desc
, desired_formats
, measuring_mode
,
1519 transform
, palette
, layers
);
1524 static HRESULT
compute_glyph_origins(DWRITE_GLYPH_RUN
const *run
, DWRITE_MEASURING_MODE measuring_mode
,
1525 D2D1_POINT_2F baseline_origin
, DWRITE_MATRIX
const *transform
, D2D1_POINT_2F
*origins
)
1527 IDWriteFontFace1
*fontface1
= NULL
;
1528 DWRITE_FONT_METRICS metrics
;
1533 rtl_factor
= run
->bidiLevel
& 1 ? -1.0f
: 1.0f
;
1535 if (run
->fontFace
) {
1536 IDWriteFontFace_GetMetrics(run
->fontFace
, &metrics
);
1537 if (FAILED(hr
= IDWriteFontFace_QueryInterface(run
->fontFace
, &IID_IDWriteFontFace1
, (void **)&fontface1
)))
1538 WARN("Failed to get IDWriteFontFace1, %#x.\n", hr
);
1541 for (i
= 0; i
< run
->glyphCount
; i
++) {
1544 /* Use nominal advances if not provided by caller. */
1545 if (run
->glyphAdvances
)
1546 advance
= rtl_factor
* run
->glyphAdvances
[i
];
1551 switch (measuring_mode
)
1553 case DWRITE_MEASURING_MODE_NATURAL
:
1554 if (SUCCEEDED(IDWriteFontFace1_GetDesignGlyphAdvances(fontface1
, 1, run
->glyphIndices
+ i
, &a
,
1556 advance
= rtl_factor
* get_scaled_advance_width(a
, run
->fontEmSize
, &metrics
);
1558 case DWRITE_MEASURING_MODE_GDI_CLASSIC
:
1559 case DWRITE_MEASURING_MODE_GDI_NATURAL
:
1560 if (SUCCEEDED(IDWriteFontFace1_GetGdiCompatibleGlyphAdvances(fontface1
, run
->fontEmSize
,
1561 1.0f
, transform
, measuring_mode
== DWRITE_MEASURING_MODE_GDI_NATURAL
,
1562 run
->isSideways
, 1, run
->glyphIndices
+ i
, &a
)))
1563 advance
= rtl_factor
* floorf(a
* run
->fontEmSize
/ metrics
.designUnitsPerEm
+ 0.5f
);
1570 origins
[i
] = baseline_origin
;
1572 /* Apply offsets. */
1573 if (run
->glyphOffsets
) {
1574 FLOAT advanceoffset
= rtl_factor
* run
->glyphOffsets
[i
].advanceOffset
;
1575 FLOAT ascenderoffset
= -run
->glyphOffsets
[i
].ascenderOffset
;
1577 if (run
->isSideways
) {
1578 origins
[i
].x
+= ascenderoffset
;
1579 origins
[i
].y
+= advanceoffset
;
1582 origins
[i
].x
+= advanceoffset
;
1583 origins
[i
].y
+= ascenderoffset
;
1587 if (run
->isSideways
)
1588 baseline_origin
.y
+= advance
;
1590 baseline_origin
.x
+= advance
;
1594 IDWriteFontFace1_Release(fontface1
);
1598 static HRESULT WINAPI
dwritefactory4_ComputeGlyphOrigins_(IDWriteFactory5
*iface
, DWRITE_GLYPH_RUN
const *run
,
1599 D2D1_POINT_2F baseline_origin
, D2D1_POINT_2F
*origins
)
1601 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1603 TRACE("(%p)->(%p (%f,%f) %p)\n", This
, run
, baseline_origin
.x
, baseline_origin
.y
, origins
);
1605 return compute_glyph_origins(run
, DWRITE_MEASURING_MODE_NATURAL
, baseline_origin
, NULL
, origins
);
1608 static HRESULT WINAPI
dwritefactory4_ComputeGlyphOrigins(IDWriteFactory5
*iface
, DWRITE_GLYPH_RUN
const *run
,
1609 DWRITE_MEASURING_MODE measuring_mode
, D2D1_POINT_2F baseline_origin
, DWRITE_MATRIX
const *transform
,
1610 D2D1_POINT_2F
*origins
)
1612 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1614 TRACE("(%p)->(%p %d (%f,%f) %p %p)\n", This
, run
, measuring_mode
, baseline_origin
.x
, baseline_origin
.y
,
1615 transform
, origins
);
1617 return compute_glyph_origins(run
, measuring_mode
, baseline_origin
, transform
, origins
);
1620 static HRESULT WINAPI
dwritefactory5_CreateFontSetBuilder(IDWriteFactory5
*iface
, IDWriteFontSetBuilder1
**builder
)
1622 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1624 FIXME("(%p)->(%p): stub\n", This
, builder
);
1629 static HRESULT WINAPI
dwritefactory5_CreateInMemoryFontFileLoader(IDWriteFactory5
*iface
, IDWriteFontFileLoader
**loader
)
1631 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1633 TRACE("(%p)->(%p)\n", This
, loader
);
1635 return create_inmemory_fileloader(loader
);
1638 static HRESULT WINAPI
dwritefactory5_CreateHttpFontFileLoader(IDWriteFactory5
*iface
, WCHAR
const *referrer_url
, WCHAR
const *extra_headers
,
1639 IDWriteRemoteFontFileLoader
**loader
)
1641 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1643 FIXME("(%p)->(%s %s %p): stub\n", This
, debugstr_w(referrer_url
), wine_dbgstr_w(extra_headers
), loader
);
1648 static DWRITE_CONTAINER_TYPE WINAPI
dwritefactory5_AnalyzeContainerType(IDWriteFactory5
*iface
, void const *data
, UINT32 data_size
)
1650 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1652 TRACE("(%p)->(%p %u)\n", This
, data
, data_size
);
1654 return opentype_analyze_container_type(data
, data_size
);
1657 static HRESULT WINAPI
dwritefactory5_UnpackFontFile(IDWriteFactory5
*iface
, DWRITE_CONTAINER_TYPE container_type
, void const *data
,
1658 UINT32 data_size
, IDWriteFontFileStream
**stream
)
1660 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1662 FIXME("(%p)->(%d %p %u %p): stub\n", This
, container_type
, data
, data_size
, stream
);
1667 static const struct IDWriteFactory5Vtbl dwritefactoryvtbl
= {
1668 dwritefactory_QueryInterface
,
1669 dwritefactory_AddRef
,
1670 dwritefactory_Release
,
1671 dwritefactory_GetSystemFontCollection
,
1672 dwritefactory_CreateCustomFontCollection
,
1673 dwritefactory_RegisterFontCollectionLoader
,
1674 dwritefactory_UnregisterFontCollectionLoader
,
1675 dwritefactory_CreateFontFileReference
,
1676 dwritefactory_CreateCustomFontFileReference
,
1677 dwritefactory_CreateFontFace
,
1678 dwritefactory_CreateRenderingParams
,
1679 dwritefactory_CreateMonitorRenderingParams
,
1680 dwritefactory_CreateCustomRenderingParams
,
1681 dwritefactory_RegisterFontFileLoader
,
1682 dwritefactory_UnregisterFontFileLoader
,
1683 dwritefactory_CreateTextFormat
,
1684 dwritefactory_CreateTypography
,
1685 dwritefactory_GetGdiInterop
,
1686 dwritefactory_CreateTextLayout
,
1687 dwritefactory_CreateGdiCompatibleTextLayout
,
1688 dwritefactory_CreateEllipsisTrimmingSign
,
1689 dwritefactory_CreateTextAnalyzer
,
1690 dwritefactory_CreateNumberSubstitution
,
1691 dwritefactory_CreateGlyphRunAnalysis
,
1692 dwritefactory1_GetEudcFontCollection
,
1693 dwritefactory1_CreateCustomRenderingParams
,
1694 dwritefactory2_GetSystemFontFallback
,
1695 dwritefactory2_CreateFontFallbackBuilder
,
1696 dwritefactory2_TranslateColorGlyphRun
,
1697 dwritefactory2_CreateCustomRenderingParams
,
1698 dwritefactory2_CreateGlyphRunAnalysis
,
1699 dwritefactory3_CreateGlyphRunAnalysis
,
1700 dwritefactory3_CreateCustomRenderingParams
,
1701 dwritefactory3_CreateFontFaceReference_
,
1702 dwritefactory3_CreateFontFaceReference
,
1703 dwritefactory3_GetSystemFontSet
,
1704 dwritefactory3_CreateFontSetBuilder
,
1705 dwritefactory3_CreateFontCollectionFromFontSet
,
1706 dwritefactory3_GetSystemFontCollection
,
1707 dwritefactory3_GetFontDownloadQueue
,
1708 dwritefactory4_TranslateColorGlyphRun
,
1709 dwritefactory4_ComputeGlyphOrigins_
,
1710 dwritefactory4_ComputeGlyphOrigins
,
1711 dwritefactory5_CreateFontSetBuilder
,
1712 dwritefactory5_CreateInMemoryFontFileLoader
,
1713 dwritefactory5_CreateHttpFontFileLoader
,
1714 dwritefactory5_AnalyzeContainerType
,
1715 dwritefactory5_UnpackFontFile
,
1718 static ULONG WINAPI
shareddwritefactory_AddRef(IDWriteFactory5
*iface
)
1720 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1721 TRACE("(%p)\n", This
);
1725 static ULONG WINAPI
shareddwritefactory_Release(IDWriteFactory5
*iface
)
1727 struct dwritefactory
*This
= impl_from_IDWriteFactory5(iface
);
1728 TRACE("(%p)\n", This
);
1732 static const struct IDWriteFactory5Vtbl shareddwritefactoryvtbl
= {
1733 dwritefactory_QueryInterface
,
1734 shareddwritefactory_AddRef
,
1735 shareddwritefactory_Release
,
1736 dwritefactory_GetSystemFontCollection
,
1737 dwritefactory_CreateCustomFontCollection
,
1738 dwritefactory_RegisterFontCollectionLoader
,
1739 dwritefactory_UnregisterFontCollectionLoader
,
1740 dwritefactory_CreateFontFileReference
,
1741 dwritefactory_CreateCustomFontFileReference
,
1742 dwritefactory_CreateFontFace
,
1743 dwritefactory_CreateRenderingParams
,
1744 dwritefactory_CreateMonitorRenderingParams
,
1745 dwritefactory_CreateCustomRenderingParams
,
1746 dwritefactory_RegisterFontFileLoader
,
1747 dwritefactory_UnregisterFontFileLoader
,
1748 dwritefactory_CreateTextFormat
,
1749 dwritefactory_CreateTypography
,
1750 dwritefactory_GetGdiInterop
,
1751 dwritefactory_CreateTextLayout
,
1752 dwritefactory_CreateGdiCompatibleTextLayout
,
1753 dwritefactory_CreateEllipsisTrimmingSign
,
1754 dwritefactory_CreateTextAnalyzer
,
1755 dwritefactory_CreateNumberSubstitution
,
1756 dwritefactory_CreateGlyphRunAnalysis
,
1757 dwritefactory1_GetEudcFontCollection
,
1758 dwritefactory1_CreateCustomRenderingParams
,
1759 dwritefactory2_GetSystemFontFallback
,
1760 dwritefactory2_CreateFontFallbackBuilder
,
1761 dwritefactory2_TranslateColorGlyphRun
,
1762 dwritefactory2_CreateCustomRenderingParams
,
1763 dwritefactory2_CreateGlyphRunAnalysis
,
1764 dwritefactory3_CreateGlyphRunAnalysis
,
1765 dwritefactory3_CreateCustomRenderingParams
,
1766 dwritefactory3_CreateFontFaceReference_
,
1767 dwritefactory3_CreateFontFaceReference
,
1768 dwritefactory3_GetSystemFontSet
,
1769 dwritefactory3_CreateFontSetBuilder
,
1770 dwritefactory3_CreateFontCollectionFromFontSet
,
1771 dwritefactory3_GetSystemFontCollection
,
1772 dwritefactory3_GetFontDownloadQueue
,
1773 dwritefactory4_TranslateColorGlyphRun
,
1774 dwritefactory4_ComputeGlyphOrigins_
,
1775 dwritefactory4_ComputeGlyphOrigins
,
1776 dwritefactory5_CreateFontSetBuilder
,
1777 dwritefactory5_CreateInMemoryFontFileLoader
,
1778 dwritefactory5_CreateHttpFontFileLoader
,
1779 dwritefactory5_AnalyzeContainerType
,
1780 dwritefactory5_UnpackFontFile
,
1783 static void init_dwritefactory(struct dwritefactory
*factory
, DWRITE_FACTORY_TYPE type
)
1785 factory
->IDWriteFactory5_iface
.lpVtbl
= type
== DWRITE_FACTORY_TYPE_SHARED
?
1786 &shareddwritefactoryvtbl
: &dwritefactoryvtbl
;
1788 factory
->localfontfileloader
= get_local_fontfile_loader();
1789 factory
->system_collection
= NULL
;
1790 factory
->eudc_collection
= NULL
;
1791 factory
->gdiinterop
= NULL
;
1792 factory
->fallback
= NULL
;
1794 list_init(&factory
->collection_loaders
);
1795 list_init(&factory
->file_loaders
);
1796 list_init(&factory
->localfontfaces
);
1798 InitializeCriticalSection(&factory
->cs
);
1799 factory
->cs
.DebugInfo
->Spare
[0] = (DWORD_PTR
)(__FILE__
": dwritefactory.lock");
1802 void factory_detach_fontcollection(IDWriteFactory5
*iface
, IDWriteFontCollection1
*collection
)
1804 struct dwritefactory
*factory
= impl_from_IDWriteFactory5(iface
);
1805 InterlockedCompareExchangePointer((void **)&factory
->system_collection
, NULL
, collection
);
1806 InterlockedCompareExchangePointer((void **)&factory
->eudc_collection
, NULL
, collection
);
1807 IDWriteFactory5_Release(iface
);
1810 void factory_detach_gdiinterop(IDWriteFactory5
*iface
, IDWriteGdiInterop1
*interop
)
1812 struct dwritefactory
*factory
= impl_from_IDWriteFactory5(iface
);
1813 factory
->gdiinterop
= NULL
;
1814 IDWriteFactory5_Release(iface
);
1817 HRESULT WINAPI
DWriteCreateFactory(DWRITE_FACTORY_TYPE type
, REFIID riid
, IUnknown
**ret
)
1819 struct dwritefactory
*factory
;
1822 TRACE("(%d, %s, %p)\n", type
, debugstr_guid(riid
), ret
);
1826 if (type
== DWRITE_FACTORY_TYPE_SHARED
&& shared_factory
)
1827 return IDWriteFactory5_QueryInterface(shared_factory
, riid
, (void**)ret
);
1829 factory
= heap_alloc(sizeof(struct dwritefactory
));
1830 if (!factory
) return E_OUTOFMEMORY
;
1832 init_dwritefactory(factory
, type
);
1834 if (type
== DWRITE_FACTORY_TYPE_SHARED
)
1835 if (InterlockedCompareExchangePointer((void**)&shared_factory
, &factory
->IDWriteFactory5_iface
, NULL
)) {
1836 release_shared_factory(&factory
->IDWriteFactory5_iface
);
1837 return IDWriteFactory5_QueryInterface(shared_factory
, riid
, (void**)ret
);
1840 hr
= IDWriteFactory5_QueryInterface(&factory
->IDWriteFactory5_iface
, riid
, (void**)ret
);
1841 IDWriteFactory5_Release(&factory
->IDWriteFactory5_iface
);