服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Android - Android使用Opengl录像时添加水印

Android使用Opengl录像时添加水印

2022-12-07 14:09feng海涛 Android

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

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

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

Android使用Opengl录像时添加水印

一、静态水印

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

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

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
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;
  }
}
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
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方法中初始化并设置阴影;

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@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绑定并返回:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
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中绘制画面的同时把水印绘制进去;

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
/**
    * 绘图到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的混合模式,即图像叠加,不然你绘制的水印会覆盖原先的预览画面

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

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

原文链接:https://blog.csdn.net/weixin_42574892/article/details/100765600

延伸 · 阅读

精彩推荐