2 * Copyright (c) 1998 Lionel ULMER
4 * This file contains the implementation of Direct3DViewport2.
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
25 #define NONAMELESSUNION
26 #define NONAMELESSSTRUCT
34 #include "wine/debug.h"
36 #include "d3d_private.h"
38 WINE_DEFAULT_DEBUG_CHANNEL(ddraw);
40 static void activate(IDirect3DViewportImpl* This) {
41 IDirect3DLightImpl* light;
44 /* Activate all the lights associated with this context */
47 while (light != NULL) {
48 light->activate(light);
52 /* And copy the values in the structure used by the device */
54 vp.dwX = This->viewports.vp2.dwX;
55 vp.dwY = This->viewports.vp2.dwY;
56 vp.dwHeight = This->viewports.vp2.dwHeight;
57 vp.dwWidth = This->viewports.vp2.dwWidth;
58 vp.dvMinZ = This->viewports.vp2.dvMinZ;
59 vp.dvMaxZ = This->viewports.vp2.dvMaxZ;
61 vp.dwX = This->viewports.vp1.dwX;
62 vp.dwY = This->viewports.vp1.dwY;
63 vp.dwHeight = This->viewports.vp1.dwHeight;
64 vp.dwWidth = This->viewports.vp1.dwWidth;
65 vp.dvMinZ = This->viewports.vp1.dvMinZ;
66 vp.dvMaxZ = This->viewports.vp1.dvMaxZ;
69 /* And also set the viewport */
70 IDirect3DDevice7_SetViewport(ICOM_INTERFACE(This->active_device, IDirect3DDevice7), &vp);
74 static void _dump_D3DVIEWPORT(D3DVIEWPORT *lpvp)
76 TRACE(" - dwSize = %ld dwX = %ld dwY = %ld\n",
77 lpvp->dwSize, lpvp->dwX, lpvp->dwY);
78 TRACE(" - dwWidth = %ld dwHeight = %ld\n",
79 lpvp->dwWidth, lpvp->dwHeight);
80 TRACE(" - dvScaleX = %f dvScaleY = %f\n",
81 lpvp->dvScaleX, lpvp->dvScaleY);
82 TRACE(" - dvMaxX = %f dvMaxY = %f\n",
83 lpvp->dvMaxX, lpvp->dvMaxY);
84 TRACE(" - dvMinZ = %f dvMaxZ = %f\n",
85 lpvp->dvMinZ, lpvp->dvMaxZ);
88 static void _dump_D3DVIEWPORT2(D3DVIEWPORT2 *lpvp)
90 TRACE(" - dwSize = %ld dwX = %ld dwY = %ld\n",
91 lpvp->dwSize, lpvp->dwX, lpvp->dwY);
92 TRACE(" - dwWidth = %ld dwHeight = %ld\n",
93 lpvp->dwWidth, lpvp->dwHeight);
94 TRACE(" - dvClipX = %f dvClipY = %f\n",
95 lpvp->dvClipX, lpvp->dvClipY);
96 TRACE(" - dvClipWidth = %f dvClipHeight = %f\n",
97 lpvp->dvClipWidth, lpvp->dvClipHeight);
98 TRACE(" - dvMinZ = %f dvMaxZ = %f\n",
99 lpvp->dvMinZ, lpvp->dvMaxZ);
103 Main_IDirect3DViewportImpl_3_2_1_QueryInterface(LPDIRECT3DVIEWPORT3 iface,
107 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
108 TRACE("(%p/%p)->(%s,%p)\n", This, iface, debugstr_guid(riid), obp);
112 if ( IsEqualGUID(&IID_IUnknown, riid) ||
113 IsEqualGUID(&IID_IDirect3DViewport, riid) ||
114 IsEqualGUID(&IID_IDirect3DViewport2, riid) ||
115 IsEqualGUID(&IID_IDirect3DViewport3, riid) ) {
116 IDirect3DViewport3_AddRef(ICOM_INTERFACE(This, IDirect3DViewport3));
117 *obp = ICOM_INTERFACE(This, IDirect3DViewport3);
118 TRACE(" Creating IDirect3DViewport1/2/3 interface %p\n", *obp);
121 FIXME("(%p): interface for IID %s NOT found!\n", This, debugstr_guid(riid));
122 return OLE_E_ENUM_NOMORE;
126 Main_IDirect3DViewportImpl_3_2_1_AddRef(LPDIRECT3DVIEWPORT3 iface)
128 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
129 TRACE("(%p/%p)->() incrementing from %lu.\n", This, iface, This->ref);
130 return ++(This->ref);
134 Main_IDirect3DViewportImpl_3_2_1_Release(LPDIRECT3DVIEWPORT3 iface)
136 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
137 TRACE("(%p/%p)->() decrementing from %lu.\n", This, iface, This->ref);
138 if (!--(This->ref)) {
139 HeapFree(GetProcessHeap(), 0, This);
147 Main_IDirect3DViewportImpl_3_2_1_Initialize(LPDIRECT3DVIEWPORT3 iface,
148 LPDIRECT3D lpDirect3D)
150 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
151 TRACE("(%p/%p)->(%p) no-op...\n", This, iface, lpDirect3D);
156 Main_IDirect3DViewportImpl_3_2_1_GetViewport(LPDIRECT3DVIEWPORT3 iface,
157 LPD3DVIEWPORT lpData)
159 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
161 TRACE("(%p/%p)->(%p)\n", This, iface, lpData);
162 if (This->use_vp2 != 0) {
163 ERR(" Requesting to get a D3DVIEWPORT struct where a D3DVIEWPORT2 was set !\n");
164 return DDERR_INVALIDPARAMS;
166 dwSize = lpData->dwSize;
167 memset(lpData, 0, dwSize);
168 memcpy(lpData, &(This->viewports.vp1), dwSize);
170 if (TRACE_ON(ddraw)) {
171 TRACE(" returning D3DVIEWPORT :\n");
172 _dump_D3DVIEWPORT(lpData);
179 Main_IDirect3DViewportImpl_3_2_1_SetViewport(LPDIRECT3DVIEWPORT3 iface,
180 LPD3DVIEWPORT lpData)
182 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
183 LPDIRECT3DVIEWPORT3 current_viewport;
184 TRACE("(%p/%p)->(%p)\n", This, iface, lpData);
186 if (TRACE_ON(ddraw)) {
187 TRACE(" getting D3DVIEWPORT :\n");
188 _dump_D3DVIEWPORT(lpData);
192 memset(&(This->viewports.vp1), 0, sizeof(This->viewports.vp1));
193 memcpy(&(This->viewports.vp1), lpData, lpData->dwSize);
195 /* Tests on two games shows that these values are never used properly so overide
196 them with proper ones :-)
198 This->viewports.vp1.dvMinZ = 0.0;
199 This->viewports.vp1.dvMaxZ = 1.0;
201 if (This->active_device) {
202 IDirect3DDevice3_GetCurrentViewport(ICOM_INTERFACE(This->active_device, IDirect3DDevice3), ¤t_viewport);
203 if (ICOM_OBJECT(IDirect3DViewportImpl, IDirect3DViewport3, current_viewport) == This)
204 This->activate(This);
205 IDirect3DViewport3_Release(current_viewport);
212 Main_IDirect3DViewportImpl_3_2_1_TransformVertices(LPDIRECT3DVIEWPORT3 iface,
214 LPD3DTRANSFORMDATA lpData,
218 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
219 FIXME("(%p/%p)->(%08lx,%p,%08lx,%p): stub!\n", This, iface, dwVertexCount, lpData, dwFlags, lpOffScreen);
226 Main_IDirect3DViewportImpl_3_2_1_LightElements(LPDIRECT3DVIEWPORT3 iface,
227 DWORD dwElementCount,
228 LPD3DLIGHTDATA lpData)
230 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
231 FIXME("(%p/%p)->(%08lx,%p): stub!\n", This, iface, dwElementCount, lpData);
236 Main_IDirect3DViewportImpl_3_2_1_SetBackground(LPDIRECT3DVIEWPORT3 iface,
237 D3DMATERIALHANDLE hMat)
239 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
240 TRACE("(%p/%p)->(%08lx)\n", This, iface, (DWORD) hMat);
242 This->background = (IDirect3DMaterialImpl *) hMat;
243 TRACE(" setting background color : %f %f %f %f\n",
244 This->background->mat.u.diffuse.u1.r,
245 This->background->mat.u.diffuse.u2.g,
246 This->background->mat.u.diffuse.u3.b,
247 This->background->mat.u.diffuse.u4.a);
253 Main_IDirect3DViewportImpl_3_2_1_GetBackground(LPDIRECT3DVIEWPORT3 iface,
254 LPD3DMATERIALHANDLE lphMat,
257 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
258 FIXME("(%p/%p)->(%p,%p): stub!\n", This, iface, lphMat, lpValid);
263 Main_IDirect3DViewportImpl_3_2_1_SetBackgroundDepth(LPDIRECT3DVIEWPORT3 iface,
264 LPDIRECTDRAWSURFACE lpDDSurface)
266 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
267 FIXME("(%p/%p)->(%p): stub!\n", This, iface, lpDDSurface);
272 Main_IDirect3DViewportImpl_3_2_1_GetBackgroundDepth(LPDIRECT3DVIEWPORT3 iface,
273 LPDIRECTDRAWSURFACE* lplpDDSurface,
276 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
277 FIXME("(%p/%p)->(%p,%p): stub!\n", This, iface, lplpDDSurface, lpValid);
282 Main_IDirect3DViewportImpl_3_2_1_Clear(LPDIRECT3DVIEWPORT3 iface,
287 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
288 DWORD color = 0x00000000;
290 TRACE("(%p/%p)->(%08lx,%p,%08lx)\n", This, iface, dwCount, lpRects, dwFlags);
291 if (This->active_device == NULL) {
292 ERR(" Trying to clear a viewport not attached to a device !\n");
293 return D3DERR_VIEWPORTHASNODEVICE;
295 if (dwFlags & D3DCLEAR_TARGET) {
296 if (This->background == NULL) {
297 ERR(" Trying to clear the color buffer without background material !\n");
300 ((int) ((This->background->mat.u.diffuse.u1.r) * 255) << 16) |
301 ((int) ((This->background->mat.u.diffuse.u2.g) * 255) << 8) |
302 ((int) ((This->background->mat.u.diffuse.u3.b) * 255) << 0) |
303 ((int) ((This->background->mat.u.diffuse.u4.a) * 255) << 24);
306 return This->active_device->clear(This->active_device, dwCount, lpRects,
307 dwFlags & (D3DCLEAR_ZBUFFER | D3DCLEAR_TARGET),
308 color, 1.0, 0x00000000);
312 Main_IDirect3DViewportImpl_3_2_1_AddLight(LPDIRECT3DVIEWPORT3 iface,
313 LPDIRECT3DLIGHT lpDirect3DLight)
315 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
316 IDirect3DLightImpl *lpDirect3DLightImpl = ICOM_OBJECT(IDirect3DLightImpl, IDirect3DLight, lpDirect3DLight);
318 DWORD map = This->map_lights;
320 TRACE("(%p/%p)->(%p)\n", This, iface, lpDirect3DLight);
322 if (This->num_lights >= 8)
323 return DDERR_INVALIDPARAMS;
325 /* Find a light number and update both light and viewports objects accordingly */
330 lpDirect3DLightImpl->dwLightIndex = i;
332 This->map_lights |= 1<<i;
334 /* Add the light in the 'linked' chain */
335 lpDirect3DLightImpl->next = This->lights;
336 This->lights = lpDirect3DLightImpl;
338 /* Attach the light to the viewport */
339 lpDirect3DLightImpl->active_viewport = This;
341 /* If active, activate the light */
342 if (This->active_device != NULL) {
343 lpDirect3DLightImpl->activate(lpDirect3DLightImpl);
350 Main_IDirect3DViewportImpl_3_2_1_DeleteLight(LPDIRECT3DVIEWPORT3 iface,
351 LPDIRECT3DLIGHT lpDirect3DLight)
353 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
354 IDirect3DLightImpl *lpDirect3DLightImpl = ICOM_OBJECT(IDirect3DLightImpl, IDirect3DLight, lpDirect3DLight);
355 IDirect3DLightImpl *cur_light, *prev_light = NULL;
357 TRACE("(%p/%p)->(%p)\n", This, iface, lpDirect3DLight);
358 cur_light = This->lights;
359 while (cur_light != NULL) {
360 if (cur_light == lpDirect3DLightImpl) {
361 lpDirect3DLightImpl->desactivate(lpDirect3DLightImpl);
362 if (prev_light == NULL) This->lights = cur_light->next;
363 else prev_light->next = cur_light->next;
364 /* Detach the light to the viewport */
365 cur_light->active_viewport = NULL;
367 This->map_lights &= ~(1<<lpDirect3DLightImpl->dwLightIndex);
370 prev_light = cur_light;
371 cur_light = cur_light->next;
373 return DDERR_INVALIDPARAMS;
377 Main_IDirect3DViewportImpl_3_2_1_NextLight(LPDIRECT3DVIEWPORT3 iface,
378 LPDIRECT3DLIGHT lpDirect3DLight,
379 LPDIRECT3DLIGHT* lplpDirect3DLight,
382 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
383 FIXME("(%p/%p)->(%p,%p,%08lx): stub!\n", This, iface, lpDirect3DLight, lplpDirect3DLight, dwFlags);
388 Main_IDirect3DViewportImpl_3_2_GetViewport2(LPDIRECT3DVIEWPORT3 iface,
389 LPD3DVIEWPORT2 lpData)
391 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
393 TRACE("(%p/%p)->(%p)\n", This, iface, lpData);
394 if (This->use_vp2 != 1) {
395 ERR(" Requesting to get a D3DVIEWPORT2 struct where a D3DVIEWPORT was set !\n");
396 return DDERR_INVALIDPARAMS;
398 dwSize = lpData->dwSize;
399 memset(lpData, 0, dwSize);
400 memcpy(lpData, &(This->viewports.vp2), dwSize);
402 if (TRACE_ON(ddraw)) {
403 TRACE(" returning D3DVIEWPORT2 :\n");
404 _dump_D3DVIEWPORT2(lpData);
411 Main_IDirect3DViewportImpl_3_2_SetViewport2(LPDIRECT3DVIEWPORT3 iface,
412 LPD3DVIEWPORT2 lpData)
414 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
415 LPDIRECT3DVIEWPORT3 current_viewport;
416 TRACE("(%p/%p)->(%p)\n", This, iface, lpData);
418 if (TRACE_ON(ddraw)) {
419 TRACE(" getting D3DVIEWPORT2 :\n");
420 _dump_D3DVIEWPORT2(lpData);
424 memset(&(This->viewports.vp2), 0, sizeof(This->viewports.vp2));
425 memcpy(&(This->viewports.vp2), lpData, lpData->dwSize);
427 if (This->active_device) {
428 IDirect3DDevice3_GetCurrentViewport(ICOM_INTERFACE(This->active_device, IDirect3DDevice3), ¤t_viewport);
429 if (ICOM_OBJECT(IDirect3DViewportImpl, IDirect3DViewport3, current_viewport) == This)
430 This->activate(This);
431 IDirect3DViewport3_Release(current_viewport);
438 Main_IDirect3DViewportImpl_3_SetBackgroundDepth2(LPDIRECT3DVIEWPORT3 iface,
439 LPDIRECTDRAWSURFACE4 lpDDS)
441 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
442 FIXME("(%p/%p)->(%p): stub!\n", This, iface, lpDDS);
447 Main_IDirect3DViewportImpl_3_GetBackgroundDepth2(LPDIRECT3DVIEWPORT3 iface,
448 LPDIRECTDRAWSURFACE4* lplpDDS,
451 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
452 FIXME("(%p/%p)->(%p,%p): stub!\n", This, iface, lplpDDS, lpValid);
457 Main_IDirect3DViewportImpl_3_Clear2(LPDIRECT3DVIEWPORT3 iface,
465 ICOM_THIS_FROM(IDirect3DViewportImpl, IDirect3DViewport3, iface);
466 TRACE("(%p/%p)->(%08lx,%p,%08lx,%08lx,%f,%08lx)\n", This, iface, dwCount, lpRects, dwFlags, dwColor, dvZ, dwStencil);
467 if (This->active_device == NULL) {
468 ERR(" Trying to clear a viewport not attached to a device !\n");
469 return D3DERR_VIEWPORTHASNODEVICE;
471 return This->active_device->clear(This->active_device, dwCount, lpRects, dwFlags, dwColor, dvZ, dwStencil);
474 #if !defined(__STRICT_ANSI__) && defined(__GNUC__)
475 # define XCAST(fun) (typeof(VTABLE_IDirect3DViewport3.fun))
477 # define XCAST(fun) (void*)
480 IDirect3DViewport3Vtbl VTABLE_IDirect3DViewport3 =
482 XCAST(QueryInterface) Main_IDirect3DViewportImpl_3_2_1_QueryInterface,
483 XCAST(AddRef) Main_IDirect3DViewportImpl_3_2_1_AddRef,
484 XCAST(Release) Main_IDirect3DViewportImpl_3_2_1_Release,
485 XCAST(Initialize) Main_IDirect3DViewportImpl_3_2_1_Initialize,
486 XCAST(GetViewport) Main_IDirect3DViewportImpl_3_2_1_GetViewport,
487 XCAST(SetViewport) Main_IDirect3DViewportImpl_3_2_1_SetViewport,
488 XCAST(TransformVertices) Main_IDirect3DViewportImpl_3_2_1_TransformVertices,
489 XCAST(LightElements) Main_IDirect3DViewportImpl_3_2_1_LightElements,
490 XCAST(SetBackground) Main_IDirect3DViewportImpl_3_2_1_SetBackground,
491 XCAST(GetBackground) Main_IDirect3DViewportImpl_3_2_1_GetBackground,
492 XCAST(SetBackgroundDepth) Main_IDirect3DViewportImpl_3_2_1_SetBackgroundDepth,
493 XCAST(GetBackgroundDepth) Main_IDirect3DViewportImpl_3_2_1_GetBackgroundDepth,
494 XCAST(Clear) Main_IDirect3DViewportImpl_3_2_1_Clear,
495 XCAST(AddLight) Main_IDirect3DViewportImpl_3_2_1_AddLight,
496 XCAST(DeleteLight) Main_IDirect3DViewportImpl_3_2_1_DeleteLight,
497 XCAST(NextLight) Main_IDirect3DViewportImpl_3_2_1_NextLight,
498 XCAST(GetViewport2) Main_IDirect3DViewportImpl_3_2_GetViewport2,
499 XCAST(SetViewport2) Main_IDirect3DViewportImpl_3_2_SetViewport2,
500 XCAST(SetBackgroundDepth2) Main_IDirect3DViewportImpl_3_SetBackgroundDepth2,
501 XCAST(GetBackgroundDepth2) Main_IDirect3DViewportImpl_3_GetBackgroundDepth2,
502 XCAST(Clear2) Main_IDirect3DViewportImpl_3_Clear2,
505 #if !defined(__STRICT_ANSI__) && defined(__GNUC__)
512 HRESULT d3dviewport_create(IDirect3DViewportImpl **obj, IDirectDrawImpl *d3d)
514 IDirect3DViewportImpl *object;
516 object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IDirect3DViewportImpl));
517 if (object == NULL) return DDERR_OUTOFMEMORY;
521 object->activate = activate;
522 object->use_vp2 = 0xFF;
524 object->lights = NULL;
525 object->num_lights = 0;
526 object->map_lights = 0;
528 ICOM_INIT_INTERFACE(object, IDirect3DViewport3, VTABLE_IDirect3DViewport3);
532 TRACE(" creating implementation at %p.\n", *obj);