热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android使用Opengl录像时添加水印

这篇文章主要为大家详细介绍了Android使用Opengl录像时添加水印,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

最近需要开发一个类似行车记录仪的app,其中需要给录制的视频添加动态水印。我使用的是OpenGL开发的,刚开始实现的是静态水印,后面才实现的动态水印。

先上效果图,左下角的是静态水印,中间偏下的是时间水印(动态水印):

一、静态水印

实现原理:录像时是通过OpenGL把图像渲染到GLSurfaceView上的,通俗的讲,就是把图片画到一块画布上,然后展示出来。添加图片水印,就是把水印图片跟录制的图像一起画到画布上。

这是加载纹理跟阴影的Java类

package com.audiovideo.camera.blog;


import android.opengl.GLES20;


/**
 * Created by fenghaitao on 2019/9/12.
 */

public class WaterSignSProgram{

  private static int programId;
  private static final String VERTEX_SHADER =
          "uniform mat4 uMVPMatrix;\n" +
          "attribute vec4 aPosition;\n" +
          "attribute vec4 aTextureCoord;\n" +
          "varying vec2 vTextureCoord;\n" +
          "void main() {\n" +
          "  gl_Position = uMVPMatrix * aPosition;\n" +
          "  vTextureCoord = aTextureCoord.xy;\n" +
          "}\n";

  private static final String FRAGMENT_SHADER =
          "precision mediump float;\n" +
          "varying vec2 vTextureCoord;\n" +
          "uniform sampler2D sTexture;\n" +
          "void main() {\n" +
          "  gl_FragColor = texture2D(sTexture, vTextureCoord);\n" +
          "}\n";

  public WaterSignSProgram() {
    programId = loadShader(VERTEX_SHADER, FRAGMENT_SHADER);

    uMVPMatrixLoc = GLES20.glGetUniformLocation(programId, "uMVPMatrix");
    checkLocation(uMVPMatrixLoc, "uMVPMatrix");
    aPositiOnLoc= GLES20.glGetAttribLocation(programId, "aPosition");
    checkLocation(aPositionLoc, "aPosition");
    aTextureCoordLoc = GLES20.glGetAttribLocation(programId, "aTextureCoord");
    checkLocation(aTextureCoordLoc, "aTextureCoord");
    sTextureLoc = GLES20.glGetUniformLocation(programId, "sTexture");
    checkLocation(sTextureLoc, "sTexture");
  }

  public int uMVPMatrixLoc;
  public int aPositionLoc;
  public int aTextureCoordLoc;
  public int sTextureLoc;

  public static void checkLocation(int location, String label) {
    if (location <0) {
      throw new RuntimeException("Unable to locate '" + label + "' in program");
    }
  }

/**
 * 加载编译连接阴影
 * @param vss source of vertex shader
 * @param fss source of fragment shader
 * @return
 */
public static int loadShader(final String vss, final String fss) {
  Log.v(TAG, "loadShader:");
  int vs = GLES20.glCreateShader(GLES20.GL_VERTEX_SHADER);
  GLES20.glShaderSource(vs, vss);
  GLES20.glCompileShader(vs);
  final int[] compiled = new int[1];
  GLES20.glGetShaderiv(vs, GLES20.GL_COMPILE_STATUS, compiled, 0);
  if (compiled[0] == 0) {
   Log.e(TAG, "Failed to compile vertex shader:"
      + GLES20.glGetShaderInfoLog(vs));
   GLES20.glDeleteShader(vs);
   vs = 0;
  }

  int fs = GLES20.glCreateShader(GLES20.GL_FRAGMENT_SHADER);
  GLES20.glShaderSource(fs, fss);
  GLES20.glCompileShader(fs);
  GLES20.glGetShaderiv(fs, GLES20.GL_COMPILE_STATUS, compiled, 0);
  if (compiled[0] == 0) {
   Log.w(TAG, "Failed to compile fragment shader:"
     + GLES20.glGetShaderInfoLog(fs));
   GLES20.glDeleteShader(fs);
   fs = 0;
  }

  final int program = GLES20.glCreateProgram();
  GLES20.glAttachShader(program, vs);
  GLES20.glAttachShader(program, fs);
  GLES20.glLinkProgram(program);

  return program;
}

