官方原文:http://developer.sonyeriCSSon.com/site/global/techsupport/tipstrickscode
/mobileJava3d/p_wrapper_JSr184_compliments_digitalchocolate.JSP
JSR-184與Mascot Capsule v3主要的不同就是關於照相機的實現。JSR-184支持照相機結合矩陣堆棧處理,例如,我們經常使用transform對象移動照相機。而Mascot Capsule v3依靠”look-at”方法,這是在某些3D API裡的通用方法。look-at方法從一個position,一個 look-at direction和一個up vector創建一個照相機轉換矩陣,為了方便Mascot Capsule v3與JSR-184之間的轉換,Digital Chocolate采納了Mascot Capsule v3照相機設計方法,並寫了一個支持JSR-184的包。
在JSR-184 API規范裡,NodeTransform類指定了一系列方法。這些方法有利於在JSR-184實現look-at方法。然而,它被認為與JSR-184執行說明規范有所不同,有時甚至會忽略這一系列方法。
自己實現look-at方法其實並不復雜。下面的代碼例子是Digital Chocolate公司如何處理照相機的封裝設計。請注意Digital Chocolate公司在Mascot Capsule v3中使用整數來處理,而在設計更高層的游戲類設計中使用浮點數來處理。
/**
* Wrapper method for setting look at camera.
*
* The method requires that look and up vectors normalized.
*/
public static final void setLookAt(float a_posX, float a_posY, float a_posZ,
float a_lookX, float a_lookY, float a_lookZ,
float a_upX, float a_upY, float a_upZ)
{
// JSR-184 version
if (USE_M
{
// Cross product to get side vector
float sideX = (a_lookY * a_upZ) - (a_lookZ * a_upY);
float sideY = (a_lookZ * a_upX) - (a_lookX * a_upZ);
float sideZ = (a_lookX * a_upY) - (a_lookY * a_upX);
float inv_len = 1.0f /
(float) Java.lang.Math.sqrt(sideX * sideX
+ sideY * sideY
+ sideZ * sideZ);
sideX *= inv_len;
sideY *= inv_len;
sideZ *= inv_len;
// make up vector perpendicular
a_upX = (sideY * a_lookZ) - (sideZ * a_lookY);
a_upY = (sideZ * a_lookX) - (sideX * a_lookZ);
a_upZ = (sideX * a_lookY) - (sideY * a_lookX);
// footnote: up is unit size because side and look are perpendicular
sm_mtx[0] = sideX;
sm_mtx[1] = a_upX;
sm_mtx[2] = -a_lookX;
sm_mtx[3] = a_posX;
sm_mtx[4] = sideY;
sm_mtx[5] = a_upY;
sm_mtx[6] = -a_lookY;
sm_mtx[7] = a_posY;
sm_mtx[8] = sideZ;
sm_mtx[9] = a_upZ;
sm_mtx[10] = -a_lookZ;
sm_mtx[11] = a_posZ;
sm_mtx[12] = 0.0f;
sm_mtx[13] = 0.0f;
sm_mtx[14] = 0.0f;
sm_mtx[15] = 1.0f;
sm_m3gTransform.set(sm_mtx);
}
// Mascot version
if (USE_MASCOT)
{
sm_mascotTmpVectorA.set((int)a_posX, (int)a_posY, (int)a_posZ);
sm_mascotTmpVectorB.set((int)(a_lookX * MASCOT_ONE),
(int)(a_lookY * MASCOT_ONE),
(int)(a_lookZ * MASCOT_ONE));
sm_mascotTmpVectorC.set(0, DajmGraphics.MASCOT_ONE, 0);
sm_mascotAffineTrans.lookAt(sm_mascotTmpVectorA,
sm_mascotTmpVectorB,
sm_mascotTmpVectorC);
}
}