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
21 #include "wine/port.h"
23 #include "d2d1_private.h"
25 WINE_DEFAULT_DEBUG_CHANNEL(d2d
);
27 static inline struct d2d_dc_render_target
*impl_from_IUnknown(IUnknown
*iface
)
29 return CONTAINING_RECORD(iface
, struct d2d_dc_render_target
, ID2D1DCRenderTarget_iface
);
32 static HRESULT
d2d_dc_render_target_present(IUnknown
*outer_unknown
)
34 struct d2d_dc_render_target
*render_target
= impl_from_IUnknown(outer_unknown
);
35 const RECT
*dst_rect
= &render_target
->dst_rect
;
40 if (!render_target
->hdc
)
41 return D2DERR_WRONG_STATE
;
43 if (FAILED(hr
= IDXGISurface1_GetDC(render_target
->dxgi_surface
, FALSE
, &src_hdc
)))
45 WARN("GetDC() failed, %#x.\n", hr
);
49 BitBlt(render_target
->hdc
, dst_rect
->left
, dst_rect
->top
, dst_rect
->right
- dst_rect
->left
,
50 dst_rect
->bottom
- dst_rect
->top
, src_hdc
, 0, 0, SRCCOPY
);
52 SetRectEmpty(&empty_rect
);
53 IDXGISurface1_ReleaseDC(render_target
->dxgi_surface
, &empty_rect
);
58 static inline struct d2d_dc_render_target
*impl_from_ID2D1DCRenderTarget(ID2D1DCRenderTarget
*iface
)
60 return CONTAINING_RECORD(iface
, struct d2d_dc_render_target
, ID2D1DCRenderTarget_iface
);
63 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_QueryInterface(ID2D1DCRenderTarget
*iface
, REFIID iid
, void **out
)
65 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
67 TRACE("iface %p, iid %s, out %p.\n", iface
, debugstr_guid(iid
), out
);
69 if (IsEqualGUID(iid
, &IID_ID2D1DCRenderTarget
)
70 || IsEqualGUID(iid
, &IID_ID2D1RenderTarget
)
71 || IsEqualGUID(iid
, &IID_ID2D1Resource
)
72 || IsEqualGUID(iid
, &IID_IUnknown
))
74 ID2D1DCRenderTarget_AddRef(iface
);
79 return IUnknown_QueryInterface(render_target
->dxgi_inner
, iid
, out
);
82 static ULONG STDMETHODCALLTYPE
d2d_dc_render_target_AddRef(ID2D1DCRenderTarget
*iface
)
84 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
85 ULONG refcount
= InterlockedIncrement(&render_target
->refcount
);
87 TRACE("%p increasing refcount to %u.\n", iface
, refcount
);
92 static ULONG STDMETHODCALLTYPE
d2d_dc_render_target_Release(ID2D1DCRenderTarget
*iface
)
94 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
95 ULONG refcount
= InterlockedDecrement(&render_target
->refcount
);
97 TRACE("%p decreasing refcount to %u.\n", iface
, refcount
);
101 IUnknown_Release(render_target
->dxgi_inner
);
102 IDXGISurface1_Release(render_target
->dxgi_surface
);
103 heap_free(render_target
);
109 static void STDMETHODCALLTYPE
d2d_dc_render_target_GetFactory(ID2D1DCRenderTarget
*iface
, ID2D1Factory
**factory
)
111 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
113 TRACE("iface %p, factory %p.\n", iface
, factory
);
115 ID2D1RenderTarget_GetFactory(render_target
->dxgi_target
, factory
);
118 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateBitmap(ID2D1DCRenderTarget
*iface
,
119 D2D1_SIZE_U size
, const void *src_data
, UINT32 pitch
, const D2D1_BITMAP_PROPERTIES
*desc
, ID2D1Bitmap
**bitmap
)
121 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
123 TRACE("iface %p, size {%u, %u}, src_data %p, pitch %u, desc %p, bitmap %p.\n",
124 iface
, size
.width
, size
.height
, src_data
, pitch
, desc
, bitmap
);
126 return ID2D1RenderTarget_CreateBitmap(render_target
->dxgi_target
, size
, src_data
, pitch
, desc
, bitmap
);
129 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateBitmapFromWicBitmap(ID2D1DCRenderTarget
*iface
,
130 IWICBitmapSource
*bitmap_source
, const D2D1_BITMAP_PROPERTIES
*desc
, ID2D1Bitmap
**bitmap
)
132 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
134 TRACE("iface %p, bitmap_source %p, desc %p, bitmap %p.\n",
135 iface
, bitmap_source
, desc
, bitmap
);
137 return ID2D1RenderTarget_CreateBitmapFromWicBitmap(render_target
->dxgi_target
, bitmap_source
, desc
, bitmap
);
140 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateSharedBitmap(ID2D1DCRenderTarget
*iface
,
141 REFIID iid
, void *data
, const D2D1_BITMAP_PROPERTIES
*desc
, ID2D1Bitmap
**bitmap
)
143 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
145 TRACE("iface %p, iid %s, data %p, desc %p, bitmap %p.\n",
146 iface
, debugstr_guid(iid
), data
, desc
, bitmap
);
148 return ID2D1RenderTarget_CreateSharedBitmap(render_target
->dxgi_target
, iid
, data
, desc
, bitmap
);
151 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateBitmapBrush(ID2D1DCRenderTarget
*iface
,
152 ID2D1Bitmap
*bitmap
, const D2D1_BITMAP_BRUSH_PROPERTIES
*bitmap_brush_desc
,
153 const D2D1_BRUSH_PROPERTIES
*brush_desc
, ID2D1BitmapBrush
**brush
)
155 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
157 TRACE("iface %p, bitmap %p, bitmap_brush_desc %p, brush_desc %p, brush %p.\n",
158 iface
, bitmap
, bitmap_brush_desc
, brush_desc
, brush
);
160 return ID2D1RenderTarget_CreateBitmapBrush(render_target
->dxgi_target
,
161 bitmap
, bitmap_brush_desc
, brush_desc
, brush
);
164 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateSolidColorBrush(ID2D1DCRenderTarget
*iface
,
165 const D2D1_COLOR_F
*color
, const D2D1_BRUSH_PROPERTIES
*desc
, ID2D1SolidColorBrush
**brush
)
167 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
169 TRACE("iface %p, color %p, desc %p, brush %p.\n", iface
, color
, desc
, brush
);
171 return ID2D1RenderTarget_CreateSolidColorBrush(render_target
->dxgi_target
, color
, desc
, brush
);
174 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateGradientStopCollection(ID2D1DCRenderTarget
*iface
,
175 const D2D1_GRADIENT_STOP
*stops
, UINT32 stop_count
, D2D1_GAMMA gamma
, D2D1_EXTEND_MODE extend_mode
,
176 ID2D1GradientStopCollection
**gradient
)
178 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
180 TRACE("iface %p, stops %p, stop_count %u, gamma %#x, extend_mode %#x, gradient %p.\n",
181 iface
, stops
, stop_count
, gamma
, extend_mode
, gradient
);
183 return ID2D1RenderTarget_CreateGradientStopCollection(render_target
->dxgi_target
,
184 stops
, stop_count
, gamma
, extend_mode
, gradient
);
187 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateLinearGradientBrush(ID2D1DCRenderTarget
*iface
,
188 const D2D1_LINEAR_GRADIENT_BRUSH_PROPERTIES
*gradient_brush_desc
, const D2D1_BRUSH_PROPERTIES
*brush_desc
,
189 ID2D1GradientStopCollection
*gradient
, ID2D1LinearGradientBrush
**brush
)
191 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
193 TRACE("iface %p, gradient_brush_desc %p, brush_desc %p, gradient %p, brush %p.\n",
194 iface
, gradient_brush_desc
, brush_desc
, gradient
, brush
);
196 return ID2D1RenderTarget_CreateLinearGradientBrush(render_target
->dxgi_target
,
197 gradient_brush_desc
, brush_desc
, gradient
, brush
);
200 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateRadialGradientBrush(ID2D1DCRenderTarget
*iface
,
201 const D2D1_RADIAL_GRADIENT_BRUSH_PROPERTIES
*gradient_brush_desc
, const D2D1_BRUSH_PROPERTIES
*brush_desc
,
202 ID2D1GradientStopCollection
*gradient
, ID2D1RadialGradientBrush
**brush
)
204 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
206 TRACE("iface %p, gradient_brush_desc %p, brush_desc %p, gradient %p, brush %p.\n",
207 iface
, gradient_brush_desc
, brush_desc
, gradient
, brush
);
209 return ID2D1RenderTarget_CreateRadialGradientBrush(render_target
->dxgi_target
,
210 gradient_brush_desc
, brush_desc
, gradient
, brush
);
213 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateCompatibleRenderTarget(ID2D1DCRenderTarget
*iface
,
214 const D2D1_SIZE_F
*size
, const D2D1_SIZE_U
*pixel_size
, const D2D1_PIXEL_FORMAT
*format
,
215 D2D1_COMPATIBLE_RENDER_TARGET_OPTIONS options
, ID2D1BitmapRenderTarget
**render_target
)
217 struct d2d_dc_render_target
*rt
= impl_from_ID2D1DCRenderTarget(iface
);
219 TRACE("iface %p, size %p, pixel_size %p, format %p, options %#x, render_target %p,\n",
220 iface
, size
, pixel_size
, format
, options
, render_target
);
222 return ID2D1RenderTarget_CreateCompatibleRenderTarget(rt
->dxgi_target
,
223 size
, pixel_size
, format
, options
, render_target
);
226 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateLayer(ID2D1DCRenderTarget
*iface
,
227 const D2D1_SIZE_F
*size
, ID2D1Layer
**layer
)
229 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
231 TRACE("iface %p, size %p, layer %p.\n", iface
, size
, layer
);
233 return ID2D1RenderTarget_CreateLayer(render_target
->dxgi_target
, size
, layer
);
236 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_CreateMesh(ID2D1DCRenderTarget
*iface
, ID2D1Mesh
**mesh
)
238 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
240 TRACE("iface %p, mesh %p.\n", iface
, mesh
);
242 return ID2D1RenderTarget_CreateMesh(render_target
->dxgi_target
, mesh
);
245 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawLine(ID2D1DCRenderTarget
*iface
,
246 D2D1_POINT_2F p0
, D2D1_POINT_2F p1
, ID2D1Brush
*brush
, float stroke_width
, ID2D1StrokeStyle
*stroke_style
)
248 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
250 TRACE("iface %p, p0 {%.8e, %.8e}, p1 {%.8e, %.8e}, brush %p, stroke_width %.8e, stroke_style %p.\n",
251 iface
, p0
.x
, p0
.y
, p1
.x
, p1
.y
, brush
, stroke_width
, stroke_style
);
253 ID2D1RenderTarget_DrawLine(render_target
->dxgi_target
, p0
, p1
, brush
, stroke_width
, stroke_style
);
256 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawRectangle(ID2D1DCRenderTarget
*iface
,
257 const D2D1_RECT_F
*rect
, ID2D1Brush
*brush
, float stroke_width
, ID2D1StrokeStyle
*stroke_style
)
259 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
261 TRACE("iface %p, rect %s, brush %p, stroke_width %.8e, stroke_style %p.\n",
262 iface
, debug_d2d_rect_f(rect
), brush
, stroke_width
, stroke_style
);
264 ID2D1RenderTarget_DrawRectangle(render_target
->dxgi_target
, rect
, brush
, stroke_width
, stroke_style
);
267 static void STDMETHODCALLTYPE
d2d_dc_render_target_FillRectangle(ID2D1DCRenderTarget
*iface
,
268 const D2D1_RECT_F
*rect
, ID2D1Brush
*brush
)
270 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
272 TRACE("iface %p, rect %s, brush %p.\n", iface
, debug_d2d_rect_f(rect
), brush
);
274 ID2D1RenderTarget_FillRectangle(render_target
->dxgi_target
, rect
, brush
);
277 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawRoundedRectangle(ID2D1DCRenderTarget
*iface
,
278 const D2D1_ROUNDED_RECT
*rect
, ID2D1Brush
*brush
, float stroke_width
, ID2D1StrokeStyle
*stroke_style
)
280 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
282 TRACE("iface %p, rect %p, brush %p, stroke_width %.8e, stroke_style %p.\n",
283 iface
, rect
, brush
, stroke_width
, stroke_style
);
285 ID2D1RenderTarget_DrawRoundedRectangle(render_target
->dxgi_target
, rect
, brush
, stroke_width
, stroke_style
);
288 static void STDMETHODCALLTYPE
d2d_dc_render_target_FillRoundedRectangle(ID2D1DCRenderTarget
*iface
,
289 const D2D1_ROUNDED_RECT
*rect
, ID2D1Brush
*brush
)
291 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
293 TRACE("iface %p, rect %p, brush %p.\n", iface
, rect
, brush
);
295 ID2D1RenderTarget_FillRoundedRectangle(render_target
->dxgi_target
, rect
, brush
);
298 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawEllipse(ID2D1DCRenderTarget
*iface
,
299 const D2D1_ELLIPSE
*ellipse
, ID2D1Brush
*brush
, float stroke_width
, ID2D1StrokeStyle
*stroke_style
)
301 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
303 TRACE("iface %p, ellipse %p, brush %p, stroke_width %.8e, stroke_style %p.\n",
304 iface
, ellipse
, brush
, stroke_width
, stroke_style
);
306 ID2D1RenderTarget_DrawEllipse(render_target
->dxgi_target
, ellipse
, brush
, stroke_width
, stroke_style
);
309 static void STDMETHODCALLTYPE
d2d_dc_render_target_FillEllipse(ID2D1DCRenderTarget
*iface
,
310 const D2D1_ELLIPSE
*ellipse
, ID2D1Brush
*brush
)
312 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
314 TRACE("iface %p, ellipse %p, brush %p.\n", iface
, ellipse
, brush
);
316 ID2D1RenderTarget_FillEllipse(render_target
->dxgi_target
, ellipse
, brush
);
319 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawGeometry(ID2D1DCRenderTarget
*iface
,
320 ID2D1Geometry
*geometry
, ID2D1Brush
*brush
, float stroke_width
, ID2D1StrokeStyle
*stroke_style
)
322 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
324 TRACE("iface %p, geometry %p, brush %p, stroke_width %.8e, stroke_style %p.\n",
325 iface
, geometry
, brush
, stroke_width
, stroke_style
);
327 ID2D1RenderTarget_DrawGeometry(render_target
->dxgi_target
, geometry
, brush
, stroke_width
, stroke_style
);
330 static void STDMETHODCALLTYPE
d2d_dc_render_target_FillGeometry(ID2D1DCRenderTarget
*iface
,
331 ID2D1Geometry
*geometry
, ID2D1Brush
*brush
, ID2D1Brush
*opacity_brush
)
333 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
335 TRACE("iface %p, geometry %p, brush %p, opacity_brush %p.\n", iface
, geometry
, brush
, opacity_brush
);
337 ID2D1RenderTarget_FillGeometry(render_target
->dxgi_target
, geometry
, brush
, opacity_brush
);
340 static void STDMETHODCALLTYPE
d2d_dc_render_target_FillMesh(ID2D1DCRenderTarget
*iface
,
341 ID2D1Mesh
*mesh
, ID2D1Brush
*brush
)
343 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
345 TRACE("iface %p, mesh %p, brush %p.\n", iface
, mesh
, brush
);
347 ID2D1RenderTarget_FillMesh(render_target
->dxgi_target
, mesh
, brush
);
350 static void STDMETHODCALLTYPE
d2d_dc_render_target_FillOpacityMask(ID2D1DCRenderTarget
*iface
,
351 ID2D1Bitmap
*mask
, ID2D1Brush
*brush
, D2D1_OPACITY_MASK_CONTENT content
,
352 const D2D1_RECT_F
*dst_rect
, const D2D1_RECT_F
*src_rect
)
354 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
356 TRACE("iface %p, mask %p, brush %p, content %#x, dst_rect %s, src_rect %s.\n",
357 iface
, mask
, brush
, content
, debug_d2d_rect_f(dst_rect
), debug_d2d_rect_f(src_rect
));
359 ID2D1RenderTarget_FillOpacityMask(render_target
->dxgi_target
,
360 mask
, brush
, content
, dst_rect
, src_rect
);
363 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawBitmap(ID2D1DCRenderTarget
*iface
,
364 ID2D1Bitmap
*bitmap
, const D2D1_RECT_F
*dst_rect
, float opacity
,
365 D2D1_BITMAP_INTERPOLATION_MODE interpolation_mode
, const D2D1_RECT_F
*src_rect
)
367 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
369 TRACE("iface %p, bitmap %p, dst_rect %s, opacity %.8e, interpolation_mode %#x, src_rect %s.\n",
370 iface
, bitmap
, debug_d2d_rect_f(dst_rect
), opacity
, interpolation_mode
, debug_d2d_rect_f(src_rect
));
372 ID2D1RenderTarget_DrawBitmap(render_target
->dxgi_target
,
373 bitmap
, dst_rect
, opacity
, interpolation_mode
, src_rect
);
376 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawText(ID2D1DCRenderTarget
*iface
,
377 const WCHAR
*string
, UINT32 string_len
, IDWriteTextFormat
*text_format
, const D2D1_RECT_F
*layout_rect
,
378 ID2D1Brush
*brush
, D2D1_DRAW_TEXT_OPTIONS options
, DWRITE_MEASURING_MODE measuring_mode
)
380 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
382 TRACE("iface %p, string %s, string_len %u, text_format %p, layout_rect %s, "
383 "brush %p, options %#x, measuring_mode %#x.\n",
384 iface
, debugstr_wn(string
, string_len
), string_len
, text_format
, debug_d2d_rect_f(layout_rect
),
385 brush
, options
, measuring_mode
);
387 ID2D1RenderTarget_DrawText(render_target
->dxgi_target
, string
, string_len
,
388 text_format
, layout_rect
, brush
, options
, measuring_mode
);
391 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawTextLayout(ID2D1DCRenderTarget
*iface
,
392 D2D1_POINT_2F origin
, IDWriteTextLayout
*layout
, ID2D1Brush
*brush
, D2D1_DRAW_TEXT_OPTIONS options
)
394 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
396 TRACE("iface %p, origin {%.8e, %.8e}, layout %p, brush %p, options %#x.\n",
397 iface
, origin
.x
, origin
.y
, layout
, brush
, options
);
399 ID2D1RenderTarget_DrawTextLayout(render_target
->dxgi_target
, origin
, layout
, brush
, options
);
402 static void STDMETHODCALLTYPE
d2d_dc_render_target_DrawGlyphRun(ID2D1DCRenderTarget
*iface
,
403 D2D1_POINT_2F baseline_origin
, const DWRITE_GLYPH_RUN
*glyph_run
, ID2D1Brush
*brush
,
404 DWRITE_MEASURING_MODE measuring_mode
)
406 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
408 TRACE("iface %p, baseline_origin {%.8e, %.8e}, glyph_run %p, brush %p, measuring_mode %#x.\n",
409 iface
, baseline_origin
.x
, baseline_origin
.y
, glyph_run
, brush
, measuring_mode
);
411 ID2D1RenderTarget_DrawGlyphRun(render_target
->dxgi_target
,
412 baseline_origin
, glyph_run
, brush
, measuring_mode
);
415 static void STDMETHODCALLTYPE
d2d_dc_render_target_SetTransform(ID2D1DCRenderTarget
*iface
,
416 const D2D1_MATRIX_3X2_F
*transform
)
418 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
420 TRACE("iface %p, transform %p.\n", iface
, transform
);
422 ID2D1RenderTarget_SetTransform(render_target
->dxgi_target
, transform
);
425 static void STDMETHODCALLTYPE
d2d_dc_render_target_GetTransform(ID2D1DCRenderTarget
*iface
,
426 D2D1_MATRIX_3X2_F
*transform
)
428 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
430 TRACE("iface %p, transform %p.\n", iface
, transform
);
432 ID2D1RenderTarget_GetTransform(render_target
->dxgi_target
, transform
);
435 static void STDMETHODCALLTYPE
d2d_dc_render_target_SetAntialiasMode(ID2D1DCRenderTarget
*iface
,
436 D2D1_ANTIALIAS_MODE antialias_mode
)
438 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
440 TRACE("iface %p, antialias_mode %#x.\n", iface
, antialias_mode
);
442 ID2D1RenderTarget_SetAntialiasMode(render_target
->dxgi_target
, antialias_mode
);
445 static D2D1_ANTIALIAS_MODE STDMETHODCALLTYPE
d2d_dc_render_target_GetAntialiasMode(ID2D1DCRenderTarget
*iface
)
447 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
449 TRACE("iface %p.\n", iface
);
451 return ID2D1RenderTarget_GetAntialiasMode(render_target
->dxgi_target
);
454 static void STDMETHODCALLTYPE
d2d_dc_render_target_SetTextAntialiasMode(ID2D1DCRenderTarget
*iface
,
455 D2D1_TEXT_ANTIALIAS_MODE antialias_mode
)
457 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
459 TRACE("iface %p, antialias_mode %#x.\n", iface
, antialias_mode
);
461 ID2D1RenderTarget_SetTextAntialiasMode(render_target
->dxgi_target
, antialias_mode
);
464 static D2D1_TEXT_ANTIALIAS_MODE STDMETHODCALLTYPE
d2d_dc_render_target_GetTextAntialiasMode(ID2D1DCRenderTarget
*iface
)
466 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
468 TRACE("iface %p.\n", iface
);
470 return ID2D1RenderTarget_GetTextAntialiasMode(render_target
->dxgi_target
);
473 static void STDMETHODCALLTYPE
d2d_dc_render_target_SetTextRenderingParams(ID2D1DCRenderTarget
*iface
,
474 IDWriteRenderingParams
*text_rendering_params
)
476 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
478 TRACE("iface %p, text_rendering_params %p.\n", iface
, text_rendering_params
);
480 ID2D1RenderTarget_SetTextRenderingParams(render_target
->dxgi_target
, text_rendering_params
);
483 static void STDMETHODCALLTYPE
d2d_dc_render_target_GetTextRenderingParams(ID2D1DCRenderTarget
*iface
,
484 IDWriteRenderingParams
**text_rendering_params
)
486 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
488 TRACE("iface %p, text_rendering_params %p.\n", iface
, text_rendering_params
);
490 ID2D1RenderTarget_GetTextRenderingParams(render_target
->dxgi_target
, text_rendering_params
);
493 static void STDMETHODCALLTYPE
d2d_dc_render_target_SetTags(ID2D1DCRenderTarget
*iface
, D2D1_TAG tag1
, D2D1_TAG tag2
)
495 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
497 TRACE("iface %p, tag1 %s, tag2 %s.\n", iface
, wine_dbgstr_longlong(tag1
), wine_dbgstr_longlong(tag2
));
499 ID2D1RenderTarget_SetTags(render_target
->dxgi_target
, tag1
, tag2
);
502 static void STDMETHODCALLTYPE
d2d_dc_render_target_GetTags(ID2D1DCRenderTarget
*iface
, D2D1_TAG
*tag1
, D2D1_TAG
*tag2
)
504 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
506 TRACE("iface %p, tag1 %p, tag2 %p.\n", iface
, tag1
, tag2
);
508 ID2D1RenderTarget_GetTags(render_target
->dxgi_target
, tag1
, tag2
);
511 static void STDMETHODCALLTYPE
d2d_dc_render_target_PushLayer(ID2D1DCRenderTarget
*iface
,
512 const D2D1_LAYER_PARAMETERS
*layer_parameters
, ID2D1Layer
*layer
)
514 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
516 TRACE("iface %p, layer_parameters %p, layer %p.\n", iface
, layer_parameters
, layer
);
518 ID2D1RenderTarget_PushLayer(render_target
->dxgi_target
, layer_parameters
, layer
);
521 static void STDMETHODCALLTYPE
d2d_dc_render_target_PopLayer(ID2D1DCRenderTarget
*iface
)
523 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
525 TRACE("iface %p.\n", iface
);
527 ID2D1RenderTarget_PopLayer(render_target
->dxgi_target
);
530 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_Flush(ID2D1DCRenderTarget
*iface
, D2D1_TAG
*tag1
, D2D1_TAG
*tag2
)
532 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
534 TRACE("iface %p, tag1 %p, tag2 %p.\n", iface
, tag1
, tag2
);
536 return ID2D1RenderTarget_Flush(render_target
->dxgi_target
, tag1
, tag2
);
539 static void STDMETHODCALLTYPE
d2d_dc_render_target_SaveDrawingState(ID2D1DCRenderTarget
*iface
,
540 ID2D1DrawingStateBlock
*state_block
)
542 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
544 TRACE("iface %p, state_block %p.\n", iface
, state_block
);
546 ID2D1RenderTarget_SaveDrawingState(render_target
->dxgi_target
, state_block
);
549 static void STDMETHODCALLTYPE
d2d_dc_render_target_RestoreDrawingState(ID2D1DCRenderTarget
*iface
,
550 ID2D1DrawingStateBlock
*state_block
)
552 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
554 TRACE("iface %p, state_block %p.\n", iface
, state_block
);
556 ID2D1RenderTarget_RestoreDrawingState(render_target
->dxgi_target
, state_block
);
559 static void STDMETHODCALLTYPE
d2d_dc_render_target_PushAxisAlignedClip(ID2D1DCRenderTarget
*iface
,
560 const D2D1_RECT_F
*clip_rect
, D2D1_ANTIALIAS_MODE antialias_mode
)
562 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
564 TRACE("iface %p, clip_rect %s, antialias_mode %#x.\n", iface
, debug_d2d_rect_f(clip_rect
), antialias_mode
);
566 ID2D1RenderTarget_PushAxisAlignedClip(render_target
->dxgi_target
, clip_rect
, antialias_mode
);
569 static void STDMETHODCALLTYPE
d2d_dc_render_target_PopAxisAlignedClip(ID2D1DCRenderTarget
*iface
)
571 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
573 TRACE("iface %p.\n", iface
);
575 ID2D1RenderTarget_PopAxisAlignedClip(render_target
->dxgi_target
);
578 static void STDMETHODCALLTYPE
d2d_dc_render_target_Clear(ID2D1DCRenderTarget
*iface
, const D2D1_COLOR_F
*color
)
580 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
582 TRACE("iface %p, color %p.\n", iface
, color
);
584 ID2D1RenderTarget_Clear(render_target
->dxgi_target
, color
);
587 static void STDMETHODCALLTYPE
d2d_dc_render_target_BeginDraw(ID2D1DCRenderTarget
*iface
)
589 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
591 TRACE("iface %p.\n", iface
);
593 ID2D1RenderTarget_BeginDraw(render_target
->dxgi_target
);
596 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_EndDraw(ID2D1DCRenderTarget
*iface
,
597 D2D1_TAG
*tag1
, D2D1_TAG
*tag2
)
599 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
601 TRACE("iface %p, tag1 %p, tag2 %p.\n", iface
, tag1
, tag2
);
603 return ID2D1RenderTarget_EndDraw(render_target
->dxgi_target
, tag1
, tag2
);
606 static D2D1_PIXEL_FORMAT
* STDMETHODCALLTYPE
d2d_dc_render_target_GetPixelFormat(ID2D1DCRenderTarget
*iface
,
607 D2D1_PIXEL_FORMAT
*format
)
609 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
611 TRACE("iface %p, format %p.\n", iface
, format
);
613 *format
= ID2D1RenderTarget_GetPixelFormat(render_target
->dxgi_target
);
617 static void STDMETHODCALLTYPE
d2d_dc_render_target_SetDpi(ID2D1DCRenderTarget
*iface
, float dpi_x
, float dpi_y
)
619 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
621 TRACE("iface %p, dpi_x %.8e, dpi_y %.8e.\n", iface
, dpi_x
, dpi_y
);
623 ID2D1RenderTarget_SetDpi(render_target
->dxgi_target
, dpi_x
, dpi_y
);
626 static void STDMETHODCALLTYPE
d2d_dc_render_target_GetDpi(ID2D1DCRenderTarget
*iface
, float *dpi_x
, float *dpi_y
)
628 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
630 TRACE("iface %p, dpi_x %p, dpi_y %p.\n", iface
, dpi_x
, dpi_y
);
632 ID2D1RenderTarget_GetDpi(render_target
->dxgi_target
, dpi_x
, dpi_y
);
635 static D2D1_SIZE_F
* STDMETHODCALLTYPE
d2d_dc_render_target_GetSize(ID2D1DCRenderTarget
*iface
, D2D1_SIZE_F
*size
)
637 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
639 TRACE("iface %p, size %p.\n", iface
, size
);
641 if (render_target
->hdc
)
642 *size
= ID2D1RenderTarget_GetSize(render_target
->dxgi_target
);
644 size
->width
= size
->height
= 0.0f
;
649 static D2D1_SIZE_U
* STDMETHODCALLTYPE
d2d_dc_render_target_GetPixelSize(ID2D1DCRenderTarget
*iface
,
650 D2D1_SIZE_U
*pixel_size
)
652 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
654 TRACE("iface %p, pixel_size %p.\n", iface
, pixel_size
);
656 if (render_target
->hdc
)
657 *pixel_size
= ID2D1RenderTarget_GetPixelSize(render_target
->dxgi_target
);
659 pixel_size
->width
= pixel_size
->height
= 0;
664 static UINT32 STDMETHODCALLTYPE
d2d_dc_render_target_GetMaximumBitmapSize(ID2D1DCRenderTarget
*iface
)
666 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
668 TRACE("iface %p.\n", iface
);
670 return ID2D1RenderTarget_GetMaximumBitmapSize(render_target
->dxgi_target
);
673 static BOOL STDMETHODCALLTYPE
d2d_dc_render_target_IsSupported(ID2D1DCRenderTarget
*iface
,
674 const D2D1_RENDER_TARGET_PROPERTIES
*desc
)
676 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
678 TRACE("iface %p, desc %p.\n", iface
, desc
);
680 return ID2D1RenderTarget_IsSupported(render_target
->dxgi_target
, desc
);
683 static HRESULT STDMETHODCALLTYPE
d2d_dc_render_target_BindDC(ID2D1DCRenderTarget
*iface
,
684 const HDC hdc
, const RECT
*rect
)
686 struct d2d_dc_render_target
*render_target
= impl_from_ID2D1DCRenderTarget(iface
);
687 D3D10_TEXTURE2D_DESC texture_desc
;
688 IDXGISurface1
*dxgi_surface
;
689 ID3D10Texture2D
*texture
;
690 ID3D10Device
*device
;
693 TRACE("iface %p, hdc %p, rect %s.\n", iface
, hdc
, wine_dbgstr_rect(rect
));
698 /* Recreate surface using specified size and original texture description. */
699 if (FAILED(hr
= IDXGISurface1_QueryInterface(render_target
->dxgi_surface
, &IID_ID3D10Texture2D
, (void **)&texture
)))
701 WARN("Failed to get texture interface, hr %#x.\n", hr
);
705 ID3D10Texture2D_GetDesc(texture
, &texture_desc
);
706 ID3D10Texture2D_Release(texture
);
707 texture_desc
.Width
= rect
->right
- rect
->left
;
708 texture_desc
.Height
= rect
->bottom
- rect
->top
;
710 if (FAILED(hr
= IDXGISurface1_GetDevice(render_target
->dxgi_surface
, &IID_ID3D10Device
, (void **)&device
)))
712 WARN("Failed to get device from dxgi surface, hr %#x.\n", hr
);
716 hr
= ID3D10Device_CreateTexture2D(device
, &texture_desc
, NULL
, &texture
);
717 ID3D10Device_Release(device
);
720 WARN("Failed to create texture, hr %#x.\n", hr
);
724 hr
= ID3D10Texture2D_QueryInterface(texture
, &IID_IDXGISurface1
, (void **)&dxgi_surface
);
725 ID3D10Texture2D_Release(texture
);
728 WARN("Failed to get surface interface from a texture, hr %#x.\n", hr
);
732 /* Switch dxgi target to new surface. */
733 if (FAILED(hr
= d2d_d3d_render_target_create_rtv(render_target
->dxgi_target
, dxgi_surface
)))
735 WARN("Failed to set new surface, hr %#x.\n", hr
);
736 IDXGISurface1_Release(dxgi_surface
);
740 IDXGISurface1_Release(render_target
->dxgi_surface
);
741 render_target
->dxgi_surface
= dxgi_surface
;
743 render_target
->hdc
= hdc
;
744 render_target
->dst_rect
= *rect
;
749 static const struct ID2D1DCRenderTargetVtbl d2d_dc_render_target_vtbl
=
751 d2d_dc_render_target_QueryInterface
,
752 d2d_dc_render_target_AddRef
,
753 d2d_dc_render_target_Release
,
754 d2d_dc_render_target_GetFactory
,
755 d2d_dc_render_target_CreateBitmap
,
756 d2d_dc_render_target_CreateBitmapFromWicBitmap
,
757 d2d_dc_render_target_CreateSharedBitmap
,
758 d2d_dc_render_target_CreateBitmapBrush
,
759 d2d_dc_render_target_CreateSolidColorBrush
,
760 d2d_dc_render_target_CreateGradientStopCollection
,
761 d2d_dc_render_target_CreateLinearGradientBrush
,
762 d2d_dc_render_target_CreateRadialGradientBrush
,
763 d2d_dc_render_target_CreateCompatibleRenderTarget
,
764 d2d_dc_render_target_CreateLayer
,
765 d2d_dc_render_target_CreateMesh
,
766 d2d_dc_render_target_DrawLine
,
767 d2d_dc_render_target_DrawRectangle
,
768 d2d_dc_render_target_FillRectangle
,
769 d2d_dc_render_target_DrawRoundedRectangle
,
770 d2d_dc_render_target_FillRoundedRectangle
,
771 d2d_dc_render_target_DrawEllipse
,
772 d2d_dc_render_target_FillEllipse
,
773 d2d_dc_render_target_DrawGeometry
,
774 d2d_dc_render_target_FillGeometry
,
775 d2d_dc_render_target_FillMesh
,
776 d2d_dc_render_target_FillOpacityMask
,
777 d2d_dc_render_target_DrawBitmap
,
778 d2d_dc_render_target_DrawText
,
779 d2d_dc_render_target_DrawTextLayout
,
780 d2d_dc_render_target_DrawGlyphRun
,
781 d2d_dc_render_target_SetTransform
,
782 d2d_dc_render_target_GetTransform
,
783 d2d_dc_render_target_SetAntialiasMode
,
784 d2d_dc_render_target_GetAntialiasMode
,
785 d2d_dc_render_target_SetTextAntialiasMode
,
786 d2d_dc_render_target_GetTextAntialiasMode
,
787 d2d_dc_render_target_SetTextRenderingParams
,
788 d2d_dc_render_target_GetTextRenderingParams
,
789 d2d_dc_render_target_SetTags
,
790 d2d_dc_render_target_GetTags
,
791 d2d_dc_render_target_PushLayer
,
792 d2d_dc_render_target_PopLayer
,
793 d2d_dc_render_target_Flush
,
794 d2d_dc_render_target_SaveDrawingState
,
795 d2d_dc_render_target_RestoreDrawingState
,
796 d2d_dc_render_target_PushAxisAlignedClip
,
797 d2d_dc_render_target_PopAxisAlignedClip
,
798 d2d_dc_render_target_Clear
,
799 d2d_dc_render_target_BeginDraw
,
800 d2d_dc_render_target_EndDraw
,
801 d2d_dc_render_target_GetPixelFormat
,
802 d2d_dc_render_target_SetDpi
,
803 d2d_dc_render_target_GetDpi
,
804 d2d_dc_render_target_GetSize
,
805 d2d_dc_render_target_GetPixelSize
,
806 d2d_dc_render_target_GetMaximumBitmapSize
,
807 d2d_dc_render_target_IsSupported
,
808 d2d_dc_render_target_BindDC
,
811 static const struct d2d_device_context_ops d2d_dc_render_target_ops
=
813 d2d_dc_render_target_present
,
816 HRESULT
d2d_dc_render_target_init(struct d2d_dc_render_target
*render_target
, ID2D1Factory
*factory
,
817 ID3D10Device1
*device
, const D2D1_RENDER_TARGET_PROPERTIES
*desc
)
819 D3D10_TEXTURE2D_DESC texture_desc
;
820 ID3D10Texture2D
*texture
;
823 render_target
->ID2D1DCRenderTarget_iface
.lpVtbl
= &d2d_dc_render_target_vtbl
;
825 /* Set with BindDC(). */
826 SetRectEmpty(&render_target
->dst_rect
);
827 render_target
->hdc
= NULL
;
829 /* Dummy 1x1 texture, recreated on BindDC(). */
830 texture_desc
.Width
= 1;
831 texture_desc
.Height
= 1;
832 texture_desc
.MipLevels
= 1;
833 texture_desc
.ArraySize
= 1;
835 texture_desc
.Format
= desc
->pixelFormat
.format
;
836 switch (texture_desc
.Format
)
838 case DXGI_FORMAT_B8G8R8A8_UNORM
:
839 if (desc
->pixelFormat
.alphaMode
== D2D1_ALPHA_MODE_PREMULTIPLIED
840 || desc
->pixelFormat
.alphaMode
== D2D1_ALPHA_MODE_IGNORE
)
844 FIXME("Unhandled format %#x, alpha mode %u.\n", texture_desc
.Format
, desc
->pixelFormat
.alphaMode
);
845 return D2DERR_UNSUPPORTED_PIXEL_FORMAT
;
848 texture_desc
.SampleDesc
.Count
= 1;
849 texture_desc
.SampleDesc
.Quality
= 0;
850 texture_desc
.Usage
= D3D10_USAGE_DEFAULT
;
851 texture_desc
.BindFlags
= D3D10_BIND_RENDER_TARGET
| D3D10_BIND_SHADER_RESOURCE
;
852 texture_desc
.CPUAccessFlags
= 0;
853 texture_desc
.MiscFlags
= D3D10_RESOURCE_MISC_GDI_COMPATIBLE
;
855 if (FAILED(hr
= ID3D10Device1_CreateTexture2D(device
, &texture_desc
, NULL
, &texture
)))
857 WARN("Failed to create texture, hr %#x.\n", hr
);
861 hr
= ID3D10Texture2D_QueryInterface(texture
, &IID_IDXGISurface1
, (void **)&render_target
->dxgi_surface
);
862 ID3D10Texture2D_Release(texture
);
865 WARN("Failed to get DXGI surface interface, hr %#x.\n", hr
);
869 if (FAILED(hr
= d2d_d3d_create_render_target(factory
, (IDXGISurface
*)render_target
->dxgi_surface
,
870 (IUnknown
*)&render_target
->ID2D1DCRenderTarget_iface
, &d2d_dc_render_target_ops
,
871 desc
, (void **)&render_target
->dxgi_inner
)))
873 WARN("Failed to create DXGI surface render target, hr %#x.\n", hr
);
874 IDXGISurface1_Release(render_target
->dxgi_surface
);
878 if (FAILED(hr
= IUnknown_QueryInterface(render_target
->dxgi_inner
,
879 &IID_ID2D1RenderTarget
, (void **)&render_target
->dxgi_target
)))
881 WARN("Failed to retrieve ID2D1RenderTarget interface, hr %#x.\n", hr
);
882 IUnknown_Release(render_target
->dxgi_inner
);
883 IDXGISurface1_Release(render_target
->dxgi_surface
);