Android 系列 5.2使用OpenGL ES繪製旋轉立方體
阿新 • • 發佈:2019-02-19
5.2使用OpenGL ES繪製旋轉立方體
問題
您想建立一個基本的OpenGL ES應用程式。
解
建立一個GLSurfaceView和一個自定義渲染器,將繪製一個旋轉立方體。
討論
Android通過OpenGL ES API支援3D圖形,這是一種專為嵌入式裝置設計的OpenGL風格。這個食譜不是OpenGL教程;它假設讀者已經有了基本的OpenGL知識。最終結果如圖5-2所示。
圖5-2。 GL圖形樣本
首先我們寫一個新的Activity,並在onCreate方法中建立我們需要使用OpenGL API的兩個基本物件:一個GLSurfaceView和一個Renderer(見例5-3)。
實施例5-3。 OpenGL示例活動
例5-4是我們的Renderer的程式碼,它使用一個簡單的Cube物件,我們將在後面描述它顯示一個旋轉立方體。
例項5-4。呈現的實現
實施例5-5。 Cube類
問題
您想建立一個基本的OpenGL ES應用程式。
解
建立一個GLSurfaceView和一個自定義渲染器,將繪製一個旋轉立方體。
討論
Android通過OpenGL ES API支援3D圖形,這是一種專為嵌入式裝置設計的OpenGL風格。這個食譜不是OpenGL教程;它假設讀者已經有了基本的OpenGL知識。最終結果如圖5-2所示。
圖5-2。 GL圖形樣本
首先我們寫一個新的Activity,並在onCreate方法中建立我們需要使用OpenGL API的兩個基本物件:一個GLSurfaceView和一個Renderer(見例5-3)。
實施例5-3。 OpenGL示例活動
public class OpenGLDemoActivity extends Activity { @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // Go fullscreen requestWindowFeature(Window.FEATURE_NO_TITLE); getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN); GLSurfaceView view = new GLSurfaceView(this); view.setRenderer(new OpenGLRenderer()); setContentView(view); } }
例5-4是我們的Renderer的程式碼,它使用一個簡單的Cube物件,我們將在後面描述它顯示一個旋轉立方體。
例項5-4。呈現的實現
我們的onSurfaceChanged和onDrawFrame方法基本上等同於GLUT glutReshapeFunc和glutDisplayFunc。第一個是在表面調整大小時呼叫的,例如,當手機在橫向和縱向模式之間切換時。第二個在每一幀被呼叫,這就是我們把程式碼繪製我們的立方體(見例5-5)。class OpenGLRenderer implements Renderer { private Cube mCube = new Cube(); private float mCubeRotation; @Override public void onSurfaceCreated(GL10 gl, EGLConfig config) { gl.glClearColor(0.0f, 0.0f, 0.0f, 0.5f); gl.glClearDepthf(1.0f); gl.glEnable(GL10.GL_DEPTH_TEST); gl.glDepthFunc(GL10.GL_LEQUAL); gl.glHint(GL10.GL_PERSPECTIVE_CORRECTION_HINT, GL10.GL_NICEST); } @Override public void onDrawFrame(GL10 gl) { gl.glClear(GL10.GL_COLOR_BUFFER_BIT | GL10.GL_DEPTH_BUFFER_BIT); gl.glLoadIdentity(); gl.glTranslatef(0.0f, 0.0f, -10.0f); gl.glRotatef(mCubeRotation, 1.0f, 1.0f, 1.0f); mCube.draw(gl); gl.glLoadIdentity(); mCubeRotation -= 0.15f; } @Override public void onSurfaceChanged(GL10 gl, int width, int height) { gl.glViewport(0, 0, width, height); gl.glMatrixMode(GL10.GL_PROJECTION); gl.glLoadIdentity(); GLU.gluPerspective(gl, 45.0f, (float)width / (float)height, 0.1f, 100.0f); gl.glViewport(0, 0, width, height); gl.glMatrixMode(GL10.GL_MODELVIEW); gl.glLoadIdentity(); } }
實施例5-5。 Cube類
立方體使用兩個FloatBuffer物件來儲存頂點和顏色資訊,以及一個ByteBuffer來儲存面索引。為了使緩衝區工作,重要的是根據平臺的位元組順序使用順序方法來設定它們的順序。一旦緩衝區已經用陣列中的值填充,內部游標必須使用buffer.position(0)恢復到資料的開頭。class Cube { private FloatBuffer mVertexBuffer; private FloatBuffer mColorBuffer; private ByteBuffer mIndexBuffer; private float vertices[] = { -1.0f, -1.0f, -1.0f, 1.0f, -1.0f, -1.0f, 1.0f, 1.0f, -1.0f, -1.0f, 1.0f, -1.0f, -1.0f, -1.0f, 1.0f, 1.0f, -1.0f, 1.0f, 1.0f, 1.0f, 1.0f, -1.0f, 1.0f, 1.0f }; private float colors[] = { 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 1.0f, 1.0f, 0.5f, 0.0f, 1.0f, 1.0f, 0.5f, 0.0f, 1.0f, 1.0f, 0.0f, 0.0f, 1.0f, 1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 1.0f, 1.0f, 1.0f, 0.0f, 1.0f, 1.0f }; private byte indices[] = { 0, 4, 5, 0, 5, 1, 1, 5, 6, 1, 6, 2, 2, 6, 7, 2, 7, 3, 3, 7, 4, 3, 4, 0, 4, 7, 6, 4, 6, 5, 3, 0, 1, 3, 1, 2 }; public Cube() { ByteBuffer byteBuf = ByteBuffer.allocateDirect(vertices.length * 4); byteBuf.order(ByteOrder.nativeOrder()); mVertexBuffer = byteBuf.asFloatBuffer(); mVertexBuffer.put(vertices); mVertexBuffer.position(0); byteBuf = ByteBuffer.allocateDirect(colors.length * 4); byteBuf.order(ByteOrder.nativeOrder()); mColorBuffer = byteBuf.asFloatBuffer(); mColorBuffer.put(colors); mColorBuffer.position(0); mIndexBuffer = ByteBuffer.allocateDirect(indices.length); mIndexBuffer.put(indices); mIndexBuffer.position(0); } public void draw(GL10 gl) { gl.glFrontFace(GL10.GL_CW); gl.glVertexPointer(3, GL10.GL_FLOAT, 0, mVertexBuffer); gl.glColorPointer(4, GL10.GL_FLOAT, 0, mColorBuffer); gl.glEnableClientState(GL10.GL_VERTEX_ARRAY); gl.glEnableClientState(GL10.GL_COLOR_ARRAY); gl.glDrawElements(GL10.GL_TRIANGLES, 36, GL10.GL_UNSIGNED_BYTE, mIndexBuffer); gl.glDisableClientState(GL10.GL_VERTEX_ARRAY); gl.glDisableClientState(GL10.GL_COLOR_ARRAY); } }