  /**
   * terminatinng, this should be called in GL context
   */
  public static void release() {
    if (programId >= 0)
      GLES20.glDeleteProgram(programId);
    programId = -1;
  }
}
package com.audiovideo.camera.blog;

import android.opengl.GLES20;
import android.opengl.Matrix;

import com.audiovideo.camera.glutils.GLDrawer2D;
import com.audiovideo.camera.utils.LogUtil;

import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.FloatBuffer;


这是画水印的Java类
/**
 * Created by fenghaitao on 2019/9/12.
 */

public class WaterSignature {

  private static final String VERTEX_SHADER =
      "uniform mat4 uMVPMatrix;\n" +
          "attribute vec4 aPosition;\n" +
          "attribute vec4 aTextureCoord;\n" +
          "varying vec2 vTextureCoord;\n" +
          "void main() {\n" +
          "  gl_Position = uMVPMatrix * aPosition;\n" +
          "  vTextureCoord = aTextureCoord.xy;\n" +
          "}\n";

  private static final String FRAGMENT_SHADER =
      "precision mediump float;\n" +
          "varying vec2 vTextureCoord;\n" +
          "uniform sampler2D sTexture;\n" +
          "void main() {\n" +
          "  gl_FragColor = texture2D(sTexture, vTextureCoord);\n" +
          "}\n";

  public static final int SIZE_OF_FLOAT = 4;
  /**
   * 一个“完整”的正方形,从两维延伸到-1到1。
   * 当 模型/视图/投影矩阵是都为单位矩阵的时候,这将完全覆盖视口。
   * 纹理坐标相对于矩形是y反的。
   * (This seems to work out right with external textures from SurfaceTexture.)
   */
  private static final float FULL_RECTANGLE_COORDS[] = {
      -1.0f, -1.0f,  // 0 bottom left
      1.0f, -1.0f,  // 1 bottom right
      -1.0f, 1.0f,  // 2 top left
      1.0f, 1.0f,  // 3 top right
  };
  private static final float FULL_RECTANGLE_TEX_COORDS[] = {
      0.0f, 1.0f,   //0 bottom left   //0.0f, 0.0f, // 0 bottom left
      1.0f, 1.0f,   //1 bottom right  //1.0f, 0.0f, // 1 bottom right
      0.0f, 0.0f,   //2 top left    //0.0f, 1.0f, // 2 top left
      1.0f, 0.0f,   //3 top right    //1.0f, 1.0f, // 3 top right
  };

  private FloatBuffer mVertexArray;
  private FloatBuffer mTexCoordArray;
  private int mCoordsPerVertex;
  private int mCoordsPerTexture;
  private int mVertexCount;
  private int mVertexStride;
  private int mTexCoordStride;
  private int hProgram;

  public float[] mProjectiOnMatrix= new float[16];// 投影矩阵
  public float[] mViewMatrix = new float[16]; // 摄像机位置朝向9参数矩阵
  public float[] mModelMatrix = new float[16];// 模型变换矩阵
  public float[] mMVPMatrix = new float[16];// 获取具体物体的总变换矩阵
  private float[] getFinalMatrix() {
    Matrix.multiplyMM(mMVPMatrix, 0, mViewMatrix, 0, mModelMatrix, 0);
    Matrix.multiplyMM(mMVPMatrix, 0, mProjectionMatrix, 0, mMVPMatrix, 0);
    return mMVPMatrix;
  }

