CCamera.cpp
上传用户:arsena_zhu
上传日期:2022-07-12
资源大小:399k
文件大小:8k
源码类别:

OpenGL

开发平台:

Visual C++

  1. /*
  2.    Class Name:
  3.       CCamera.
  4.    Created by:
  5.       Allen Sherrod (Programming Ace of www.UltimateGameProgramming.com).
  6.    Description:
  7.       This class represents a camera in a 3D scene.
  8. */
  9. #include"CCamera.h"
  10. CCamera::CCamera()
  11. {
  12.    // Initialize variables...
  13.    xPos = yPos = zPos = 0;
  14.    xView = yView = zView = 0;
  15.    xUp = yUp = zUp = 0;
  16.    xStrafe = yStrafe = zStrafe = 0;
  17. }
  18. void CCamera::SetCamera(float x, float y, float z,
  19.                         float xv, float yv, float zv,
  20.                         float xu, float yu, float zu)
  21. {
  22.    // Here we set the camera to the values sent in to us.  This is mostly used to set up a
  23.    // default position.
  24.    xPos = x;   yPos = y;   zPos = z;
  25.    xView = xv; yView = yv; zView = zv;
  26.    xUp = xu;   yUp = yu;   zUp = zu;
  27. }
  28. void CCamera::MoveCamera(float direction)
  29. {
  30.    // Moving the camera requires a little more then adding 1 to the z or subracting 1.
  31.    // First we need to get the direction at which we are looking.
  32.    float xLookDirection = 0, yLookDirection = 0, zLookDirection = 0;
  33.    // The look direction is the view minus the position (where we are).
  34.    xLookDirection = xView - xPos;
  35. yLookDirection = yView - yPos;
  36. zLookDirection = zView - zPos;
  37.    // Normalize the direction.
  38.    float dp = 1 /(float)sqrt(xLookDirection * xLookDirection + yLookDirection * yLookDirection +
  39.                              zLookDirection * zLookDirection);
  40.    xLookDirection *= dp;
  41.    yLookDirection *= dp;
  42.    zLookDirection *= dp;
  43.    // Call UpdateCamera to move our camera in the direction we want.
  44.    UpdateCamera(xLookDirection, yLookDirection, zLookDirection, direction);
  45. }
  46. void CCamera::UpdateCamera(float xDir, float yDir, float zDir, float dir)
  47. {
  48.    // Move the camera on the X and Z axis.  Notice I commented out the yPos and yView
  49.    // updates.  This is because without them we can keep the camera on the ground in
  50.    // the simple camera tutorials without having to alter them afterwards.
  51. xPos += xDir * dir;
  52.    yPos += yDir * dir;
  53. zPos += zDir * dir;
  54.    // Move the view along with the position
  55. xView += xDir * dir;
  56.    yView += yDir * dir;
  57. zView += zDir * dir;
  58. }
  59. void CCamera::StrafeCam(float direction)
  60. {
  61.    // Calculate the strafe then update the camera position.
  62.    CalculateStrafe();
  63.    UpdateCamera(xStrafe, yStrafe, zStrafe, direction);
  64. }
  65. void CCamera::CalculateStrafe()
  66. {
  67.    float xDir = 0, yDir = 0, zDir = 0;
  68.    float xCross = 0, yCross = 0, zCross = 0;
  69.    // Strafing is just like moving the camera forward and backward.
  70.    // First we will get the direction we are looking.
  71.    xDir = xView - xPos;
  72. yDir = yView - yPos;
  73. zDir = zView - zPos;
  74.    // Normalize the direction.
  75.    float dp = 1 /(float)sqrt(xDir * xDir + yDir * yDir + zDir * zDir);
  76.    xDir *= dp;
  77.    yDir *= dp;
  78.    zDir *= dp;
  79.    // Now if we were to call UpdateCamera() we will be moving the camera foward or backwards.
  80.    // We don't want that here.  We want to strafe.  To do so we have to get the cross product
  81.    // of our direction and Up direction view.  The up was set in SetCamera to be 1 positive
  82.    // y.  That is because anything positive on the y is considered up.  After we get the
  83.    // cross product we can save it to the strafe variables so that can be added to the
  84.    // camera using UpdateCamera().
  85.    // Get the cross product of the direction we are looking and the up direction.
  86.    xCross = (yDir * zUp) - (zDir * yUp);
  87.    yCross = (zDir * xUp) - (xDir * zUp);
  88.    zCross = (xDir * yUp) - (yDir * xUp);
  89.    // Save our strafe (cross product) values in xStrafe, yStrafe, and zStrafe.
  90.    xStrafe = xCross;
  91.    yStrafe = yCross;
  92.    zStrafe = zCross;
  93. }
  94. void CCamera::RotateCamera(float AngleDir, float xSpeed, float ySpeed, float zSpeed)
  95. {
  96. float xNewLookDirection = 0, yNewLookDirection = 0, zNewLookDirection = 0;
  97. float xLookDirection = 0, yLookDirection = 0, zLookDirection = 0;
  98.    float CosineAngle = 0, SineAngle = 0;
  99.    // First we will need to calculate the cos and sine of our angle.  I creaetd two macros to
  100.    // do this in the CCamera.h header file called GET_COS and GET_SINE.  To use the macros
  101.    // we just send in the variable we ant to store the results and the angle we need to
  102.    // calculate.
  103.    CosineAngle = (float)cos(AngleDir);
  104.    SineAngle = (float)sin(AngleDir);
  105. // Next get the look direction (where we are looking) just like in the move camera function.
  106.    xLookDirection = xView - xPos;
  107. yLookDirection = yView - yPos;
  108. zLookDirection = zView - zPos;
  109.    // Normalize the direction.
  110.    float dp = 1 /(float)sqrt(xLookDirection * xLookDirection + yLookDirection * yLookDirection +
  111.                              zLookDirection * zLookDirection);
  112.    xLookDirection *= dp;
  113.    yLookDirection *= dp;
  114.    zLookDirection *= dp;
  115. // Calculate the new X position.
  116. xNewLookDirection = (CosineAngle + (1 - CosineAngle) * xSpeed) * xLookDirection;
  117. xNewLookDirection += ((1 - CosineAngle) * xSpeed * ySpeed - zSpeed * SineAngle)* yLookDirection;
  118. xNewLookDirection += ((1 - CosineAngle) * xSpeed * zSpeed + ySpeed * SineAngle) * zLookDirection;
  119. // Calculate the new Y position.
  120. yNewLookDirection = ((1 - CosineAngle) * xSpeed * ySpeed + zSpeed * SineAngle) * xLookDirection;
  121. yNewLookDirection += (CosineAngle + (1 - CosineAngle) * ySpeed) * yLookDirection;
  122. yNewLookDirection += ((1 - CosineAngle) * ySpeed * zSpeed - xSpeed * SineAngle) * zLookDirection;
  123. // Calculate the new Z position.
  124. zNewLookDirection = ((1 - CosineAngle) * xSpeed * zSpeed - ySpeed * SineAngle) * xLookDirection;
  125. zNewLookDirection += ((1 - CosineAngle) * ySpeed * zSpeed + xSpeed * SineAngle) * yLookDirection;
  126. zNewLookDirection += (CosineAngle + (1 - CosineAngle) * zSpeed) * zLookDirection;
  127. // Last we add the new rotations to the old view to correctly rotate the camera.
  128. xView = xPos + xNewLookDirection;
  129. yView = yPos + yNewLookDirection;
  130. zView = zPos + zNewLookDirection;
  131. }
  132. void CCamera::RotateByMouse(int mousePosX, int mousePosY, int midX, int midY)
  133. {
  134. float yDirection = 0.0f;         // Direction angle.
  135. float yRotation = 0.0f;          // Rotation angle.
  136. // If the mouseX and mouseY are at the middle of the screen then we can't rotate the view.
  137. if((mousePosX == midX) && (mousePosY == midY)) return;
  138. // Next we get the direction of each axis.  We divide by 1000 to get a smaller value back.
  139. yDirection = (float)((midX - mousePosX)) / 1000.0f;
  140. yRotation = (float)((midY - mousePosY)) / 1000.0f;
  141. // We use curentRotX to help use keep the camera from rotating too far in either direction.
  142. currentRotationAngle -= yRotation;  
  143. // Stop the camera from going to high...
  144. if(currentRotationAngle > 1.67f)
  145.       {
  146.    currentRotationAngle = 1.67f;
  147.          return;
  148.       }
  149. // Stop the camera from going to low...
  150. if(currentRotationAngle < -1.67f)
  151. {
  152.          currentRotationAngle = -1.67f;
  153.          return;
  154.       }
  155.    // Next we get the axis which is a perpendicular vector of the view direction and up values.
  156.    // We use the cross product of that to get the axis then we normalize it.
  157.    float xAxis = 0, yAxis = 0, zAxis = 0;
  158.    float xDir = 0, yDir = 0, zDir = 0;
  159.    // Get the Direction of the view.
  160.    xDir = xView - xPos;
  161. yDir = yView - yPos;
  162. zDir = zView - zPos;
  163.    // Get the cross product of the direction and the up.
  164.    xAxis = (yDir * zUp) - (zDir * yUp);
  165.    yAxis = (zDir * xUp) - (xDir * zUp);
  166.    zAxis = (xDir * yUp) - (yDir * xUp);
  167.    // Normalize it.
  168.    float len = 1 /(float)sqrt(xAxis * xAxis + yAxis * yAxis + zAxis * zAxis);
  169.    xAxis *= len;
  170.    yAxis *= len;
  171.    zAxis *= len;
  172.    // Rotate the camera.
  173.    RotateCamera(yRotation, xAxis, yAxis, zAxis);
  174.    RotateCamera(yDirection, 0, 1, 0);
  175. }
  176. // Copyright October 2004
  177. // All Rights Reserved!
  178. // Allen Sherrod
  179. // ProgrammingAce@UltimateGameProgramming.com
  180. // www.UltimateGameProgramming.com