两种基于opengl 画球面体的方式

114 篇文章 1 订阅
25 篇文章 0 订阅

1 以球体中心(0,0,0)去连接各个三角形,然后通过GLES20.glDrawArrays(GLES20.GL_TRIANGLE_FAN,0,vSize);进行

public class Ball1 extends Shape {

    private float step=2f;
    private FloatBuffer vertexBuffer;
    private int vSize;

    private int mProgram;
    private float[] mViewMatrix=new float[16];
    private float[] mProjectMatrix=new float[16];
    private float[] mMVPMatrix=new float[16];

    public Ball(View mView) {
        super(mView);
        float[] dataPos=createBallPos();
        ByteBuffer buffer=ByteBuffer.allocateDirect(dataPos.length*4);
        buffer.order(ByteOrder.nativeOrder());
        vertexBuffer=buffer.asFloatBuffer();
        vertexBuffer.put(dataPos);
        vertexBuffer.position(0);
        vSize=dataPos.length/3;
    }
    
    private float[] createBallPos(){
        //球以(0,0,0)为中心,以R为半径,则球上任意一点的坐标为
        // ( R * cos(a) * sin(b),y0 = R * sin(a),R * cos(a) * cos(b))
        // 其中,a为圆心到点的线段与xz平面的夹角,b为圆心到点的线段在xz平面的投影与z轴的夹角
        ArrayList<Float> data=new ArrayList<>();
        float r1,r2;
        float h1,h2;
        float sin,cos;
        for(float i=-90;i<90+step;i+=step){
            r1 = (float)Math.cos(i * Math.PI / 180.0);
            r2 = (float)Math.cos((i + step) * Math.PI / 180.0);
            h1 = (float)Math.sin(i * Math.PI / 180.0);
            h2 = (float)Math.sin((i + step) * Math.PI / 180.0);
            // 固定纬度, 360 度旋转遍历一条纬线
            float step2=step*2;
            for (float j = 0.0f; j <360.0f+step; j +=step2 ) {
                cos = (float) Math.cos(j * Math.PI / 180.0);
                sin = -(float) Math.sin(j * Math.PI / 180.0);

                data.add(r2 * cos);
                data.add(h2);
                data.add(r2 * sin);
                data.add(r1 * cos);
                data.add(h1);
                data.add(r1 * sin);
            }
        }
        float[] f=new float[data.size()];
        for(int i=0;i<f.length;i++){
            f[i]=data.get(i);
        }
        return f;
    }

    @Override
    public void onSurfaceCreated(GL10 gl, EGLConfig config) {
        GLES20.glEnable(GLES20.GL_DEPTH_TEST);
        mProgram= ShaderUtils.createProgram(mView.getResources(),"vshader/Ball.sh","fshader/Cone.sh");
    }

    @Override
    public void onSurfaceChanged(GL10 gl, int width, int height) {
        //计算宽高比
        float ratio=(float)width/height;
        //设置透视投影
        Matrix.frustumM(mProjectMatrix, 0, -ratio, ratio, -1, 1, 3, 20);//offset,left,right,bottom,top
        //设置相机位置
        Matrix.setLookAtM(mViewMatrix, 0, 1.0f, -10.0f, -4.0f, 0f, 0f, 0f, 0f, 1.0f, 0.0f);//ex,ey,ez,cx,cy,cz,upx,upy,upz
        //计算变换矩阵
        Matrix.multiplyMM(mMVPMatrix,0,mProjectMatrix,0,mViewMatrix,0);
    }

    @Override
    public void onDrawFrame(GL10 gl) {
        GLES20.glUseProgram(mProgram);
        Log.e("wuwang","mProgram:"+mProgram);
        int mMatrix=GLES20.glGetUniformLocation(mProgram,"vMatrix");
        GLES20.glUniformMatrix4fv(mMatrix,1,false,mMVPMatrix,0);
        int mPositionHandle=GLES20.glGetAttribLocation(mProgram,"vPosition");
        Log.e("wuwang","Get Position:"+mPositionHandle);
        GLES20.glEnableVertexAttribArray(mPositionHandle);
        GLES20.glVertexAttribPointer(mPositionHandle,3,GLES20.GL_FLOAT,false,0,vertexBuffer);
        GLES20.glDrawArrays(GLES20.GL_TRIANGLE_FAN,0,vSize);
        GLES20.glDisableVertexAttribArray(mPositionHandle);
    }
}



2 设置球体半径及有把球面分为多少行和列,然后通过
GLES20.glDrawElements(GLES20.GL_TRIANGLE_STRIP, itmp/*indexbuf*/, GLES20.GL_UNSIGNED_SHORT, 0);
方式进行


public OpenGlSphere(float radius, int rows, int columns) {
    float[] vtmp = new float[rows * columns * 3];
    // Generate position grid.
    for (int i = 0; i < rows; i++) {
        for (int j = 0; j < columns; j++) {
            float theta = i * (float) Math.PI / (rows - 1);
            float phi = j * 2 * (float) Math.PI / (columns - 1);
            float x = (float) (radius * Math.sin(theta) * Math.cos(phi));
            float y = (float) (radius * Math.cos(theta));
            float z = (float) -(radius * Math.sin(theta) * Math.sin(phi));
            int index = i * columns + j;
            vtmp[3 * index] = x;
            vtmp[3 * index + 1] = y;
            vtmp[3 * index + 2] = z;
        }
    }

    // Create texture grid.
    float[] ttmp = new float[rows * columns * 2];
    for (int i = 0; i < rows; i++) {
        for (int j = 0; j < columns; j++) {
            int index = i * columns + j;
            ttmp[index * 2] = (float) j / (columns - 1);
            ttmp[index * 2 + 1] = (float) i / (rows - 1);
        }
    }

    // Create indices.
    int numIndices = 2 * (rows - 1) * columns;
    short[] itmp = new short[numIndices];
    short index = 0;
    for (int i = 0; i < rows - 1; i++) {
        if ((i & 1) == 0) {
            for (int j = 0; j < columns; j++) {
                itmp[index++] = (short) (i * columns + j);
                itmp[index++] = (short) ((i + 1) * columns + j);
            }
        } else {
            for (int j = columns - 1; j >= 0; j--) {
                itmp[index++] = (short) ((i + 1) * columns + j);
                itmp[index++] = (short) (i * columns + j);
            }
        }
    }

    mMesh = new OpenGlMesh(vtmp, 3, ttmp, 2, itmp);
}

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值