  public WaterSignature() {
    mVertexArray = createFloatBuffer(FULL_RECTANGLE_COORDS);
    mTexCoordArray = createFloatBuffer(FULL_RECTANGLE_TEX_COORDS);
    mCoordsPerVertex = 2;
    mCoordsPerTexture = 2;
    mVertexCount = FULL_RECTANGLE_COORDS.length / mCoordsPerVertex; // 4
    mTexCoordStride = 2 * SIZE_OF_FLOAT;
    mVertexStride = 2 * SIZE_OF_FLOAT;

    Matrix.setIdentityM(mProjectionMatrix, 0);
    Matrix.setIdentityM(mViewMatrix, 0);
    Matrix.setIdentityM(mModelMatrix, 0);
    Matrix.setIdentityM(mMVPMatrix, 0);
    hProgram = GLDrawer2D.loadShader(VERTEX_SHADER, FRAGMENT_SHADER);
    GLES20.glUseProgram(hProgram);
  }

  private FloatBuffer createFloatBuffer(float[] coords) {
    ByteBuffer bb = ByteBuffer.allocateDirect(coords.length * SIZE_OF_FLOAT);
    bb.order(ByteOrder.nativeOrder());
    FloatBuffer fb = bb.asFloatBuffer();
    fb.put(coords);
    fb.position(0);
    return fb;
  }

  private WaterSignSProgram mProgram;

  public void setShaderProgram(WaterSignSProgram mProgram) {
    this.mProgram = mProgram;
  }

  public void drawFrame(int mTextureId) {
    GLES20.glUseProgram(hProgram);
    // 设置纹理
    GLES20.glActiveTexture(GLES20.GL_TEXTURE0);
    GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, mTextureId);
    GLES20.glUniform1i(mProgram.sTextureLoc, 0);
    GlUtil.checkGlError("GL_TEXTURE_2D sTexture");
    // 设置 model / view / projection 矩阵
    GLES20.glUniformMatrix4fv(mProgram.uMVPMatrixLoc, 1, false, getFinalMatrix(), 0);
    GlUtil.checkGlError("glUniformMatrix4fv uMVPMatrixLoc");
    // 使用简单的VAO 设置顶点坐标数据
    GLES20.glEnableVertexAttribArray(mProgram.aPositionLoc);
    GLES20.glVertexAttribPointer(mProgram.aPositionLoc, mCoordsPerVertex,
        GLES20.GL_FLOAT, false, mVertexStride, mVertexArray);
    GlUtil.checkGlError("VAO aPositionLoc");
    // 使用简单的VAO 设置纹理坐标数据
    GLES20.glEnableVertexAttribArray(mProgram.aTextureCoordLoc);
    GLES20.glVertexAttribPointer(mProgram.aTextureCoordLoc, mCoordsPerTexture,
        GLES20.GL_FLOAT, false, mTexCoordStride, mTexCoordArray);
    GlUtil.checkGlError("VAO aTextureCoordLoc");
    // GL_TRIANGLE_STRIP三角形带,这就为啥只需要指出4个坐标点,就能画出两个三角形了。
    GLES20.glDrawArrays(GLES20.GL_TRIANGLE_STRIP, 0, mVertexCount);
    // Done -- 解绑~
    GLES20.glDisableVertexAttribArray(mProgram.aPositionLoc);
    GLES20.glDisableVertexAttribArray(mProgram.aTextureCoordLoc);
    GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, 0);
    GLES20.glUseProgram(0);
  }

  /**
   * terminatinng, this should be called in GL context
   */
  public void release() {
    if (hProgram >= 0)
      GLES20.glDeleteProgram(hProgram);
    hProgram = -1;
  }

  /**
   * 删除texture
   */
  public static void deleteTex(final int hTex) {
    LogUtil.v("WaterSignature", "deleteTex:");
    final int[] tex = new int[] {hTex};
    GLES20.glDeleteTextures(1, tex, 0);
  }

}

没时间了。先写到这,后面是调用,迟点再写。

下面是如何把水印绘制到画布上:

1、在SurfaceTexture的onSurfaceCreated方法中初始化并设置阴影;

