当前位置:首页 > 嵌入式 > 嵌入式软件
[导读]Android 3D游戏实现入门

 此示例展示了一个立方体的具体实现过程,与之前的纯Opengl es实现相比,它采用了JPCT-AE来实现,因为个人认为这个框架很方便,于是从今天开始通过其网站上的Wiki来介绍JPCT-AE的实现。通过这个示例能让你快速了解JPCT-AE的帮助文档,也就是入门。

        (1)什么是JPCT:一种封装了OPENGL es的3D游戏引擎,有j2se与android两个版本。

        (2)如何获得其jar包及帮助文档:http://download.csdn.net/user/Simdanfeg处下载


        第一个示例:同样的立方体,不同的实现

package com.threed.jpct.example;

import java.lang.reflect.Field;

import javax.microedition.khronos.egl.EGL10;
import javax.microedition.khronos.egl.EGLConfig;
import javax.microedition.khronos.egl.EGLDisplay;
import javax.microedition.khronos.opengles.GL10;

import android.app.Activity;
import android.opengl.GLSurfaceView;
import android.os.Bundle;
import android.view.MotionEvent;

import com.threed.jpct.Camera;
import com.threed.jpct.FrameBuffer;
import com.threed.jpct.Light;
import com.threed.jpct.Logger;
import com.threed.jpct.Object3D;
import com.threed.jpct.Primitives;
import com.threed.jpct.RGBColor;
import com.threed.jpct.SimpleVector;
import com.threed.jpct.Texture;
import com.threed.jpct.TextureManager;
import com.threed.jpct.World;
import com.threed.jpct.util.BitmapHelper;
import com.threed.jpct.util.MemoryHelper;

/**
 * 一个简单的例子。比起展示如何写一个正确的android应用它更着重于展示如何使用JPCT-AE这个3D游戏框架。
 * 它包含了Activity类去处理pause和resume等方法
 * 
 * @author EgonOlsen
 * 
 */
public class HelloWorld extends Activity {

    // HelloWorld对象用来处理Activity的onPause和onResume方法
    private static HelloWorld master = null;

    // GLSurfaceView对象
    private GLSurfaceView mGLView;

    // 类MyRenderer对象
    private MyRenderer renderer = null;

    // 当JPCT渲染背景时FrameBuffer类提供了一个缓冲,它的结果本质上是一个能显示或者修改甚至能进行更多后处理的图片。
    private FrameBuffer fb = null;

    // World类是JPCT时最重要的一个类,它好像胶水一样把事物"粘"起来。它包含的对象和光线定义了JPCT的场景
    private World world = null;

    // 类似java.awt.*中的Color类
    private RGBColor back = new RGBColor(50, 50, 100);

    private float touchTurn = 0;
    private float touchTurnUp = 0;

    private float xpos = -1;
    private float ypos = -1;

    // Object3D类是一个三维对象,千万不要傻呼呼的认为它与java.lang.Object类似。
    // 一个Object3D对象作为一个实例被添加到在渲染的World对象中。Object3D在World
    // 中一次添加一个实例 ,他们可能被联系起作为孩子/父母来在他们中建立一个制度.
    // 人体模型当然也能应用在以上的规则中。他们常常不加到一个World实例中,而是
    // 绑定到其它对象中(人体模型或非人体模型)。有些方法 在这个类中需要一个实例
    // 添加到一个World实例中(用World.addObject()方法可以实现)。
    private Object3D cube = null;

    // 每秒帧数
    private int fps = 0;

    // 光照类
    private Light sun = null;

