출처 : http://i5on9i.blogspot.kr/2013/11/glsurfaceview.html
penGL ES 를 사용해 보자. 안드로이드에서 API 를 제공한다.
GLSurfaceView 라는 모양으로 제공한다.
GLSurfaceView
간단한 GLSurfaceView 를 이용한 application 을 만들어보자. 예제는 ref. 1을 참고했다.GLSurfaceView 사용
public class ClearActivity extends Activity { private GLSurfaceView mGLView; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); mGLView = new GLSurfaceView(this); mGLView.setRenderer(new ClearRenderer()); setContentView(mGLView); } @Override protected void onPause() { super.onPause(); mGLView.onPause(); } @Override protected void onResume() { super.onResume(); mGLView.onResume(); } }
class ClearRenderer implements GLSurfaceView.Renderer { public void onSurfaceCreated(GL10 gl, EGLConfig config) { // Do nothing special. } public void onSurfaceChanged(GL10 gl, int w, int h) { gl.glViewport(0, 0, w, h); } public void onDrawFrame(GL10 gl) { gl.glClear(GL10.GL_COLOR_BUFFER_BIT | GL10.GL_DEPTH_BUFFER_BIT); } }
- onSurfaceCreated() :
rendering(렌더링) 을 시작할 때와 OpenGL ES drawing 컨텍스트가 다시 만들어져야 할 때 호출된다. - onSurfaceChanged() :
surface 의 size 가 바뀔 때 호출된다. 여기서 OpneGL viewport 를 설정하면 된다.
scene 주위를 움직이지 않는 fixed camera 라면, camera 도 여기서 set 하면 된다. - onDrawFrame() :
모든 frame 마다 이 함수가 호출된다. 이 함수에서 scene 을 그린다.
일반적으로 시작할 때 framebuffer 를 clear 하기 위해 glClear 를 호출한다. 그리고 그 뒤로 현재 scene 을 그리기 위한 다른 OpenGl ES 호출들이 따라온다.
User Input 을 위한 GLSurfaceView 상속
실제 draw 부분은 renderer 에서 작업해 주고, GLSurfaceView 에서는 event 를 날려주기만 하면 된다.
queueEvent
event 를 renderer thread 에게 던져주는 함수이다.
renderer thread 에게 event 를 주는 것이기 때문에 renderer 가 set 되어 있어야 한다.
class ClearGLSurfaceView extends GLSurfaceView { public ClearGLSurfaceView(Context context) { super(context); mRenderer = new ClearRenderer(); setRenderer(mRenderer); setRenderMode(GLSurfaceView.RENDERMODE_WHEN_DIRTY); } public boolean onTouchEvent(final MotionEvent event) { queueEvent(new Runnable(){ public void run() { mRenderer.setColor(event.getX() / getWidth(), event.getY() / getHeight(), 1.0f); requestRender(); }}); return true; } ClearRenderer mRenderer; } class ClearRenderer implements GLSurfaceView.Renderer { public void onSurfaceCreated(GL10 gl, EGLConfig config) { // Do nothing special. } public void onSurfaceChanged(GL10 gl, int w, int h) { gl.glViewport(0, 0, w, h); } public void onDrawFrame(GL10 gl) { gl.glClearColor(mRed, mGreen, mBlue, 1.0f); gl.glClear(GL10.GL_COLOR_BUFFER_BIT | GL10.GL_DEPTH_BUFFER_BIT); } public void setColor(float r, float g, float b) { mRed = r; mGreen = g; mBlue = b; } private float mRed; private float mGreen; private float mBlue; }
Activity
GLSurfaceView 에서 Renderer 를 set 해주기 때문에, Activity 에서는 GLSurfaceView 만 set 해주면 된다.@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mGLView = new ClearGLSurfaceView(this);
setContentView(mGLView);
}
Fragment
Fragment 에서는 아래처럼 해주면 된다.@Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View fragment = inflater.inflate(R.layout.main_fragment, container, false); mGLView = new ClearGLSurfaceView(fragment.getContext()); return mGLView; }
GLThread in GLSurfaceView.java
- GLSurfaceView 는 Window 에 attach 될 때 GLThread 를 만들게 된다.
- 이 GLThread 가 renderer thread 인데 이 thread 가 guardedRun() 을 호출 한다.
- 이 gurdedRun() 이 무한 루프를 돌면서 mEventQueue 에 있는 event 들을 run 시킨다.
@Override protected void onAttachedToWindow() { super.onAttachedToWindow(); ... if (mDetached && (mRenderer != null)) { ... mGLThread = new GLThread(mThisWeakRef); if (renderMode != RENDERMODE_CONTINUOUSLY) { mGLThread.setRenderMode(renderMode); } mGLThread.start(); } mDetached = false; } //////////////////////////// // inner static class //////////////////////////// static class GLThread extends Thread { ... @Override public void run() { setName("GLThread " + getId()); ... try { guardedRun(); } catch (InterruptedException e) { // fall thru and exit normally } finally { sGLThreadManager.threadExiting(this); } } }
@Override protected void onDetachedFromWindow() { ... if (mGLThread != null) { mGLThread.requestExitAndWait(); } mDetached = true; super.onDetachedFromWindow(); }
private ArrayListmEventQueue = new ArrayList (); public void queueEvent(Runnable r) { if (r == null) { throw new IllegalArgumentException("r must not be null"); } synchronized(sGLThreadManager) { mEventQueue.add(r); sGLThreadManager.notifyAll(); } } private void guardedRun() throws InterruptedException { while (true) { synchronized (sGLThreadManager) { while (true) { if (mShouldExit) { return; } if (! mEventQueue.isEmpty()) { event = mEventQueue.remove(0); break; } ... } } // end of synchronized(sGLThreadManager) if (event != null) { event.run(); event = null; continue; } } }
RENDERMODE_WHEN_DIRTY
3D 게임같은 프로그램에서는 계속해서 화면을 모두 다 그려야 줄 필요가 있지만, 일반적인 어플리케이션에서는 굳이 그렇게 할 필요가 없다. 그저 변경된 부분만 다시 그려주면 된다. 이것은 아래처럼 render mode 를 WHEN_DIRTY 로 설정 해 주면 된다.- GLSurfaceView.setRenderMode(RENDERMODE_WHEN_DIRTY),
그리고 나서, 필요한 곳에서 rendering 을 요청하면 된다. rendering 의 요청은 아래로 가능하다.
- GLSurfaceView.requestRender()
glFrontFace(GL10.GL_CW)
GL_CW 는 clockwise, GL_CCW 는 count-clockwise 이다. 이 방향이 앞면 폴리곤(front-facing polygon) 의 방향이 된다.
(Specifies the orientation of front-facing polygons)
아래같이 vertex 가 4개 있다고 하자. 이때 vertex 의 순서를 glDrawElements() 로 넘겨줘서 polygon 을 그리게 한다.
(IndexBuffer 를 만드는 과정은 일단 생략한다.)
- byte indices[] = { 1, 2, 3, 2, 4, 3 }
- gl.glDrawElements(GL10.GL_TRIANGLES, indices.length, GL10.GL_UNSIGNED_BYTE, mIndexBuffer);
이 때 glFrontFace(GL10.GL_CW) 라면, 오른쪽 방향의 순서대로 vertex 들의 순서를 정하면, 그 순서로 만들어진 polygon 이 정면이 된다.
만약 순서를 vertex 순서를 시계반대방향(왼쪽)으로 순서를 정하면 그 polygon 은 뒷면이 된다.
예를 들어 만약 indices 를 아래처럼 set 하면, 아마 반쪽짜리 삼각형 polygon 을 볼 것이다.
그 이유는 1 -> 2 -> 3 은 정상적으로 삼각형을 보여줄 테지만, 3 -> 4 -> 2 는 한바퀴 돌려야 보인다.
- byte indices[] = { 1, 2, 3, 3, 4, 2 }
Other examples
ApiDemo 에 보면 예제들이 더 있다.- <Android_sdk>\samples\android-19\legacy\ApiDemos\src\com\example\android\apis\graphics\
- GLSurfaceView - a spinning triangle
- Kube - a cube puzzle demo
- Translucent GLSurfaceView - shows how to display 3D graphics on a translucent background
- Textured Triangle - shows how to draw a textured 3D triangle
- Sprite Text - shows how to draw text into a texture and then composite it into a 3D scene
- Touch Rotate - shows how to rotate a 3D object in response to user input.
GLSurfaceView 에 대한 좀 더 많은 정보는 ref. 1 을 참고하자.
See Also
- How to capture the app's screen
How to programatically take a screenshot on Android?, StackOverflow
Reference
- Introducing GLSurfaceView | Android Developers Blog
- http://openaphid.github.io/blog/2012/05/21/how-to-implement-flipboard-animation-on-android/
android.opengl.GLSurfaceview 클래스는 아래와 같은 기능을 제공해 어플리케이션에서 OpenGL ES 렌더링을 쉽게 사용할수 있도록 해준다.
- View 시스템에 OpenGL ES를 연결할수 있는 연결코드를 제공한다.
- OpenGL ES 작업이 Activity 라이프 싸이클과 함께 동작하도록 해주는 연결코드를 제공한다.
- 적절한 프레임 버퍼 픽셀 형식을 쉽게 선택할수 있도록 해준다.
- 부드러운 에니메이션이 가능하도록 분리된 렌더링 쓰레드를 생성하고 관리할수 있게 한다.
- OpenGL ES API 호출 추적과 에러 체크를 위한 사용하기 쉬운 디버깅 툴을 제공한다.
GLSurfaceView는 부분적으로 혹은 전체적으로 렌더링에 사용하는 어플리케이션을 작성하기 위해 좋은 기반이 된다.
Google Map StreetView 와 같은 2D, 3D 데이터 비쥬얼 어플리케이션과 같은 2D, 3D 액션 게임이 좋은 후보가 될 것이다.
단순한 GLSurfaceView 어플리케이션
가장 단순한 OpenGL ES 어플리케이션의 소스코드를 보자.
이 프로그램은 많은 일을 하지는 않는다. 모든 프레임에 검은색으로 화면을 지운다.
하지만 안드로이드 activity 라이프 싸이클을 정확하게 구현한 완벽한 OpenGL 어플리케이션이다.
이 프로그램은 activity가 pause 상태일때 렌더링을 pause 하고 activity가 resume 될때 렌더링을 resume 한다.
이 프로그램을 기본적인 비상호적 데모 프로그램으로 사용할 수 있다.
ClearRenderer.onDrawFrame()메서드를 호출하기 위해 단순히 몇개의 OpenGL 호출을 추가했다.
GLSurfaceview 를 상속할 필요가 없음에 주목하자.
GLSurfaceView.Render 인터페이스는 3가지 메소드를 가지는데
- onSurfaceCreate() 메서드는 렌더링의 시작 시점과 OpenGL ES 드로잉 context가 재생성될때마다 호출된다.
(보통 드로잉 context는 activity가 pause나 resume 될때 소멸되어 다시 재생성된다.)
OnSurfaceCreate()는 텍스쳐(역자 주 : 이미지를 OpenGL 객체에 입히는 작업)와 같은 생명 주기가 긴 OpenGL 리소스를 생성하기에 적절한 곳이다. - onSurfaceChanged() 메서드는 surface가 크기를 변경할 경우 호출된다. 이것은 OpenGL viewport를 설정하기에 적절한 곳이다.
장면주위를 움직이지 않는 고정된 카메라라면 여기서 카메라 설정을 할수 있다. - onDrawFrame() 메서드는 모든 프레임에서 호출되고 장면을 그리는 책임을 진다.
현재 장면을 그리기 위해 다른 OpenGL ES 호출 다음에 보통 프레임 버터를 삭제하기 위해 glClear를 호출할때 시작된다.
사용자 입력은 어떡해?
게임과 같은 상호 작용을 하는 어플리케이션을 원한다면 입력 이벤트를 받는 쉬운 방법을 제공받기 위해 GLSurfaceView를 상속해야한다.
어떻게 하는지 보여주는 조금 긴 예제를 보자.
이 어플리케이션은 모든 프레임을 위해 화면을 지운다.
사용자가 화면을 tap할때 사용자 touch 이벤트의 좌표값(x,y)에 기반해 색을 지운다.
ClearGLSurfaceView.onTouchEvent()의 queueEvent()사용에 주목하자.
queueEvent() 메서드는 UI 쓰레드와 렌더링 쓰레드간의 안전한 통신을 위해 사용된다.
원한다면 Renderer클래스 자체에서 동기화 메서드와 같은 자바 쓰레드간 통신기술을 사용할 수 있다.
하지만, 이벤트를 큐에 담는 것이 쓰레드간 통신을 다루는 것보다 더 안전한 방법이다.
다른 GLSurfaceView 샘플
단순히 화면을 지우는 것이 지루한가? 안드로이드 SDK에 포함된 API데모를 통해 더 재미있는 예제들을 볼수 있다.
모든 OpenGL ES 샘플들은 GLSurfaceView를 사용하기 위해 변경 되었다.
- GLSurfaceView - 삼각형 회전
- Kube - 큐브 퍼즐 데모
- 반투명 GLSurface 뷰 - 반투명 배경에서 3D 그래픽이 어떡해 디스플레이되는지를 보여준다.
- 텍스쳐 삼각형 : 텍스쳐된 3D 삼각형을 어떡해 그리는지 보여준다.
- 스프라이트 텍스트 - 텍스쳐에 텍스트를 그리고 3D 장면에 어떡해 구성되는지를 보여준다.
- 터치 회전 : 사용자 입력에 반응하여 3D객체를 어떡해 회전시키는지를 보여준다.
surface 선택
GLSurfaceView는 렌더링하고자하는 surface의 유형을 선택할수 있도록 도와준다.
서로 다른 안드로이드 디바이스 공통 부분을 가지지 않는 서로 다른 Surface 유형을 지원한다.
이것은 각 디바이스별로 가장 적절한 Surface를 선택하는데 어려움을 준다.
디폴트로 GLSurfaceView는 16비트 깊이를 가지는 버퍼와 16비트 RGB 프레임 버퍼에 가능한 가까운 surface를 찾으려고 시도한다.
개발하고자하는 어플리케이션의 니즈에 따라 이 동작을 변경하고자 할수도 있을 것이다.
예를 들어 반투명 GLSurfaceView 샘플은 반투명 데이터를 렌더링 하기 위해 알파 채널을 필요로 할 것이다.
GLSurfaceView는 어떤 surface 유형을 사용할지 선택할수 있도록 하기 위해 setEGLSurfaceChooser() 메서드를 재정의할수 있도록 한다.
setEGLConfigChooser(boolean needDepth)
16비트 프레임 버퍼를 갖지 않을수 있는 R5G6B5에 가장 가까운 구성을 선택한다.
setEGLConfigChooser(int redSize, int greenSize, int blueSize, int alphaSize, int depthSize, int stencilSize)
적어도 생성자에서 명시된 것만큼의 채널당 비트수를 가지는 픽셀당 비트수가 가장 적은 구성을 선택한다.
setEGLConfigChooser(EGLConfigChooser configChooser)
구성 선택에 대한 총괄적인 제어를 가능하도록 해준다.
디바이스의 성능을 조사하여 구성을 선택할수 있는 EGLConfigChooser를 직접 구현하여 전달할수 있다.
지속적인 렌더링 v.s 필요할때 렌더링
게임과 시뮬레이션과 같은 대부분의 3D 어플리케이션은 지속적으로 에니메이션된다.
그러나 몇몇 3D어플리케이션은 더 많은 반응을 하게되는데 사용자가 무언가를 수행할때까지 기다리고 그것에 대응하는 것과 같은 것이다.
이런 유형의 어플리케이션들위해 화면을 지속적으로 다시 그리는 디폴트 GLSurfaceView 동작은 시간 낭비이다.
반응하는 어플리케이션을 개발한다면 지속적인 에니메이션 하지는 GLSurfaceView.setRenderMode(RENDERMODE_WHEN_DIRTY) 를 호출 할 수 있고 다시 렌더링을 원할때는 GLSurfaceView.requestRender()를 호출 할 수 있다.
디버깅의 도움
GLSurfaceView는 OpenGL ES 어플리케이션 디버깅을 위해 유용한 내장 기능을 가지고 있다.
GLSurface.setDebugFlags() 메서드는 OpenGL ES호출시 오류를 확인 하거나 로깅을 가능하게 하는데 사용될 수 있다.
setRenderer()를 호출하기 전에 GLSurfaceView의 생성자에서 이메소드를 호출한다.
'IT_Programming > Android_Java' 카테고리의 다른 글
[펌] 안드로이드 텍스트 뷰에서 지원하는 HTML 태그들 (0) | 2015.09.10 |
---|---|
[펌] 안드로이드 오픈지엘(OpenGL ES2.0)의 기본 (0) | 2015.09.08 |
GLSurfaceView 배경 투명하게 만들기 (0) | 2015.09.08 |
[펌] 실질적인 안드로이드 디자인 (0) | 2015.08.24 |
[펌] Android Design Support Library를 소개합니다 (0) | 2015.08.19 |