@Override
   public void onSurfaceCreated(final GL10 unused, final EGLConfig config) {
     LogUtil.v(TAG, "onSurfaceCreated:");
     // This renderer required OES_EGL_image_external extension
     final String extensiOns= GLES20.glGetString(GLES20.GL_EXTENSIONS);  // API >= 8
   // 使用黄色清除界面
     GLES20.glClearColor(1.0f, 1.0f, 0.0f, 1.0f);
     //设置水印
      if (mWaterSign == null) {
        mWaterSign = new WaterSignature();
      }
      //设置阴影
     mWaterSign.setShaderProgram(new WaterSignSProgram());
    mSignTexId = loadTexture(MyApplication.getContext(), R.mipmap.watermark);
   }

这里是生成mSignTexId 的方法,把该图像与纹理id绑定并返回:

public static int loadTexture(Context context, int resourceId) {
  final int[] textureObjectIds = new int[1];
  GLES20.glGenTextures(1, textureObjectIds, 0);
  if(textureObjectIds[0] == 0){
    Log.e(TAG,"Could not generate a new OpenGL texture object!");
    return 0;
  }
  final BitmapFactory.Options optiOns= new BitmapFactory.Options();
  options.inScaled = false;  //指定需要的是原始数据,非压缩数据
  final Bitmap bitmap = BitmapFactory.decodeResource(context.getResources(), resourceId, options);
  if(bitmap == null){
    Log.e(TAG, "Resource ID "+resourceId + "could not be decode");
    GLES20.glDeleteTextures(1, textureObjectIds, 0);
    return 0;
  }
  //告诉OpenGL后面纹理调用应该是应用于哪个纹理对象
  GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, textureObjectIds[0]);
  //设置缩小的时候(GL_TEXTURE_MIN_FILTER)使用mipmap三线程过滤
  GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_MIN_FILTER, GLES20.GL_LINEAR_MIPMAP_LINEAR);
  //设置放大的时候(GL_TEXTURE_MAG_FILTER)使用双线程过滤
  GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_MAG_FILTER, GLES20.GL_LINEAR);
  //Android设备y坐标是反向的,正常图显示到设备上是水平颠倒的,解决方案就是设置纹理包装,纹理T坐标(y)设置镜面重复
  //ball读取纹理的时候 t范围坐标取正常值+1
  //GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_WRAP_T, GLES20.GL_MIRRORED_REPEAT);
  GLUtils.texImage2D(GLES20.GL_TEXTURE_2D, 0, bitmap, 0);
  bitmap.recycle();
  //快速生成mipmap贴图
  GLES20.glGenerateMipmap(GLES20.GL_TEXTURE_2D);
  //解除纹理操作的绑定
  GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, 0);
  return textureObjectIds[0];
}

2、在绘制方法onDrawFrame中绘制画面的同时把水印绘制进去;

/**
    * 绘图到glsurface
    * 我们将rendermode设置为glsurfaceview.rendermode_when_dirty,
    * 仅当调用requestrender时调用此方法(=需要更新纹理时)
    * 如果不在脏时设置rendermode,则此方法的最大调用速度为60fps。
    */
   @Override
   public void onDrawFrame(final GL10 unused) {
     GLES20.glClear(GLES20.GL_COLOR_BUFFER_BIT);
     GLES20.glEnable(GLES20.GL_BLEND);
     //开启GL的混合模式,即图像叠加
     GLES20.glBlendFunc(GLES20.GL_ONE, GLES20.GL_ONE_MINUS_SRC_ALPHA);
     /**
     *中间这里是你绘制的预览画面
     */
   //画水印(非动态)
    GLES20.glViewport(20, 20, 288, 120);
   mWaterSign.drawFrame(mSignTexId);
    }

这里最重要的是要开启GL的混合模式,即图像叠加,不然你绘制的水印会覆盖原先的预览画面