    protected void onCreate(Bundle savedInstanceState) {
        // Logger类中 jPCT中一个普通的用于打印和存储消息,错误和警告的日志类。
        // 每一个JPCT生成的消息将被加入到这个类的队列中
        Logger.log("onCreate");
        // 如果本类对象不为NULL,将从Object中所有属性装入该类
        if (master != null) {
            copy(master);
        }

        super.onCreate(savedInstanceState);

        // 实例化GLSurfaceView
        mGLView = new GLSurfaceView(this);
        // 使用自己实现的 EGLConfigChooser,该实现必须在setRenderer(renderer)之前
        // 如果没有setEGLConfigChooser方法被调用,则默认情况下,视图将选择一个与当前android.view.Surface兼容至少16位深度缓冲深度EGLConfig。
        mGLView.setEGLConfigChooser(new GLSurfaceView.EGLConfigChooser() {
            public EGLConfig chooseConfig(EGL10 egl, EGLDisplay display) {
                // Ensure that we get a 16bit framebuffer. Otherwise, we''''''''ll fall
                // back to Pixelflinger on some device (read: Samsung I7500)
                int[] attributes = new int[] { EGL10.EGL_DEPTH_SIZE, 16,
                        EGL10.EGL_NONE };
                EGLConfig[] configs = new EGLConfig[1];
                int[] result = new int[1];
                egl.eglChooseConfig(display, attributes, configs, 1, result);
                return configs[0];
            }
        });
        // 实例化MyRenderer
        renderer = new MyRenderer();
        // 设置View的渲染器,同时启动线程调用渲染,以至启动渲染
        mGLView.setRenderer(renderer);
        // 设置一个明确的视图
        setContentView(mGLView);
    }

    // 重写onPause()
    @Override
    protected void onPause() {
        super.onPause();
        mGLView.onPause();
    }

    // 重写onResume()
    @Override
    protected void onResume() {
        super.onResume();
        mGLView.onResume();
    }

    // 重写onStop()
    @Override
    protected void onStop() {
        super.onStop();
    }
 


    private void copy(Object src) {
        try {
            // 打印日志
            Logger.log("Copying data from master Activity!");
            // 返回一个数组,其中包含目前这个类的的所有字段的Filed对象
            Field[] fs = src.getClass().getDeclaredFields();
            // 遍历fs数组
            for (Field f : fs) {
                // 尝试设置无障碍标志的值。标志设置为false将使访问检查,设置为true,将其禁用。
                f.setAccessible(true);
                // 将取到的值全部装入当前类中
                f.set(this, f.get(src));
            }
        } catch (Exception e) {
            // 抛出运行时异常
            throw new RuntimeException(e);
        }
    }



    public boolean onTouchEvent(MotionEvent me) {

        // 按键开始
        if (me.getAction() == MotionEvent.ACTION_DOWN) {
            // 保存按下的初始x,y位置于xpos,ypos中
            xpos = me.getX();
            ypos = me.getY();
            return true;
        }
        // 按键结束
        if (me.getAction() == MotionEvent.ACTION_UP) {
            // 设置x,y及旋转角度为初始值
            xpos = -1;
            ypos = -1;
            touchTurn = 0;
            touchTurnUp = 0;
            return true;
        }

        if (me.getAction() == MotionEvent.ACTION_MOVE) {
            // 计算x,y偏移位置及x,y轴上的旋转角度
            float xd = me.getX() - xpos;
            float yd = me.getY() - ypos;
            // Logger.log("me.getX() - xpos----------->>"
            // + (me.getX() - xpos));
            xpos = me.getX();
            ypos = me.getY();
            Logger.log("xpos------------>>" + xpos);
            // Logger.log("ypos------------>>" + ypos);
            // 以x轴为例,鼠标从左向右拉为正,从右向左拉为负
            touchTurn = xd / -100f;
            touchTurnUp = yd / -100f;
            Logger.log("touchTurn------------>>" + touchTurn);
            // Logger.log("touchTurnUp------------>>" + touchTurnUp);
            return true;
        }

        // 每Move一下休眠毫秒
        try {
            Thread.sleep(15);
        } catch (Exception e) {
            // No need for this...
        }

        return super.onTouchEvent(me);
    }

    // MyRenderer类实现GLSurfaceView.Renderer接口
    class MyRenderer implements GLSurfaceView.Renderer {
        // 当前系统的毫秒数
        private long time = System.currentTimeMillis();
        // 是否停止
        private boolean stop = false;

        // 停止
        public void stop() {
            stop = true;
        }

