// Copyright 2014 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. package app /* #cgo android LDFLAGS: -llog -landroid -lEGL -lGLESv2 #include #include #include #include #include // TODO(crawshaw): Test configuration on more devices. const EGLint RGB_888[] = { EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT, EGL_SURFACE_TYPE, EGL_WINDOW_BIT, EGL_BLUE_SIZE, 8, EGL_GREEN_SIZE, 8, EGL_RED_SIZE, 8, EGL_DEPTH_SIZE, 16, EGL_CONFIG_CAVEAT, EGL_NONE, EGL_NONE }; EGLint windowWidth; EGLint windowHeight; EGLDisplay display; EGLSurface surface; #define LOG_ERROR(...) __android_log_print(ANDROID_LOG_ERROR, "Go", __VA_ARGS__) void createEGLWindow(ANativeWindow* window) { EGLint numConfigs, format; EGLConfig config; EGLContext context; display = eglGetDisplay(EGL_DEFAULT_DISPLAY); if (!eglInitialize(display, 0, 0)) { LOG_ERROR("EGL initialize failed"); return; } if (!eglChooseConfig(display, RGB_888, &config, 1, &numConfigs)) { LOG_ERROR("EGL choose RGB_888 config failed"); return; } if (numConfigs <= 0) { LOG_ERROR("EGL no config found"); return; } eglGetConfigAttrib(display, config, EGL_NATIVE_VISUAL_ID, &format); if (ANativeWindow_setBuffersGeometry(window, 0, 0, format) != 0) { LOG_ERROR("EGL set buffers geometry failed"); return; } surface = eglCreateWindowSurface(display, config, window, NULL); if (surface == EGL_NO_SURFACE) { LOG_ERROR("EGL create surface failed"); return; } const EGLint contextAttribs[] = { EGL_CONTEXT_CLIENT_VERSION, 2, EGL_NONE }; context = eglCreateContext(display, config, EGL_NO_CONTEXT, contextAttribs); if (eglMakeCurrent(display, surface, surface, context) == EGL_FALSE) { LOG_ERROR("eglMakeCurrent failed"); return; } eglQuerySurface(display, surface, EGL_WIDTH, &windowWidth); eglQuerySurface(display, surface, EGL_HEIGHT, &windowHeight); } #undef LOG_ERROR */ import "C" import ( "log" "code.google.com/p/go.mobile/event" "code.google.com/p/go.mobile/geom" "code.google.com/p/go.mobile/gl" ) func windowDrawLoop(cb Callbacks, w *C.ANativeWindow, queue *C.AInputQueue) { C.createEGLWindow(w) gl.Enable(gl.CULL_FACE) gl.ClearColor(0, 0, 0, 1) gl.Clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT) C.eglSwapBuffers(C.display, C.surface) if errv := gl.GetError(); errv != gl.NO_ERROR { log.Printf("GL initialization error: %s", errv) } geom.Width = geom.Pt(float32(C.windowWidth) / geom.Scale) geom.Height = geom.Pt(float32(C.windowHeight) / geom.Scale) for { processEvents(cb, queue) select { case <-windowDestroyed: return default: cb.Draw() C.eglSwapBuffers(C.display, C.surface) } } } func processEvents(cb Callbacks, queue *C.AInputQueue) { var event *C.AInputEvent for C.AInputQueue_getEvent(queue, &event) >= 0 { if C.AInputQueue_preDispatchEvent(queue, event) != 0 { continue } processEvent(cb, event) C.AInputQueue_finishEvent(queue, event, 0) } } func processEvent(cb Callbacks, e *C.AInputEvent) { switch C.AInputEvent_getType(e) { case C.AINPUT_EVENT_TYPE_KEY: log.Printf("TODO input event: key") case C.AINPUT_EVENT_TYPE_MOTION: if cb.Touch == nil { return } x := C.AMotionEvent_getX(e, 0) y := C.AMotionEvent_getY(e, 0) var ty event.TouchType switch C.AMotionEvent_getAction(e) { case C.AMOTION_EVENT_ACTION_DOWN: ty = event.TouchStart case C.AMOTION_EVENT_ACTION_MOVE: ty = event.TouchMove case C.AMOTION_EVENT_ACTION_UP: ty = event.TouchEnd } cb.Touch(event.Touch{ Type: ty, Loc: geom.Point{ X: geom.Pt(float32(x) / geom.Scale), Y: geom.Pt(float32(y) / geom.Scale), }, }) default: log.Printf("unknown input event, type=%d", C.AInputEvent_getType(e)) } }