OpenGL ES 学习笔记(2)-复杂图形绘制

结合球体和圆环,学习复杂图形绘制

1. 球体绘制

1.1. 思路

在上一篇笔记中说过,OpenGL可以画点、线和三角形,为了能让球体有立体显示的效果我们采用线带(line_strip)的方式就行绘制.
要想使用OpenGL画复杂图形,最重要的就是分解图形.
首先看下图中球体:


球体

分析上面的球体, 我们可以首先对球体横向切片,这样就会得到若干个圆台(顶部特殊).


圆台

拆分圆台:


拆分圆台

当我们拆开圆台之后,在上下圆上打点并连线,形成带状:
打点连线

综上,画一个球,实际就是把球拆分成一个个带状拼接起来形成.

1.2. 求坐标

通过1.1中的球体可以看出,我们无非就是求得a、b点坐标,然后让OpenGL引擎画出来.
已知半径R,横向切片次数stack,因为需要横向切开180°,故横向切角步长为stackStep = (π/stack).
设第i次切片,r0为圆台的半径
通过上述条件求出,图中 alpha = (-π/2) + (i * stackStep);
所以, y0 = R * sin(alpha); r0 = R*cos(alpha)
接下来看圆台的俯视图:


圆台俯视图

设设第j次打点, 打点次数为slice,打点范围为360°,故打点的步长为sliceStep = (2π/since);
故图中的beta = j * sliceStep;
可轻松求得: x0和y0的坐标:
x0 = r0 * cos(beta);
z0 = r0 * sin(beta);
同样的办法求得b(x1,y1,z1)的坐标.

1.3. 核心代码

通过上面分析,很容易编写代码:

//计算球体坐标
float R = 0.5f;//球的半径
int stack = 8;//水平层数
float stackStep = ((float) (Math.PI / stack));//单位角度值 180度
int slice = 12;//竖直
float sliceStep = (float) ((Math.PI*2) / slice);//水平圆递增角度 360度

//上下两个圆的坐标 半径
float r0, r1, x0, x1, y0, y1, z0, z1;
//两个切片的夹角
float alpha0 = 0;
float alpha1 = 0;
//切片圆的角度
float beta = 0;

//顶点坐标
List<Float> coords = new ArrayList<>();
//外层循环 水平切片!
for (int i = 0; i <= stack; i++) {
    alpha0 = (float) (-Math.PI/2 + (i * stackStep));
    alpha1 = (float) (-Math.PI/2 + ((i+1) * stackStep));
    y0 = (float) (R * Math.sin(alpha0));
    r0 = (float) (R * Math.cos(alpha0));

    y1 = (float) (R * Math.sin(alpha1));
    r1 = (float) (R * Math.cos(alpha1));

    //循环每一层的圆
    for (int j = 0; j <= slice * 2; j++) {
        beta = j * sliceStep;
        x0 = (float) (r0 * Math.cos(beta));
        z0 = -(float) (r0 * Math.sin(beta));

        x1 = (float) (r1 * Math.cos(beta));
        z1 = -(float) (r1 * Math.sin(beta));
        coords.add(x0);
        coords.add(y0);
        coords.add(z0);
        coords.add(x1);
        coords.add(y1);
        coords.add(z1);
    }
}
FloatBuffer fbb = BufferUtil.list2FloatBuffer(coords);
gl.glVertexPointer(3, GL10.GL_FLOAT, 0, fbb);
gl.glDrawArrays(GL10.GL_LINE_STRIP, 0, coords.size()/3);

2. 圆环绘制

2.1. 思路

如果球体的绘制屡清楚了,圆环就更不在话下了,同样还是把圆环拆分.
如下图所示, 圆环就是一个一个圆套在一起形成圆环,我们的目标就是求得a点坐标.
为了描述清楚我把圆环中的圆单独也抽在坐标系中.


圆环
抽离坐标系

2.2. 求坐标

已知条件,内环圆半径 r, 环中圆半径r'.
通过2.1图中所示:
x0 = (r + r' + r' * cos(beta)) * cos(alpha);
y0 = (r + r' + r' * cos(beta)) * sin(alpha);
z0 = r' * sin(beta).
同样的道理再求得下一个点坐标.

2.3 核心代码


//开始画
float Rinner = 0.2f; //内环半径
float Rring = 0.3f; //环半径

int count = 20; //环 圆与圆之间的循环次数
float alphaStep = (float) ((2 * Math.PI) / count);
float alpha;

int countRing = 20;//环上圆循环次数
float betaStep = (float) ((2 * Math.PI) / countRing);
float beta;

float x0, y0, z0, x1, y1, z1;

List<Float> coordsList = new ArrayList<>();

//外层 圆与圆之间
for (int i = 0; i < count; i++) {
    alpha = alphaStep * i;

    //环上圆的点 (需要闭合)
    for (int j = 0; j <= countRing; j++) {
        beta = betaStep * j;

        x0 = (float) ((Rinner + Rring * (1 + Math.cos(beta))) * Math.cos(alpha));
        y0 = (float) ((Rinner + Rring * (1 + Math.cos(beta))) * Math.sin(alpha));
        z0 = -(float) (Rring * Math.sin(beta));

        x1 = (float) ((Rinner + Rring * (1 + Math.cos(beta))) * Math.cos(alpha + alphaStep));
        y1 = (float) ((Rinner + Rring * (1 + Math.cos(beta))) * Math.sin(alpha + alphaStep));
        z1 = -(float) (Rring * Math.sin(beta));

        coordsList.add(x0);
        coordsList.add(y0);
        coordsList.add(z0);
        coordsList.add(x1);
        coordsList.add(y1);
        coordsList.add(z1);
    }
}

gl.glVertexPointer(3, GL10.GL_FLOAT, 0, BufferUtil.list2ByteBuffer(coordsList));
gl.glDrawArrays(GL10.GL_LINE_STRIP, 0, coordsList.size()/3);

2.4 预览图

效果预览

3. 项目地址

GitHub: https://github.com/changer0/OpenGLDemo

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 223,207评论 6 521
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 95,455评论 3 400
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 170,031评论 0 366
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 60,334评论 1 300
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 69,322评论 6 398
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,895评论 1 314
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 41,300评论 3 424
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 40,264评论 0 277
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,784评论 1 321
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,870评论 3 343
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,989评论 1 354
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,649评论 5 351
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 42,331评论 3 336
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,814评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,940评论 1 275
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 49,452评论 3 379
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,995评论 2 361

推荐阅读更多精彩内容