        // 当屏幕改变时
        public void onSurfaceChanged(GL10 gl, int w, int h) {
            // 如果FrameBuffer不为NULL,释放fb所占资源
            if (fb != null) {
                fb.dispose();
            }
            // 创建一个宽度为w,高为h的FrameBuffer
            fb = new FrameBuffer(gl, w, h);
            Logger.log(master + "");
            // 如果master为空
            if (master == null) {

                // 实例化World对象
                world = new World();

                // 设置了环境光源强度。设置此值是负的整个场景会变暗,而为正将照亮了一切。
                world.setAmbientLight(20, 20, 20);

                // 在World中创建一个新的光源
                sun = new Light(world);

                // 设置光照强度
                sun.setIntensity(250, 250, 250);

                // 创建一个纹理
                // 构造方法Texture(Bitmap image)
                // static Bitmap rescale(Bitmap bitmap, int width, int height)
                // static Bitmap convert(Drawable drawable)
                Texture texture = new Texture(BitmapHelper.rescale(
                        BitmapHelper.convert(getResources().getDrawable(
                                R.drawable.glass)), 64, 64));

                // TextureManager.getInstance()取得一个Texturemanager对象
                // addTexture("texture",texture)添加一个纹理
                TextureManager.getInstance().addTexture("texture", texture);

                // Object3D对象开始了:-)

                // Primitives提供了一些基本的三维物体,假如你为了测试而生成一些对象或为
                // 其它目的使用这些类将很明智,因为它即快速又简单,不需要载入和编辑。
                // 调用public static Object3D getCube(float scale) scale:角度
                // 返回一个立方体
                cube = Primitives.getCube(10);

                // 以纹理的方式给对象所有面"包装"上纹理
                cube.calcTextureWrapSpherical();

                // 给对象设置纹理
                cube.setTexture("texture");

                // 除非你想在事后再用PolygonManager修改,否则释放那些不再需要数据的内存
                cube.strip();

                // 初始化一些基本的对象是几乎所有进一步处理所需的过程。
                // 如果对象是"准备渲染"(装载,纹理分配,安置,渲染模式设置,
                // 动画和顶点控制器分配),那么build()必须被调用,
                cube.build();

                // 将Object3D对象添加到world集合
                world.addObject(cube);

                // 该Camera代表了Camera/viewer在当前场景的位置和方向,它也包含了当前视野的有关信息
                // 你应该记住Camera的旋转矩阵实际上是应用在World中的对象的一个旋转矩阵。
                // 这一点很重要,当选择了Camera的旋转角度,一个Camera(虚拟)围绕w旋转和通过围绕World围绕w旋转、
                // 将起到相同的效果,因此,考虑到旋转角度,World围绕camera时,camera的视角是静态的。假如你不喜欢
                // 这种习惯,你可以使用rotateCamera()方法
                Camera cam = world.getCamera();

                // 以50有速度向后移动Camera(相对于目前的方向)
                cam.moveCamera(Camera.CAMERA_MOVEOUT, 50);

                // cub.getTransformedCenter()返回对象的中心
                // cam.lookAt(SimpleVector lookAt))
                // 旋转这样camera以至于它看起来是在给定的world-space 的位置
                cam.lookAt(cube.getTransformedCenter());

                // SimpleVector是一个代表三维矢量的基础类,几乎每一个矢量都
                // 是用SimpleVector或者至少是一个SimpleVector变体构成的(有时由于
                // 某些原因比如性能可能会用(float x,float y,float z)之类)。
                SimpleVector sv = new SimpleVector();

                // 将当前SimpleVector的x,y,z值设为给定的SimpleVector(cube.getTransformedCenter())的值
                sv.set(cube.getTransformedCenter());

                // Y方向上减去100
                sv.y -= 100;

                // Z方向上减去100
                sv.z -= 100;

                // 设置光源位置
                sun.setPosition(sv);

                // 强制GC和finalization工作来试图去释放一些内存,同时将当时的内存写入日志,
                // 这样可以避免动画不连贯的情况,然而,它仅仅是减少这种情况发生的机率
                MemoryHelper.compact();

                // 如果master为空,使用日志记录且设master为HelloWorld本身
                if (master == null) {
                    Logger.log("Saving master Activity!");
                    master = HelloWorld.this;
                }
            }
        }

        // 需实现的onSurfaceCreated(GL10 gl, EGLConfig config)
        public void onSurfaceCreated(GL10 gl, EGLConfig config) {
        }

