一步一步学android OpenGL ES2.0编程(6 大结局)

本文介绍如何通过覆写GLSurfaceView的onTouchEvent方法,让OpenGLES应用响应触摸事件,并利用触摸移动实现对象旋转。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

响应触摸事件

使你的OpenGL ES应用能响应触摸的关键是扩展你实现的GLSurfaceView 代码,覆写onTouchEvent() 方法来监听触摸事件。

本文向你展示如何监听用户的触摸事件以使用户可以旋转某个OpenGL ES对象。

设置一个触摸监听器

为了使你的OpenGL Es应用响应触摸事件,你必须在你的GLSurfaceView 类中实现onTouchEvent()事件。下面的例子演示了如何监听MotionEvent.ACTION_MOVE 事件然后把它们转换成一个形状的旋转角度。

  1. @Override  
  2. public boolean onTouchEvent(MotionEvent e) {  
  3.     // MotionEvent带有从触摸屏幕来的输入的详细信息以及其它输入控制   
  4.     // 此处,你只需对触摸位置的改变感兴趣即可。   
  5.   
  6.     float x = e.getX();  
  7.     float y = e.getY();  
  8.   
  9.     switch (e.getAction()) {  
  10.         case MotionEvent.ACTION_MOVE:  
  11.   
  12.             float dx = x - mPreviousX;  
  13.             float dy = y - mPreviousY;  
  14.   
  15.             // reverse direction of rotation above the mid-line   
  16.             if (y > getHeight() / 2) {  
  17.               dx = dx * -1 ;  
  18.             }  
  19.   
  20.             // reverse direction of rotation to left of the mid-line   
  21.             if (x < getWidth() / 2) {  
  22.               dy = dy * -1 ;  
  23.             }  
  24.   
  25.             mRenderer.mAngle += (dx + dy) * TOUCH_SCALE_FACTOR;  // = 180.0f / 320   
  26.             requestRender();  
  27.     }  
  28.   
  29.     mPreviousX = x;  
  30.     mPreviousY = y;  
  31.     return true;  
  32. }  
@Override
public boolean onTouchEvent(MotionEvent e) {
    // MotionEvent带有从触摸屏幕来的输入的详细信息以及其它输入控制
    // 此处,你只需对触摸位置的改变感兴趣即可。

    float x = e.getX();
    float y = e.getY();

    switch (e.getAction()) {
        case MotionEvent.ACTION_MOVE:

            float dx = x - mPreviousX;
            float dy = y - mPreviousY;

            // reverse direction of rotation above the mid-line
            if (y > getHeight() / 2) {
              dx = dx * -1 ;
            }

            // reverse direction of rotation to left of the mid-line
            if (x < getWidth() / 2) {
              dy = dy * -1 ;
            }

            mRenderer.mAngle += (dx + dy) * TOUCH_SCALE_FACTOR;  // = 180.0f / 320
            requestRender();
    }

    mPreviousX = x;
    mPreviousY = y;
    return true;
}

注意在计算完旋转角度之后,本方法调用requestRender() 来告诉renderer要渲染帧了。这样做是很高效的,因为在没有发生旋转时不需要重画帧。然而,在你没有要求只在数据发生改变才重画之前,还不能达到最高效,即别忘了解除这一句的注释:

  1. public MyGLSurfaceView(Context context) {  
  2.     ...  
  3.     // Render the view only when there is a change in the drawing data   
  4.     setRenderMode(GLSurfaceView.RENDERMODE_WHEN_DIRTY);  
  5. }  
public MyGLSurfaceView(Context context) {
    ...
    // Render the view only when there is a change in the drawing data
    setRenderMode(GLSurfaceView.RENDERMODE_WHEN_DIRTY);
}

曝露出旋转角度

上面的例子要求你向其它类曝露出你的旋转角度,所以你要为你的renderer添加一个public成员。既然renderer的代码运行于主界面之外的单独线程中,你必须声明这个公开变量为volatile. 。下面的代码就是这样做的:

  1. public class MyGLRenderer implements GLSurfaceView.Renderer {  
  2.     ...  
  3.     public volatile float mAngle;  
  4. }  
public class MyGLRenderer implements GLSurfaceView.Renderer {
    ...
    public volatile float mAngle;
}

应用旋转

要应用触摸所产生的旋转, 注释掉产生角度的代码并且添加mAngle,它包活了触摸所产生的角度:

  1. public void onDrawFrame(GL10 gl) {  
  2.     ...  
  3.     // Create a rotation for the triangle   
  4.     // long time = SystemClock.uptimeMillis() % 4000L;   
  5.     // float angle = 0.090f * ((int) time);   
  6.     Matrix.setRotateM(mRotationMatrix, 0, mAngle, 00, -1.0f);  
  7.   
  8.     // 合并旋转矩阵到投影和相机视口矩阵   
  9.     Matrix.multiplyMM(mMVPMatrix, 0, mRotationMatrix, 0, mMVPMatrix, 0);  
  10.   
  11.     // 画一个角度   
  12.     mTriangle.draw(mMVPMatrix);  
  13. }  
public void onDrawFrame(GL10 gl) {
    ...
    // Create a rotation for the triangle
    // long time = SystemClock.uptimeMillis() % 4000L;
    // float angle = 0.090f * ((int) time);
    Matrix.setRotateM(mRotationMatrix, 0, mAngle, 0, 0, -1.0f);

    // 合并旋转矩阵到投影和相机视口矩阵
    Matrix.multiplyMM(mMVPMatrix, 0, mRotationMatrix, 0, mMVPMatrix, 0);

    // 画一个角度
    mTriangle.draw(mMVPMatrix);
}

当你完成了上述的几步,运行程序然后在陪同幕上拖动你的手指头,你会看到下面这样:


 Figure 1. 跟据触摸输入的转动的三角形(圈圈显示了触摸的位置)。

资源下载链接为: https://pan.quark.cn/s/22ca96b7bd39 在 IT 领域,文档格式转换是常见需求,尤其在处理多种文件类型时。本文将聚焦于利用 Java 技术栈,尤其是 Apache POI 和 iTextPDF 库,实现 doc、xls(涵盖 Excel 2003 及 Excel 2007+)以及 txt、图片等格式文件向 PDF 的转换,并实现在线浏览功能。 先从 Apache POI 说起,它是一个强大的 Java 库,专注于处理 Microsoft Office 格式文件,比如 doc 和 xls。Apache POI 提供了 HSSF 和 XSSF 两个 API,其中 HSSF 用于读写老版本的 BIFF8 格式(Excel 97-2003),XSSF 则针对新的 XML 格式(Excel 2007+)。这两个 API 均具备读取和写入工作表、单元格、公式、样式等功能。读取 Excel 文件时,可通过创建 HSSFWorkbook 或 XSSFWorkbook 对象来打开相应格式的文件,进而遍历工作簿中的每个 Sheet,获取行和列数据。写入 Excel 文件时,创建新的 Workbook 对象,添加 Sheet、Row 和 Cell,即可构建新 Excel 文件。 再看 iTextPDF,它是一个用于生成和修改 PDF 文档的 Java 库,拥有丰富的 API。创建 PDF 文档时,借助 Document 对象,可定义页面尺寸、边距等属性来定制 PDF 外观。添加内容方面,可使用 Paragraph、List、Table 等元素将文本、列表和表格加入 PDF,图片可通过 Image 类加载插入。iTextPDF 支持多种字体和样式,可设置文本颜色、大小、样式等。此外,iTextPDF 的 TextRenderer 类能将 HTML、
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值