2 * Copyright 2014 Henri Verbeet for CodeWeavers
3 * Copyright 2016 Nikolay Sivov for CodeWeavers
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20 #include "d2d1_private.h"
22 WINE_DEFAULT_DEBUG_CHANNEL(d2d
);
24 static inline struct d2d_dc_render_target
*impl_from_IUnknown(IUnknown
*iface
)
26 return CONTAINING_RECORD(iface
, struct d2d_dc_render_target
, ID2D1DCRenderTarget_iface
);
29 static HRESULT
d2d_dc_render_target_present(IUnknown
*outer_unknown
)
31 struct d2d_dc_render_target
*render_target
= impl_from_IUnknown(outer_unknown
);
32 const RECT
*dst_rect
= &render_target
->dst_rect
;
37 if (!render_target
->hdc
)
38 return D2DERR_WRONG_STATE
;
40 if (FAILED(hr
= IDXGISurface1_GetDC(render_target
->dxgi_surface
, FALSE
, &src_hdc
)))
42 WARN("GetDC() failed, %#lx.\n", hr
);
46 BitBlt(render_target
->hdc
, dst_rect
->left
, dst_rect
->top
, dst_rect
->right
- dst_rect
->left
,
47 dst_rect
->bottom
- dst_rect
->top
, src_hdc
, 0, 0, SRCCOPY
);
49 SetRectEmpty(&empty_rect
);
50 IDXGISurface1_ReleaseDC(render_target
->dxgi_surface
, &empty_rect
);
55 static inline struct d2d_dc_render_target
*impl_from_ID2D1DCRenderTarget(ID2D1DCRenderTarget
*iface
)
57 return CONTAINING_RECORD(iface
, struct d2d_dc_render_target
, ID2D1DCRenderTarget_iface
);
60 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_QueryInterface(ID2D1DCRenderTarget
*iface
, REFIID iid
, void **out
)
62 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
64 TRACE("iface %p, iid %s, out %p.\n", iface
, debugstr_guid(iid
), out
);
66 if (IsEqualGUID(iid
, &IID_ID2D1DCRenderTarget
)
67 || IsEqualGUID(iid
, &IID_ID2D1RenderTarget
)
68 || IsEqualGUID(iid
, &IID_ID2D1Resource
)
69 || IsEqualGUID(iid
, &IID_IUnknown
))
71 ID2D1DCRenderTarget_AddRef(iface
);
76 return IUnknown_QueryInterface(render_target
->dxgi_inner
, iid
, out
);
79 static ULONG STDMETHODCALLTYPE
d2d_dc_render_target_AddRef(ID2D1DCRenderTarget
*iface
)
81 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
82 ULONG refcount
= InterlockedIncrement(&render_target
->refcount
);
84 TRACE("%p increasing refcount to %lu.\n", iface
, refcount
);
89 static ULONG STDMETHODCALLTYPE
d2d_dc_render_target_Release(ID2D1DCRenderTarget
*iface
)
91 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
92 ULONG refcount
= InterlockedDecrement(&render_target
->refcount
);
94 TRACE("%p decreasing refcount to %lu.\n", iface
, refcount
);
98 IUnknown_Release(render_target
->dxgi_inner
);
99 if (render_target
->dxgi_surface
)
100 IDXGISurface1_Release(render_target
->dxgi_surface
);
101 ID3D10Device1_Release(render_target
->d3d_device
);
108 static void STDMETHODCALLTYPE
d2d_dc_render_target_GetFactory(ID2D1DCRenderTarget
*iface
, ID2D1Factory
**factory
)
110 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
112 TRACE("iface %p, factory %p.\n", iface
, factory
);
114 ID2D1RenderTarget_GetFactory(render_target
->dxgi_target
, factory
);
117 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateBitmap(ID2D1DCRenderTarget
*iface
,
118 D2D1_SIZE_U size
, const void *src_data
, UINT32 pitch
, const D2D1_BITMAP_PROPERTIES
*desc
, ID2D1Bitmap
**bitmap
)
120 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
122 TRACE("iface %p, size {%u, %u}, src_data %p, pitch %u, desc %p, bitmap %p.\n",
123 iface
, size
.width
, size
.height
, src_data
, pitch
, desc
, bitmap
);
125 return ID2D1RenderTarget_CreateBitmap(render_target
->dxgi_target
, size
, src_data
, pitch
, desc
, bitmap
);
128 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateBitmapFromWicBitmap(ID2D1DCRenderTarget
*iface
,
129 IWICBitmapSource
*bitmap_source
, const D2D1_BITMAP_PROPERTIES
*desc
, ID2D1Bitmap
**bitmap
)
131 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
133 TRACE("iface %p, bitmap_source %p, desc %p, bitmap %p.\n",
134 iface
, bitmap_source
, desc
, bitmap
);
136 return ID2D1RenderTarget_CreateBitmapFromWicBitmap(render_target
->dxgi_target
, bitmap_source
, desc
, bitmap
);
139 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateSharedBitmap(ID2D1DCRenderTarget
*iface
,
140 REFIID iid
, void *data
, const D2D1_BITMAP_PROPERTIES
*desc
, ID2D1Bitmap
**bitmap
)
142 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
144 TRACE("iface %p, iid %s, data %p, desc %p, bitmap %p.\n",
145 iface
, debugstr_guid(iid
), data
, desc
, bitmap
);
147 return ID2D1RenderTarget_CreateSharedBitmap(render_target
->dxgi_target
, iid
, data
, desc
, bitmap
);
150 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateBitmapBrush(ID2D1DCRenderTarget
*iface
,
151 ID2D1Bitmap
*bitmap
, const D2D1_BITMAP_BRUSH_PROPERTIES
*bitmap_brush_desc
,
152 const D2D1_BRUSH_PROPERTIES
*brush_desc
, ID2D1BitmapBrush
**brush
)
154 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
156 TRACE("iface %p, bitmap %p, bitmap_brush_desc %p, brush_desc %p, brush %p.\n",
157 iface
, bitmap
, bitmap_brush_desc
, brush_desc
, brush
);
159 return ID2D1RenderTarget_CreateBitmapBrush(render_target
->dxgi_target
,
160 bitmap
, bitmap_brush_desc
, brush_desc
, brush
);
163 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateSolidColorBrush(ID2D1DCRenderTarget
*iface
,
164 const D2D1_COLOR_F
*color
, const D2D1_BRUSH_PROPERTIES
*desc
, ID2D1SolidColorBrush
**brush
)
166 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
168 TRACE("iface %p, color %p, desc %p, brush %p.\n", iface
, color
, desc
, brush
);
170 return ID2D1RenderTarget_CreateSolidColorBrush(render_target
->dxgi_target
, color
, desc
, brush
);
173 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateGradientStopCollection(ID2D1DCRenderTarget
*iface
,
174 const D2D1_GRADIENT_STOP
*stops
, UINT32 stop_count
, D2D1_GAMMA gamma
, D2D1_EXTEND_MODE extend_mode
,
175 ID2D1GradientStopCollection
**gradient
)
177 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
179 TRACE("iface %p, stops %p, stop_count %u, gamma %#x, extend_mode %#x, gradient %p.\n",
180 iface
, stops
, stop_count
, gamma
, extend_mode
, gradient
);
182 return ID2D1RenderTarget_CreateGradientStopCollection(render_target
->dxgi_target
,
183 stops
, stop_count
, gamma
, extend_mode
, gradient
);
186 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateLinearGradientBrush(ID2D1DCRenderTarget
*iface
,
187 const D2D1_LINEAR_GRADIENT_BRUSH_PROPERTIES
*gradient_brush_desc
, const D2D1_BRUSH_PROPERTIES
*brush_desc
,
188 ID2D1GradientStopCollection
*gradient
, ID2D1LinearGradientBrush
**brush
)
190 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
192 TRACE("iface %p, gradient_brush_desc %p, brush_desc %p, gradient %p, brush %p.\n",
193 iface
, gradient_brush_desc
, brush_desc
, gradient
, brush
);
195 return ID2D1RenderTarget_CreateLinearGradientBrush(render_target
->dxgi_target
,
196 gradient_brush_desc
, brush_desc
, gradient
, brush
);
199 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateRadialGradientBrush(ID2D1DCRenderTarget
*iface
,
200 const D2D1_RADIAL_GRADIENT_BRUSH_PROPERTIES
*gradient_brush_desc
, const D2D1_BRUSH_PROPERTIES
*brush_desc
,
201 ID2D1GradientStopCollection
*gradient
, ID2D1RadialGradientBrush
**brush
)
203 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
205 TRACE("iface %p, gradient_brush_desc %p, brush_desc %p, gradient %p, brush %p.\n",
206 iface
, gradient_brush_desc
, brush_desc
, gradient
, brush
);
208 return ID2D1RenderTarget_CreateRadialGradientBrush(render_target
->dxgi_target
,
209 gradient_brush_desc
, brush_desc
, gradient
, brush
);
212 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateCompatibleRenderTarget(ID2D1DCRenderTarget
*iface
,
213 const D2D1_SIZE_F
*size
, const D2D1_SIZE_U
*pixel_size
, const D2D1_PIXEL_FORMAT
*format
,
214 D2D1_COMPATIBLE_RENDER_TARGET_OPTIONS options
, ID2D1BitmapRenderTarget
**render_target
)
216 struct d2d_dc_render_target
*rt
= impl_from_ID2D1DCRenderTarget(iface
);
218 TRACE("iface %p, size %p, pixel_size %p, format %p, options %#x, render_target %p,\n",
219 iface
, size
, pixel_size
, format
, options
, render_target
);
221 return ID2D1RenderTarget_CreateCompatibleRenderTarget(rt
->dxgi_target
,
222 size
, pixel_size
, format
, options
, render_target
);
225 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateLayer(ID2D1DCRenderTarget
*iface
,
226 const D2D1_SIZE_F
*size
, ID2D1Layer
**layer
)
228 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
230 TRACE("iface %p, size %p, layer %p.\n", iface
, size
, layer
);
232 return ID2D1RenderTarget_CreateLayer(render_target
->dxgi_target
, size
, layer
);
235 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateMesh(ID2D1DCRenderTarget
*iface
, ID2D1Mesh
**mesh
)
237 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
239 TRACE("iface %p, mesh %p.\n", iface
, mesh
);
241 return ID2D1RenderTarget_CreateMesh(render_target
->dxgi_target
, mesh
);
244 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawLine(ID2D1DCRenderTarget
*iface
,
245 D2D1_POINT_2F p0
, D2D1_POINT_2F p1
, ID2D1Brush
*brush
, float stroke_width
, ID2D1StrokeStyle
*stroke_style
)
247 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
249 TRACE("iface %p, p0 %s, p1 %s, brush %p, stroke_width %.8e, stroke_style %p.\n",
250 iface
, debug_d2d_point_2f(&p0
), debug_d2d_point_2f(&p1
), brush
, stroke_width
, stroke_style
);
252 ID2D1RenderTarget_DrawLine(render_target
->dxgi_target
, p0
, p1
, brush
, stroke_width
, stroke_style
);
255 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawRectangle(ID2D1DCRenderTarget
*iface
,
256 const D2D1_RECT_F
*rect
, ID2D1Brush
*brush
, float stroke_width
, ID2D1StrokeStyle
*stroke_style
)
258 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
260 TRACE("iface %p, rect %s, brush %p, stroke_width %.8e, stroke_style %p.\n",
261 iface
, debug_d2d_rect_f(rect
), brush
, stroke_width
, stroke_style
);
263 ID2D1RenderTarget_DrawRectangle(render_target
->dxgi_target
, rect
, brush
, stroke_width
, stroke_style
);
266 static void STDMETHODCALLTYPE
d2d_dc_render_target_FillRectangle(ID2D1DCRenderTarget
*iface
,
267 const D2D1_RECT_F
*rect
, ID2D1Brush
*brush
)
269 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
271 TRACE("iface %p, rect %s, brush %p.\n", iface
, debug_d2d_rect_f(rect
), brush
);
273 ID2D1RenderTarget_FillRectangle(render_target
->dxgi_target
, rect
, brush
);
276 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawRoundedRectangle(ID2D1DCRenderTarget
*iface
,
277 const D2D1_ROUNDED_RECT
*rect
, ID2D1Brush
*brush
, float stroke_width
, ID2D1StrokeStyle
*stroke_style
)
279 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
281 TRACE("iface %p, rect %p, brush %p, stroke_width %.8e, stroke_style %p.\n",
282 iface
, rect
, brush
, stroke_width
, stroke_style
);
284 ID2D1RenderTarget_DrawRoundedRectangle(render_target
->dxgi_target
, rect
, brush
, stroke_width
, stroke_style
);
287 static void STDMETHODCALLTYPE
d2d_dc_render_target_FillRoundedRectangle(ID2D1DCRenderTarget
*iface
,
288 const D2D1_ROUNDED_RECT
*rect
, ID2D1Brush
*brush
)
290 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
292 TRACE("iface %p, rect %p, brush %p.\n", iface
, rect
, brush
);
294 ID2D1RenderTarget_FillRoundedRectangle(render_target
->dxgi_target
, rect
, brush
);
297 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawEllipse(ID2D1DCRenderTarget
*iface
,
298 const D2D1_ELLIPSE
*ellipse
, ID2D1Brush
*brush
, float stroke_width
, ID2D1StrokeStyle
*stroke_style
)
300 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
302 TRACE("iface %p, ellipse %p, brush %p, stroke_width %.8e, stroke_style %p.\n",
303 iface
, ellipse
, brush
, stroke_width
, stroke_style
);
305 ID2D1RenderTarget_DrawEllipse(render_target
->dxgi_target
, ellipse
, brush
, stroke_width
, stroke_style
);
308 static void STDMETHODCALLTYPE
d2d_dc_render_target_FillEllipse(ID2D1DCRenderTarget
*iface
,
309 const D2D1_ELLIPSE
*ellipse
, ID2D1Brush
*brush
)
311 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
313 TRACE("iface %p, ellipse %p, brush %p.\n", iface
, ellipse
, brush
);
315 ID2D1RenderTarget_FillEllipse(render_target
->dxgi_target
, ellipse
, brush
);
318 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawGeometry(ID2D1DCRenderTarget
*iface
,
319 ID2D1Geometry
*geometry
, ID2D1Brush
*brush
, float stroke_width
, ID2D1StrokeStyle
*stroke_style
)
321 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
323 TRACE("iface %p, geometry %p, brush %p, stroke_width %.8e, stroke_style %p.\n",
324 iface
, geometry
, brush
, stroke_width
, stroke_style
);
326 ID2D1RenderTarget_DrawGeometry(render_target
->dxgi_target
, geometry
, brush
, stroke_width
, stroke_style
);
329 static void STDMETHODCALLTYPE
d2d_dc_render_target_FillGeometry(ID2D1DCRenderTarget
*iface
,
330 ID2D1Geometry
*geometry
, ID2D1Brush
*brush
, ID2D1Brush
*opacity_brush
)
332 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
334 TRACE("iface %p, geometry %p, brush %p, opacity_brush %p.\n", iface
, geometry
, brush
, opacity_brush
);
336 ID2D1RenderTarget_FillGeometry(render_target
->dxgi_target
, geometry
, brush
, opacity_brush
);
339 static void STDMETHODCALLTYPE
d2d_dc_render_target_FillMesh(ID2D1DCRenderTarget
*iface
,
340 ID2D1Mesh
*mesh
, ID2D1Brush
*brush
)
342 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
344 TRACE("iface %p, mesh %p, brush %p.\n", iface
, mesh
, brush
);
346 ID2D1RenderTarget_FillMesh(render_target
->dxgi_target
, mesh
, brush
);
349 static void STDMETHODCALLTYPE
d2d_dc_render_target_FillOpacityMask(ID2D1DCRenderTarget
*iface
,
350 ID2D1Bitmap
*mask
, ID2D1Brush
*brush
, D2D1_OPACITY_MASK_CONTENT content
,
351 const D2D1_RECT_F
*dst_rect
, const D2D1_RECT_F
*src_rect
)
353 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
355 TRACE("iface %p, mask %p, brush %p, content %#x, dst_rect %s, src_rect %s.\n",
356 iface
, mask
, brush
, content
, debug_d2d_rect_f(dst_rect
), debug_d2d_rect_f(src_rect
));
358 ID2D1RenderTarget_FillOpacityMask(render_target
->dxgi_target
,
359 mask
, brush
, content
, dst_rect
, src_rect
);
362 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawBitmap(ID2D1DCRenderTarget
*iface
,
363 ID2D1Bitmap
*bitmap
, const D2D1_RECT_F
*dst_rect
, float opacity
,
364 D2D1_BITMAP_INTERPOLATION_MODE interpolation_mode
, const D2D1_RECT_F
*src_rect
)
366 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
368 TRACE("iface %p, bitmap %p, dst_rect %s, opacity %.8e, interpolation_mode %#x, src_rect %s.\n",
369 iface
, bitmap
, debug_d2d_rect_f(dst_rect
), opacity
, interpolation_mode
, debug_d2d_rect_f(src_rect
));
371 ID2D1RenderTarget_DrawBitmap(render_target
->dxgi_target
,
372 bitmap
, dst_rect
, opacity
, interpolation_mode
, src_rect
);
375 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawText(ID2D1DCRenderTarget
*iface
,
376 const WCHAR
*string
, UINT32 string_len
, IDWriteTextFormat
*text_format
, const D2D1_RECT_F
*layout_rect
,
377 ID2D1Brush
*brush
, D2D1_DRAW_TEXT_OPTIONS options
, DWRITE_MEASURING_MODE measuring_mode
)
379 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
381 TRACE("iface %p, string %s, string_len %u, text_format %p, layout_rect %s, "
382 "brush %p, options %#x, measuring_mode %#x.\n",
383 iface
, debugstr_wn(string
, string_len
), string_len
, text_format
, debug_d2d_rect_f(layout_rect
),
384 brush
, options
, measuring_mode
);
386 ID2D1RenderTarget_DrawText(render_target
->dxgi_target
, string
, string_len
,
387 text_format
, layout_rect
, brush
, options
, measuring_mode
);
390 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawTextLayout(ID2D1DCRenderTarget
*iface
,
391 D2D1_POINT_2F origin
, IDWriteTextLayout
*layout
, ID2D1Brush
*brush
, D2D1_DRAW_TEXT_OPTIONS options
)
393 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
395 TRACE("iface %p, origin %s, layout %p, brush %p, options %#x.\n",
396 iface
, debug_d2d_point_2f(&origin
), layout
, brush
, options
);
398 ID2D1RenderTarget_DrawTextLayout(render_target
->dxgi_target
, origin
, layout
, brush
, options
);
401 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawGlyphRun(ID2D1DCRenderTarget
*iface
,
402 D2D1_POINT_2F baseline_origin
, const DWRITE_GLYPH_RUN
*glyph_run
, ID2D1Brush
*brush
,
403 DWRITE_MEASURING_MODE measuring_mode
)
405 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
407 TRACE("iface %p, baseline_origin %s, glyph_run %p, brush %p, measuring_mode %#x.\n",
408 iface
, debug_d2d_point_2f(&baseline_origin
), glyph_run
, brush
, measuring_mode
);
410 ID2D1RenderTarget_DrawGlyphRun(render_target
->dxgi_target
,
411 baseline_origin
, glyph_run
, brush
, measuring_mode
);
414 static void STDMETHODCALLTYPE
d2d_dc_render_target_SetTransform(ID2D1DCRenderTarget
*iface
,
415 const D2D1_MATRIX_3X2_F
*transform
)
417 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
419 TRACE("iface %p, transform %p.\n", iface
, transform
);
421 ID2D1RenderTarget_SetTransform(render_target
->dxgi_target
, transform
);
424 static void STDMETHODCALLTYPE
d2d_dc_render_target_GetTransform(ID2D1DCRenderTarget
*iface
,
425 D2D1_MATRIX_3X2_F
*transform
)
427 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
429 TRACE("iface %p, transform %p.\n", iface
, transform
);
431 ID2D1RenderTarget_GetTransform(render_target
->dxgi_target
, transform
);
434 static void STDMETHODCALLTYPE
d2d_dc_render_target_SetAntialiasMode(ID2D1DCRenderTarget
*iface
,
435 D2D1_ANTIALIAS_MODE antialias_mode
)
437 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
439 TRACE("iface %p, antialias_mode %#x.\n", iface
, antialias_mode
);
441 ID2D1RenderTarget_SetAntialiasMode(render_target
->dxgi_target
, antialias_mode
);
444 static D2D1_ANTIALIAS_MODE STDMETHODCALLTYPE
d2d_dc_render_target_GetAntialiasMode(ID2D1DCRenderTarget
*iface
)
446 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
448 TRACE("iface %p.\n", iface
);
450 return ID2D1RenderTarget_GetAntialiasMode(render_target
->dxgi_target
);
453 static void STDMETHODCALLTYPE
d2d_dc_render_target_SetTextAntialiasMode(ID2D1DCRenderTarget
*iface
,
454 D2D1_TEXT_ANTIALIAS_MODE antialias_mode
)
456 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
458 TRACE("iface %p, antialias_mode %#x.\n", iface
, antialias_mode
);
460 ID2D1RenderTarget_SetTextAntialiasMode(render_target
->dxgi_target
, antialias_mode
);
463 static D2D1_TEXT_ANTIALIAS_MODE STDMETHODCALLTYPE
d2d_dc_render_target_GetTextAntialiasMode(ID2D1DCRenderTarget
*iface
)
465 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
467 TRACE("iface %p.\n", iface
);
469 return ID2D1RenderTarget_GetTextAntialiasMode(render_target
->dxgi_target
);
472 static void STDMETHODCALLTYPE
d2d_dc_render_target_SetTextRenderingParams(ID2D1DCRenderTarget
*iface
,
473 IDWriteRenderingParams
*text_rendering_params
)
475 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
477 TRACE("iface %p, text_rendering_params %p.\n", iface
, text_rendering_params
);
479 ID2D1RenderTarget_SetTextRenderingParams(render_target
->dxgi_target
, text_rendering_params
);
482 static void STDMETHODCALLTYPE
d2d_dc_render_target_GetTextRenderingParams(ID2D1DCRenderTarget
*iface
,
483 IDWriteRenderingParams
**text_rendering_params
)
485 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
487 TRACE("iface %p, text_rendering_params %p.\n", iface
, text_rendering_params
);
489 ID2D1RenderTarget_GetTextRenderingParams(render_target
->dxgi_target
, text_rendering_params
);
492 static void STDMETHODCALLTYPE
d2d_dc_render_target_SetTags(ID2D1DCRenderTarget
*iface
, D2D1_TAG tag1
, D2D1_TAG tag2
)
494 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
496 TRACE("iface %p, tag1 %s, tag2 %s.\n", iface
, wine_dbgstr_longlong(tag1
), wine_dbgstr_longlong(tag2
));
498 ID2D1RenderTarget_SetTags(render_target
->dxgi_target
, tag1
, tag2
);
501 static void STDMETHODCALLTYPE
d2d_dc_render_target_GetTags(ID2D1DCRenderTarget
*iface
, D2D1_TAG
*tag1
, D2D1_TAG
*tag2
)
503 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
505 TRACE("iface %p, tag1 %p, tag2 %p.\n", iface
, tag1
, tag2
);
507 ID2D1RenderTarget_GetTags(render_target
->dxgi_target
, tag1
, tag2
);
510 static void STDMETHODCALLTYPE
d2d_dc_render_target_PushLayer(ID2D1DCRenderTarget
*iface
,
511 const D2D1_LAYER_PARAMETERS
*layer_parameters
, ID2D1Layer
*layer
)
513 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
515 TRACE("iface %p, layer_parameters %p, layer %p.\n", iface
, layer_parameters
, layer
);
517 ID2D1RenderTarget_PushLayer(render_target
->dxgi_target
, layer_parameters
, layer
);
520 static void STDMETHODCALLTYPE
d2d_dc_render_target_PopLayer(ID2D1DCRenderTarget
*iface
)
522 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
524 TRACE("iface %p.\n", iface
);
526 ID2D1RenderTarget_PopLayer(render_target
->dxgi_target
);
529 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_Flush(ID2D1DCRenderTarget
*iface
, D2D1_TAG
*tag1
, D2D1_TAG
*tag2
)
531 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
533 TRACE("iface %p, tag1 %p, tag2 %p.\n", iface
, tag1
, tag2
);
535 return ID2D1RenderTarget_Flush(render_target
->dxgi_target
, tag1
, tag2
);
538 static void STDMETHODCALLTYPE
d2d_dc_render_target_SaveDrawingState(ID2D1DCRenderTarget
*iface
,
539 ID2D1DrawingStateBlock
*state_block
)
541 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
543 TRACE("iface %p, state_block %p.\n", iface
, state_block
);
545 ID2D1RenderTarget_SaveDrawingState(render_target
->dxgi_target
, state_block
);
548 static void STDMETHODCALLTYPE
d2d_dc_render_target_RestoreDrawingState(ID2D1DCRenderTarget
*iface
,
549 ID2D1DrawingStateBlock
*state_block
)
551 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
553 TRACE("iface %p, state_block %p.\n", iface
, state_block
);
555 ID2D1RenderTarget_RestoreDrawingState(render_target
->dxgi_target
, state_block
);
558 static void STDMETHODCALLTYPE
d2d_dc_render_target_PushAxisAlignedClip(ID2D1DCRenderTarget
*iface
,
559 const D2D1_RECT_F
*clip_rect
, D2D1_ANTIALIAS_MODE antialias_mode
)
561 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
563 TRACE("iface %p, clip_rect %s, antialias_mode %#x.\n", iface
, debug_d2d_rect_f(clip_rect
), antialias_mode
);
565 ID2D1RenderTarget_PushAxisAlignedClip(render_target
->dxgi_target
, clip_rect
, antialias_mode
);
568 static void STDMETHODCALLTYPE
d2d_dc_render_target_PopAxisAlignedClip(ID2D1DCRenderTarget
*iface
)
570 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
572 TRACE("iface %p.\n", iface
);
574 ID2D1RenderTarget_PopAxisAlignedClip(render_target
->dxgi_target
);
577 static void STDMETHODCALLTYPE
d2d_dc_render_target_Clear(ID2D1DCRenderTarget
*iface
, const D2D1_COLOR_F
*color
)
579 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
581 TRACE("iface %p, color %p.\n", iface
, color
);
583 ID2D1RenderTarget_Clear(render_target
->dxgi_target
, color
);
586 static void STDMETHODCALLTYPE
d2d_dc_render_target_BeginDraw(ID2D1DCRenderTarget
*iface
)
588 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
590 TRACE("iface %p.\n", iface
);
592 ID2D1RenderTarget_BeginDraw(render_target
->dxgi_target
);
595 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_EndDraw(ID2D1DCRenderTarget
*iface
,
596 D2D1_TAG
*tag1
, D2D1_TAG
*tag2
)
598 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
600 TRACE("iface %p, tag1 %p, tag2 %p.\n", iface
, tag1
, tag2
);
602 return ID2D1RenderTarget_EndDraw(render_target
->dxgi_target
, tag1
, tag2
);
605 static D2D1_PIXEL_FORMAT
* STDMETHODCALLTYPE
d2d_dc_render_target_GetPixelFormat(ID2D1DCRenderTarget
*iface
,
606 D2D1_PIXEL_FORMAT
*format
)
608 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
610 TRACE("iface %p, format %p.\n", iface
, format
);
612 *format
= ID2D1RenderTarget_GetPixelFormat(render_target
->dxgi_target
);
616 static void STDMETHODCALLTYPE
d2d_dc_render_target_SetDpi(ID2D1DCRenderTarget
*iface
, float dpi_x
, float dpi_y
)
618 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
620 TRACE("iface %p, dpi_x %.8e, dpi_y %.8e.\n", iface
, dpi_x
, dpi_y
);
622 ID2D1RenderTarget_SetDpi(render_target
->dxgi_target
, dpi_x
, dpi_y
);
625 static void STDMETHODCALLTYPE
d2d_dc_render_target_GetDpi(ID2D1DCRenderTarget
*iface
, float *dpi_x
, float *dpi_y
)
627 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
629 TRACE("iface %p, dpi_x %p, dpi_y %p.\n", iface
, dpi_x
, dpi_y
);
631 ID2D1RenderTarget_GetDpi(render_target
->dxgi_target
, dpi_x
, dpi_y
);
634 static D2D1_SIZE_F
* STDMETHODCALLTYPE
d2d_dc_render_target_GetSize(ID2D1DCRenderTarget
*iface
, D2D1_SIZE_F
*size
)
636 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
638 TRACE("iface %p, size %p.\n", iface
, size
);
640 if (render_target
->hdc
)
641 *size
= ID2D1RenderTarget_GetSize(render_target
->dxgi_target
);
643 size
->width
= size
->height
= 0.0f
;
648 static D2D1_SIZE_U
* STDMETHODCALLTYPE
d2d_dc_render_target_GetPixelSize(ID2D1DCRenderTarget
*iface
,
649 D2D1_SIZE_U
*pixel_size
)
651 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
653 TRACE("iface %p, pixel_size %p.\n", iface
, pixel_size
);
655 if (render_target
->hdc
)
656 *pixel_size
= ID2D1RenderTarget_GetPixelSize(render_target
->dxgi_target
);
658 pixel_size
->width
= pixel_size
->height
= 0;
663 static UINT32 STDMETHODCALLTYPE
d2d_dc_render_target_GetMaximumBitmapSize(ID2D1DCRenderTarget
*iface
)
665 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
667 TRACE("iface %p.\n", iface
);
669 return ID2D1RenderTarget_GetMaximumBitmapSize(render_target
->dxgi_target
);
672 static BOOL STDMETHODCALLTYPE
d2d_dc_render_target_IsSupported(ID2D1DCRenderTarget
*iface
,
673 const D2D1_RENDER_TARGET_PROPERTIES
*desc
)
675 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
677 TRACE("iface %p, desc %p.\n", iface
, desc
);
679 return ID2D1RenderTarget_IsSupported(render_target
->dxgi_target
, desc
);
682 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_BindDC(ID2D1DCRenderTarget
*iface
,
683 const HDC hdc
, const RECT
*rect
)
685 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
686 D2D1_BITMAP_PROPERTIES1 bitmap_desc
;
687 struct d2d_bitmap
*bitmap_impl
;
688 IDXGISurface1
*dxgi_surface
;
689 ID2D1DeviceContext
*context
;
690 D2D1_SIZE_U bitmap_size
;
695 TRACE("iface %p, hdc %p, rect %s.\n", iface
, hdc
, wine_dbgstr_rect(rect
));
697 obj_type
= GetObjectType(hdc
);
698 if (obj_type
!= OBJ_DC
&& obj_type
!= OBJ_ENHMETADC
&& obj_type
!= OBJ_MEMDC
)
701 /* Switch dxgi target to new surface. */
702 ID2D1RenderTarget_QueryInterface(render_target
->dxgi_target
, &IID_ID2D1DeviceContext
, (void **)&context
);
704 bitmap_size
.width
= rect
->right
- rect
->left
;
705 bitmap_size
.height
= rect
->bottom
- rect
->top
;
707 memset(&bitmap_desc
, 0, sizeof(bitmap_desc
));
708 bitmap_desc
.pixelFormat
= render_target
->pixel_format
;
709 bitmap_desc
.bitmapOptions
= D2D1_BITMAP_OPTIONS_TARGET
| D2D1_BITMAP_OPTIONS_CANNOT_DRAW
|
710 D2D1_BITMAP_OPTIONS_GDI_COMPATIBLE
;
711 if (FAILED(hr
= ID2D1DeviceContext_CreateBitmap(context
, bitmap_size
, NULL
, 0, &bitmap_desc
,
712 (ID2D1Bitmap1
**)&bitmap
)))
714 WARN("Failed to create target bitmap, hr %#lx.\n", hr
);
715 ID2D1DeviceContext_Release(context
);
719 bitmap_impl
= unsafe_impl_from_ID2D1Bitmap(bitmap
);
720 ID3D11Resource_QueryInterface(bitmap_impl
->resource
, &IID_IDXGISurface1
, (void **)&dxgi_surface
);
722 ID2D1DeviceContext_SetTarget(context
, (ID2D1Image
*)bitmap
);
723 ID2D1Bitmap_Release(bitmap
);
724 ID2D1DeviceContext_Release(context
);
726 if (render_target
->dxgi_surface
)
727 IDXGISurface1_Release(render_target
->dxgi_surface
);
728 render_target
->dxgi_surface
= dxgi_surface
;
730 render_target
->hdc
= hdc
;
731 render_target
->dst_rect
= *rect
;
736 static const struct ID2D1DCRenderTargetVtbl d2d_dc_render_target_vtbl
=
738 d2d_dc_render_target_QueryInterface
,
739 d2d_dc_render_target_AddRef
,
740 d2d_dc_render_target_Release
,
741 d2d_dc_render_target_GetFactory
,
742 d2d_dc_render_target_CreateBitmap
,
743 d2d_dc_render_target_CreateBitmapFromWicBitmap
,
744 d2d_dc_render_target_CreateSharedBitmap
,
745 d2d_dc_render_target_CreateBitmapBrush
,
746 d2d_dc_render_target_CreateSolidColorBrush
,
747 d2d_dc_render_target_CreateGradientStopCollection
,
748 d2d_dc_render_target_CreateLinearGradientBrush
,
749 d2d_dc_render_target_CreateRadialGradientBrush
,
750 d2d_dc_render_target_CreateCompatibleRenderTarget
,
751 d2d_dc_render_target_CreateLayer
,
752 d2d_dc_render_target_CreateMesh
,
753 d2d_dc_render_target_DrawLine
,
754 d2d_dc_render_target_DrawRectangle
,
755 d2d_dc_render_target_FillRectangle
,
756 d2d_dc_render_target_DrawRoundedRectangle
,
757 d2d_dc_render_target_FillRoundedRectangle
,
758 d2d_dc_render_target_DrawEllipse
,
759 d2d_dc_render_target_FillEllipse
,
760 d2d_dc_render_target_DrawGeometry
,
761 d2d_dc_render_target_FillGeometry
,
762 d2d_dc_render_target_FillMesh
,
763 d2d_dc_render_target_FillOpacityMask
,
764 d2d_dc_render_target_DrawBitmap
,
765 d2d_dc_render_target_DrawText
,
766 d2d_dc_render_target_DrawTextLayout
,
767 d2d_dc_render_target_DrawGlyphRun
,
768 d2d_dc_render_target_SetTransform
,
769 d2d_dc_render_target_GetTransform
,
770 d2d_dc_render_target_SetAntialiasMode
,
771 d2d_dc_render_target_GetAntialiasMode
,
772 d2d_dc_render_target_SetTextAntialiasMode
,
773 d2d_dc_render_target_GetTextAntialiasMode
,
774 d2d_dc_render_target_SetTextRenderingParams
,
775 d2d_dc_render_target_GetTextRenderingParams
,
776 d2d_dc_render_target_SetTags
,
777 d2d_dc_render_target_GetTags
,
778 d2d_dc_render_target_PushLayer
,
779 d2d_dc_render_target_PopLayer
,
780 d2d_dc_render_target_Flush
,
781 d2d_dc_render_target_SaveDrawingState
,
782 d2d_dc_render_target_RestoreDrawingState
,
783 d2d_dc_render_target_PushAxisAlignedClip
,
784 d2d_dc_render_target_PopAxisAlignedClip
,
785 d2d_dc_render_target_Clear
,
786 d2d_dc_render_target_BeginDraw
,
787 d2d_dc_render_target_EndDraw
,
788 d2d_dc_render_target_GetPixelFormat
,
789 d2d_dc_render_target_SetDpi
,
790 d2d_dc_render_target_GetDpi
,
791 d2d_dc_render_target_GetSize
,
792 d2d_dc_render_target_GetPixelSize
,
793 d2d_dc_render_target_GetMaximumBitmapSize
,
794 d2d_dc_render_target_IsSupported
,
795 d2d_dc_render_target_BindDC
,
798 static const struct d2d_device_context_ops d2d_dc_render_target_ops
=
800 d2d_dc_render_target_present
,
803 HRESULT
d2d_dc_render_target_init(struct d2d_dc_render_target
*render_target
, ID2D1Factory1
*factory
,
804 ID3D10Device1
*d3d_device
, const D2D1_RENDER_TARGET_PROPERTIES
*desc
)
806 IDXGIDevice
*dxgi_device
;
810 render_target
->ID2D1DCRenderTarget_iface
.lpVtbl
= &d2d_dc_render_target_vtbl
;
812 /* Set with BindDC(). */
813 SetRectEmpty(&render_target
->dst_rect
);
814 render_target
->hdc
= NULL
;
816 render_target
->pixel_format
= desc
->pixelFormat
;
817 switch (desc
->pixelFormat
.format
)
819 case DXGI_FORMAT_B8G8R8A8_UNORM
:
820 if (desc
->pixelFormat
.alphaMode
== D2D1_ALPHA_MODE_PREMULTIPLIED
821 || desc
->pixelFormat
.alphaMode
== D2D1_ALPHA_MODE_IGNORE
)
825 FIXME("Unhandled format %#x, alpha mode %u.\n", desc
->pixelFormat
.format
, desc
->pixelFormat
.alphaMode
);
826 return D2DERR_UNSUPPORTED_PIXEL_FORMAT
;
829 if (FAILED(hr
= ID3D10Device1_QueryInterface(d3d_device
, &IID_IDXGIDevice
, (void **)&dxgi_device
)))
831 WARN("Failed to get DXGI device interface, hr %#lx.\n", hr
);
835 hr
= ID2D1Factory1_CreateDevice(factory
, dxgi_device
, &device
);
836 IDXGIDevice_Release(dxgi_device
);
839 WARN("Failed to create D2D device, hr %#lx.\n", hr
);
843 hr
= d2d_d3d_create_render_target(device
, NULL
, (IUnknown
*)&render_target
->ID2D1DCRenderTarget_iface
,
844 &d2d_dc_render_target_ops
, desc
, (void **)&render_target
->dxgi_inner
);
845 ID2D1Device_Release(device
);
848 WARN("Failed to create DXGI surface render target, hr %#lx.\n", hr
);
852 if (FAILED(hr
= IUnknown_QueryInterface(render_target
->dxgi_inner
,
853 &IID_ID2D1RenderTarget
, (void **)&render_target
->dxgi_target
)))
855 WARN("Failed to retrieve ID2D1RenderTarget interface, hr %#lx.\n", hr
);
856 IUnknown_Release(render_target
->dxgi_inner
);
860 render_target
->d3d_device
= d3d_device
;
861 ID3D10Device1_AddRef(render_target
->d3d_device
);