Вершина нормальная для треугольников додекаэдра

Я хочу инициализировать нормаль вершины для использования освещения, но я не знаю, как этого добиться.

Ниже приведена программа, которую я сделал с использованием DirectX 9 и C++. Он имитирует простую орбиту планеты без освещения и без нормальной вершины.

что мне нужно, это нормаль вершины в массиве вершин и инициализировать массив индексов

  // include the basic windows header files and the Direct3D header file 
  #include <windows.h>
  #include <windowsx.h>
  #include <d3d9.h>
  #include <d3dx9.h> 
  #include <math.h>


  // define the screen resolution
  #define SCREEN_WIDTH 800
  #define SCREEN_HEIGHT 600

  // include the Direct3D Library files
 #pragma comment (lib, "d3d9.lib")
  #pragma comment (lib, "d3dx9.lib")

  // global declarations
  LPDIRECT3D9 d3d;
  LPDIRECT3DDEVICE9 d3ddev;
 LPDIRECT3DVERTEXBUFFER9 v_buffer = NULL;    // the pointer to the vertex buffer
   LPDIRECT3DINDEXBUFFER9 i_buffer = NULL;    // the pointer to the index buffer

  // function prototypes
  void initD3D(HWND hWnd);
  void render_frame(void);
  void cleanD3D(void);
  void init_graphics(void);

   struct CUSTOMVERTEX {FLOAT X, Y, Z; DWORD COLOR;};
   #define CUSTOMFVF (D3DFVF_XYZ | D3DFVF_DIFFUSE )

   // the WindowProc function prototype
   LRESULT CALLBACK WindowProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam);


  // the entry point for any Windows program
   int WINAPI WinMain(HINSTANCE hInstance,
               HINSTANCE hPrevInstance,
               LPSTR lpCmdLine,
               int nCmdShow)
  {
  HWND hWnd;
  WNDCLASSEX wc;

   ZeroMemory(&wc, sizeof(WNDCLASSEX));

  wc.cbSize = sizeof(WNDCLASSEX);
  wc.style = CS_HREDRAW | CS_VREDRAW;
  wc.lpfnWndProc = WindowProc;
  wc.hInstance = hInstance;
  wc.hCursor = LoadCursor(NULL, IDC_ARROW);
  wc.lpszClassName = L"WindowClass";

  RegisterClassEx(&wc);

  hWnd = CreateWindowEx(NULL, L"WindowClass", L"Our Direct3D Program",
                      WS_OVERLAPPEDWINDOW, 0, 0, SCREEN_WIDTH, SCREEN_HEIGHT,
                      NULL, NULL, hInstance, NULL);

   ShowWindow(hWnd, nCmdShow);

  initD3D(hWnd);

  MSG msg;

   while(TRUE)
   {
    while(PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }

    if(msg.message == WM_QUIT)
        break;

    render_frame();
  }

   cleanD3D();

  return msg.wParam;
 }


   // this is the main message handler for the program
  LRESULT CALLBACK WindowProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
 {
  switch(message)
  {
    case WM_DESTROY:
        {
            PostQuitMessage(0);
            return 0;
        } break;
  }

   return DefWindowProc (hWnd, message, wParam, lParam);
   }


    // this function initializes and prepares Direct3D for use
    void initD3D(HWND hWnd)
    {
   d3d = Direct3DCreate9(D3D_SDK_VERSION);

     D3DPRESENT_PARAMETERS d3dpp;

     ZeroMemory(&d3dpp, sizeof(d3dpp));
     d3dpp.Windowed = TRUE;
    d3dpp.SwapEffect = D3DSWAPEFFECT_DISCARD;
    d3dpp.hDeviceWindow = hWnd;
    d3dpp.BackBufferFormat = D3DFMT_X8R8G8B8;
    d3dpp.BackBufferWidth = SCREEN_WIDTH;
    d3dpp.BackBufferHeight = SCREEN_HEIGHT;
    d3dpp.EnableAutoDepthStencil = TRUE;
    d3dpp.AutoDepthStencilFormat = D3DFMT_D16;

   d3d->CreateDevice(D3DADAPTER_DEFAULT,
                  D3DDEVTYPE_HAL,
                  hWnd,
                  D3DCREATE_SOFTWARE_VERTEXPROCESSING,
                  &d3dpp,
                  &d3ddev);

    init_graphics();

    d3ddev->SetRenderState(D3DRS_LIGHTING, FALSE);    // turn off the 3D lighting
     d3ddev->SetRenderState(D3DRS_CULLMODE, D3DCULL_NONE);    // turn off culling
    d3ddev->SetRenderState(D3DRS_ZENABLE, TRUE);    // turn on the z-buffer
   }


    // this is the function used to render a single frame
   void render_frame(void)
   {
   d3ddev->Clear(0, NULL, D3DCLEAR_TARGET, D3DCOLOR_XRGB(0, 0, 0), 1.0f, 0);
  d3ddev->Clear(0, NULL, D3DCLEAR_ZBUFFER, D3DCOLOR_XRGB(0, 0, 0), 1.0f, 0);

   d3ddev->BeginScene();

    d3ddev->SetFVF(CUSTOMFVF);

   // set the view transform
   D3DXMATRIX matView;    // the view transform matrix
   D3DXMatrixLookAtLH(&matView,
   &D3DXVECTOR3 (1.0f, 9.0f, 50.0f),    // the camera position
   &D3DXVECTOR3 (1.0f, 1.0f, 1.0f),      // the look-at position
    &D3DXVECTOR3 (1.0f, 2.0f, 1.0f));    // the up direction
    d3ddev->SetTransform(D3DTS_VIEW, &matView);    // set the view transform to matView 

   // set the projection transform
   D3DXMATRIX matProjection;    // the projection transform matrix
   D3DXMatrixPerspectiveFovLH(&matProjection,
                           D3DXToRadian(60),    // the horizontal field of view
                           (FLOAT)SCREEN_WIDTH / (FLOAT)SCREEN_HEIGHT, // aspect ratio
                           1.0f,   // the near view-plane
                           100.0f);    // the far view-plane
   d3ddev->SetTransform(D3DTS_PROJECTION, &matProjection); // set the projection

   // set the world transform
   static float index = 0.0f; index+=0.03f; // an ever-increasing float value
   D3DXMATRIX matRotateY;    // a matrix to store the rotation for each triangle
D3DXMATRIX matTranslateX_venus;
D3DXMATRIX matTranslateX_merkurius;
D3DXMATRIX matTranslateX_sun;
D3DXMATRIX matScale_merkurius;
D3DXMATRIX matScale_sun;
D3DXMATRIX matTranslateX_earth;
D3DXMATRIX matTranslateX_mars;

  D3DXMatrixRotationY(&matRotateY, index);    // the rotation matrix
   D3DXMatrixTranslation(&matTranslateX_venus,16.0f,0.0f,0.0f);
  D3DXMatrixTranslation(&matTranslateX_earth,24.0f,0.0f,0.0f);
  D3DXMatrixTranslation(&matTranslateX_mars,32.0f,0.0f,0.0f);
   D3DXMatrixTranslation(&matTranslateX_merkurius,8.0f,0.0f,0.0f);
   D3DXMatrixTranslation(&matTranslateX_sun,0.0f,0.0f,0.0f);
   D3DXMatrixScaling(&matScale_merkurius, 0.5f, 0.5f, 0.5f);
    D3DXMatrixScaling(&matScale_sun, 2.5f, 2.5f, 2.5f);
   // set the world transform


  // select the vertex buffer to display
  d3ddev->SetStreamSource(0, v_buffer, 0, sizeof(CUSTOMVERTEX));
  d3ddev->SetIndices(i_buffer);

   // draw the pyramid

  d3ddev->SetTransform(D3DTS_WORLD, &(matScale_sun*matTranslateX_sun *matRotateY)); 
    d3ddev->DrawIndexedPrimitive(D3DPT_TRIANGLELIST, 0, 0, 12, 0, 20);


  d3ddev->SetTransform(D3DTS_WORLD, &(matTranslateX_venus * matRotateY)); 
  d3ddev->DrawIndexedPrimitive(D3DPT_TRIANGLELIST, 12, 0, 12, 0, 20);

  d3ddev->SetTransform(D3DTS_WORLD, &(matScale_merkurius*matTranslateX_merkurius
    *         matRotateY));
  d3ddev->DrawIndexedPrimitive(D3DPT_TRIANGLELIST, 24, 0, 12, 0, 20);

   d3ddev->SetTransform(D3DTS_WORLD, &(matTranslateX_earth * matRotateY));
  d3ddev->DrawIndexedPrimitive(D3DPT_TRIANGLELIST, 36, 0, 12, 0, 20);

   d3ddev->SetTransform(D3DTS_WORLD, &(matTranslateX_mars * matRotateY));
  d3ddev->DrawIndexedPrimitive(D3DPT_TRIANGLELIST, 48, 0, 12, 0, 20);
   d3ddev->EndScene(); 

   d3ddev->Present(NULL, NULL, NULL, NULL);
  }


  // this is the function that cleans up Direct3D and COM
  void cleanD3D(void)
  {
    v_buffer->Release();
    i_buffer->Release();
    d3ddev->Release();
  d3d->Release();
  }


   // this is the function that puts the 3D models into video RAM
   void init_graphics(void)
  {
// create the vertices using the CUSTOMVERTEX
float t = (1.0 + sqrt(5.0)) / 2.0;
struct CUSTOMVERTEX vertices[] =
{
    {-1, t, 0, D3DCOLOR_XRGB(255,255,0), }, 
    {1, t, 0, D3DCOLOR_XRGB(255,255,0), },
    {-1, -t, 0, D3DCOLOR_XRGB(255,255,0), },
    {1, -t, 0, D3DCOLOR_XRGB(255,255,0), },
    {0, -1, t, D3DCOLOR_XRGB(255,255,0), },
    {0, 1, t, D3DCOLOR_XRGB(255,255,0), },
    {0, -1, -t, D3DCOLOR_XRGB(255,255,0), },
    {0, 1, -t, D3DCOLOR_XRGB(255,255,0), },
    {t, 0, -1, D3DCOLOR_XRGB(255,255,0), },
    {t, 0, 1, D3DCOLOR_XRGB(255,255,0), },
    {-t, 0, -1, D3DCOLOR_XRGB(255,255,0), },
    {-t, 0, 1, D3DCOLOR_XRGB(255,255,0), },

    {-1, t, 0, D3DCOLOR_XRGB(88,90,97), }, 
    {1, t, 0, D3DCOLOR_XRGB(88,90,97), },
    {-1, -t, 0, D3DCOLOR_XRGB(88,90,97), },
    {1, -t, 0, D3DCOLOR_XRGB(88,90,97), },
    {0, -1, t, D3DCOLOR_XRGB(88,90,97), },
    {0, 1, t, D3DCOLOR_XRGB(88,90,97), },
    {0, -1, -t, D3DCOLOR_XRGB(88,90,97), },
    {0, 1, -t, D3DCOLOR_XRGB(88,90,97), },
    {t, 0, -1, D3DCOLOR_XRGB(88,90,97), },
    {t, 0, 1, D3DCOLOR_XRGB(88,90,97), },
    {-t, 0, -1, D3DCOLOR_XRGB(88,90,97), },
    {-t, 0, 1, D3DCOLOR_XRGB(88,90,97), },

    {-1, t, 0, D3DCOLOR_XRGB(136,108,57), }, 
    {1, t, 0, D3DCOLOR_XRGB(136,108,57), },
    {-1, -t, 0, D3DCOLOR_XRGB(136,108,57), },
    {1, -t, 0, D3DCOLOR_XRGB(136,108,57), },
    {0, -1, t, D3DCOLOR_XRGB(136,108,57), },
    {0, 1, t, D3DCOLOR_XRGB(136,108,57), },
    {0, -1, -t, D3DCOLOR_XRGB(136,108,57), },
    {0, 1, -t, D3DCOLOR_XRGB(136,108,57), },
    {t, 0, -1, D3DCOLOR_XRGB(136,108,57), },
    {t, 0, 1, D3DCOLOR_XRGB(136,108,57), },
    {-t, 0, -1, D3DCOLOR_XRGB(136,108,57), },
    {-t, 0, 1, D3DCOLOR_XRGB(136,108,57), },

    {-1, t, 0, D3DCOLOR_XRGB(14,51,180), }, 
    {1, t, 0, D3DCOLOR_XRGB(14,51,180), },
    {-1, -t, 0, D3DCOLOR_XRGB(14,51,180), },
    {1, -t, 0, D3DCOLOR_XRGB(14,51,180), },
    {0, -1, t, D3DCOLOR_XRGB(14,51,180), },
    {0, 1, t, D3DCOLOR_XRGB(14,51,180), },
    {0, -1, -t, D3DCOLOR_XRGB(14,51,180), },
    {0, 1, -t, D3DCOLOR_XRGB(14,51,180), },
    {t, 0, -1, D3DCOLOR_XRGB(14,51,180), },
    {t, 0, 1, D3DCOLOR_XRGB(14,51,180), },
    {-t, 0, -1, D3DCOLOR_XRGB(14,51,180), },
    {-t, 0, 1, D3DCOLOR_XRGB(14,51,180), },

    {-1, t, 0, D3DCOLOR_XRGB(192,49,1), }, 
    {1, t, 0, D3DCOLOR_XRGB(192,49,1), },
    {-1, -t, 0, D3DCOLOR_XRGB(192,49,1), },
    {1, -t, 0, D3DCOLOR_XRGB(192,49,1), },
    {0, -1, t, D3DCOLOR_XRGB(192,49,1), },
    {0, 1, t, D3DCOLOR_XRGB(192,49,1), },
    {0, -1, -t, D3DCOLOR_XRGB(192,49,1), },
    {0, 1, -t, D3DCOLOR_XRGB(192,49,1), },
    {t, 0, -1, D3DCOLOR_XRGB(192,49,1), },
    {t, 0, 1, D3DCOLOR_XRGB(192,49,1), },
    {-t, 0, -1, D3DCOLOR_XRGB(192,49,1), },
    {-t, 0, 1, D3DCOLOR_XRGB(192,49,1), },


};

// create a vertex buffer interface called v_buffer
d3ddev->CreateVertexBuffer(60*sizeof(CUSTOMVERTEX),
                           0,
                           CUSTOMFVF,
                           D3DPOOL_MANAGED,
                           &v_buffer,
                           NULL);

   VOID* pVoid;    // a void pointer

  // lock v_buffer and load the vertices into it
  v_buffer->Lock(0, 0, (void**)&pVoid, 0);
   memcpy(pVoid, vertices, sizeof(vertices));
  v_buffer->Unlock();

   // create the indices using an int array
   short indices[] =
  {
  // 5 faces around p0
  0, 11, 5,
  0, 5, 1,
  0, 1, 7,
  0, 7, 10,
  0, 10, 11,
   // 5 adjacent faces
   1, 5, 9,
   5, 11, 4,
   11, 10, 2,
  10, 7, 6,
   7, 1, 8,
    // 5 faces around point p3
  3, 9, 4,
  3, 4, 2,
  3, 2, 6,
  3, 6, 8,
  3, 8, 9,
  // 5 adjacent faces
  4, 9, 5,
  2, 4, 11,
 6, 2, 10,
  8, 6, 7,
  9, 8, 1
  };

  // create a index buffer interface called i_buffer
 d3ddev->CreateIndexBuffer(60*sizeof(short),
                      0,
                      D3DFMT_INDEX16,
                      D3DPOOL_MANAGED,
                      &i_buffer,
                      NULL);

  // lock i_buffer and load the indices into it
  i_buffer->Lock(0, 0, (void**)&pVoid, 0);
 memcpy(pVoid, indices, sizeof(indices));
  i_buffer->Unlock(); 
 }

1 ответ

Решение

У вас есть два варианта

  1. Вычислите нормали вершин самостоятельно и добавьте их в объявление вершины. в этом случае вам придется рисовать каждую грань отдельно, поскольку вершина разделяется более чем одной гранью. итого вам нужно 12(граней) * 5(каждая грань имеет 5 вершин) = 60 вершин. вершины на одной и той же грани имеют одинаковые нормали, поэтому вам нужно полностью вычислить 12 нормалей.

  2. Создайте D3DXMesh с помощью D3DXCreateMeshFVF и вызовите D3DXComputeNormals, чтобы позволить Direct3D вычислить для вас нормали.

Чтобы использовать освещение в Direct3D

  1. Включить освещение.
  2. Определите вершину с нормалями (или без, если вы используете вариант 2).
  3. Создание света и материалов (материалы описывают, как ваша поверхность будет отражать свет)
  4. Установить свет и материалы.

ссылка на додекаэдр

Другие вопросы по тегам