//开启GL的混合模式,即图像叠加
GLES20.glBlendFunc(GLES20.GL_ONE, GLES20.GL_ONE_MINUS_SRC_ALPHA);

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 本文详细介绍了如何在 Android 开发中高效地管理和使用资源,包括本地资源和系统资源的访问方法。通过实例和代码片段,帮助开发者更好地理解和应用资源管理的最佳实践。 ... [详细]
  • Python 内存管理机制详解
    本文深入探讨了Python的内存管理机制,涵盖了垃圾回收、引用计数和内存池机制。通过具体示例和专业解释,帮助读者理解Python如何高效地管理和释放内存资源。 ... [详细]
  • 解决TensorFlow CPU版本安装中的依赖问题
    本文记录了在安装CPU版本的TensorFlow过程中遇到的依赖问题及解决方案,特别是numpy版本不匹配和动态链接库(DLL)错误。通过详细的步骤说明和专业建议,帮助读者顺利安装并使用TensorFlow。 ... [详细]
  • 探索新一代API文档工具,告别Swagger的繁琐
    对于后端开发者而言,编写和维护API文档既繁琐又不可或缺。本文将介绍一款全新的API文档工具,帮助团队更高效地协作,简化API文档生成流程。 ... [详细]
  • 本文详细探讨了Android Activity中View的绘制流程和动画机制,包括Activity的生命周期、View的测量、布局和绘制过程以及动画对View的影响。通过实验验证,澄清了一些常见的误解,并提供了代码示例和执行结果。 ... [详细]
  • 本文探讨了在构建应用程序时,如何对不同类型的数据进行结构化设计。主要分为三类:全局配置、用户个人设置和用户关系链。每种类型的数据都有其独特的用途和应用场景,合理规划这些数据结构有助于提升用户体验和系统的可维护性。 ... [详细]
  • 在 Android 开发中,通过 Intent 启动 Activity 或 Service 时,可以使用 putExtra 方法传递数据。接收方可以通过 getIntent().getExtras() 获取这些数据。本文将介绍如何使用 RoboGuice 框架简化这一过程,特别是 @InjectExtra 注解的使用。 ... [详细]
  • Linux中的yum安装软件
    yum俗称大黄狗作用:解决安装软件包的依赖关系当安装依赖关系的软件包时,会将依赖的软件包一起安装。本地yum:需要yum源,光驱挂载。yum源:(刚开始查看yum源中的内容就是上图 ... [详细]
  • 鼠标悬停出现提示信息怎么做
    概述–提示:指启示,提起注意或给予提醒和解释。在excel中会经常用到给某个格子增加提醒信息,比如金额提示输入数值或最大长度值等等。设置方式也有多种,简单的,仅为单元格插入批注就可 ... [详细]
  • 本文将详细介绍多个流行的 Android 视频处理开源框架,包括 ijkplayer、FFmpeg、Vitamio、ExoPlayer 等。每个框架都有其独特的优势和应用场景,帮助开发者更高效地进行视频处理和播放。 ... [详细]
  • 气象对比分析
    本文探讨了不同地区和时间段的天气模式,通过详细的图表和数据分析,揭示了气候变化的趋势及其对环境和社会的影响。 ... [详细]
  • 本文探讨了如何利用NFC技术,将存储在Android手机中的患者信息安全高效地传输到台式计算机。重点介绍了适用于医院场景的NFC USB读卡器(如ACR122U)的应用方法。 ... [详细]
  • 探讨 HDU 1536 题目,即 S-Nim 游戏的博弈策略。通过 SG 函数分析游戏胜负的关键,并介绍如何编程实现解决方案。 ... [详细]
  • 本文回顾了2017年的转型和2018年的收获,分享了几家知名互联网公司提供的工作机会及面试体验。 ... [详细]
  • 深入解析动态代理模式:23种设计模式之三
    在设计模式中,动态代理模式是应用最为广泛的一种代理模式。它允许我们在运行时动态创建代理对象,并在调用方法时进行增强处理。本文将详细介绍动态代理的实现机制及其应用场景。 ... [详细]
author-avatar
mobiledu2502891023
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有