        // 绘制到当前屏幕哦:-D
        public void onDrawFrame(GL10 gl) {

            try {
                // 如果stop为true
                if (!stop) {
                    // 如果touchTurn不为0,向Y轴旋转touchTure角度
                    if (touchTurn != 0) {
                        // 旋转物体的旋转绕Y由给定矩阵W轴角(弧度顺时针方向为正值),应用到对象下一次渲染时。
                        cube.rotateY(touchTurn);
                        // 将touchTurn置0
                        touchTurn = 0;
                    }

                    if (touchTurnUp != 0) {
                        // 旋转物体的旋转围绕x由给定角度宽(弧度,逆时针为正值)轴矩阵,应用到对象下一次渲染时。
                        cube.rotateX(touchTurnUp);
                        // 将touchTureUp置0
                        touchTurnUp = 0;
                    }

                    // 用给定的颜色(back)清除FrameBuffer
                    fb.clear(back);
                    // 变换和灯光所有多边形
                    world.renderScene(fb);
                    // 绘制
                    world.draw(fb);
                    // 渲染图像显示
                    fb.display();

                    // 记录FPS
                    if (System.currentTimeMillis() - time >= 1000) {
                        // Logger.log(fps + "fps");
                        fps = 0;
                        time = System.currentTimeMillis();
                    }
                    fps++;

                    // 如果stop为false,释放FrameBuffer
                } else {
                    if (fb != null) {
                        fb.dispose();
                        fb = null;
                    }
                }
                // 当出现异常,打印异常信息
            } catch (Exception e) {
                Logger.log(e, Logger.MESSAGE);
            }
        }
    }
}

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

9月2日消息,不造车的华为或将催生出更大的独角兽公司,随着阿维塔和赛力斯的入局,华为引望愈发显得引人瞩目。

关键字: 阿维塔 塞力斯 华为

加利福尼亚州圣克拉拉县2024年8月30日 /美通社/ -- 数字化转型技术解决方案公司Trianz今天宣布,该公司与Amazon Web Services (AWS)签订了...

关键字: AWS AN BSP 数字化

伦敦2024年8月29日 /美通社/ -- 英国汽车技术公司SODA.Auto推出其旗舰产品SODA V,这是全球首款涵盖汽车工程师从创意到认证的所有需求的工具,可用于创建软件定义汽车。 SODA V工具的开发耗时1.5...

关键字: 汽车 人工智能 智能驱动 BSP

北京2024年8月28日 /美通社/ -- 越来越多用户希望企业业务能7×24不间断运行,同时企业却面临越来越多业务中断的风险,如企业系统复杂性的增加,频繁的功能更新和发布等。如何确保业务连续性,提升韧性,成...

关键字: 亚马逊 解密 控制平面 BSP

8月30日消息,据媒体报道,腾讯和网易近期正在缩减他们对日本游戏市场的投资。

关键字: 腾讯 编码器 CPU

8月28日消息,今天上午,2024中国国际大数据产业博览会开幕式在贵阳举行,华为董事、质量流程IT总裁陶景文发表了演讲。

关键字: 华为 12nm EDA 半导体

8月28日消息,在2024中国国际大数据产业博览会上,华为常务董事、华为云CEO张平安发表演讲称,数字世界的话语权最终是由生态的繁荣决定的。

关键字: 华为 12nm 手机 卫星通信

要点: 有效应对环境变化,经营业绩稳中有升 落实提质增效举措,毛利润率延续升势 战略布局成效显著,战新业务引领增长 以科技创新为引领,提升企业核心竞争力 坚持高质量发展策略,塑强核心竞争优势...

关键字: 通信 BSP 电信运营商 数字经济

北京2024年8月27日 /美通社/ -- 8月21日,由中央广播电视总台与中国电影电视技术学会联合牵头组建的NVI技术创新联盟在BIRTV2024超高清全产业链发展研讨会上宣布正式成立。 活动现场 NVI技术创新联...

关键字: VI 传输协议 音频 BSP

北京2024年8月27日 /美通社/ -- 在8月23日举办的2024年长三角生态绿色一体化发展示范区联合招商会上,软通动力信息技术(集团)股份有限公司(以下简称"软通动力")与长三角投资(上海)有限...

关键字: BSP 信息技术
关闭
关闭