电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

Android使用Opengl錄像時添加水印

瀏覽:107日期:2022-09-25 08:13:21

最近需要開發一個類似行車記錄儀的app,其中需要給錄制的視頻添加動態水印。我使用的是OpenGL開發的,剛開始實現的是靜態水印,后面才實現的動態水印。

先上效果圖,左下角的是靜態水印,中間偏下的是時間水印(動態水印):

Android使用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
相關文章:
主站蜘蛛池模板: 精益专家 - 设备管理软件|HSE管理系统|设备管理系统|EHS安全管理系统 | 棉柔巾代加工_洗脸巾oem_一次性毛巾_浴巾生产厂家-杭州禾壹卫品科技有限公司 | 河南道路标志牌_交通路标牌_交通标志牌厂家-郑州路畅交通 | 安徽合肥格力空调专卖店_格力中央空调_格力空调总经销公司代理-皖格制冷设备 | 打包钢带,铁皮打包带,烤蓝打包带-高密市金和金属制品厂 | 机器视觉检测系统-视觉检测系统-机器视觉系统-ccd检测系统-视觉控制器-视控一体机 -海克易邦 | 污水处理设备维修_污水处理工程改造_机械格栅_过滤设备_气浮设备_刮吸泥机_污泥浓缩罐_污水处理设备_污水处理工程-北京龙泉新禹科技有限公司 | 铣刨料沥青破碎机-沥青再生料设备-RAP热再生混合料破碎筛分设备 -江苏锡宝重工 | 混合反应量热仪-高温高压量热仪-微机差热分析仪DTA|凯璞百科 | 国产频谱分析仪-国产网络分析仪-上海坚融实业有限公司 | 黄石妇科医院_黄石东方女子医院_黄石东方妇产医院怎么样 | 无负压供水设备,消防稳压供水设备-淄博创辉供水设备有限公司 | 爱佩恒温恒湿测试箱|高低温实验箱|高低温冲击试验箱|冷热冲击试验箱-您身边的模拟环境试验设备技术专家-合作热线:400-6727-800-广东爱佩试验设备有限公司 | YT保温材料_YT无机保温砂浆_外墙保温材料_南阳银通节能建材高新技术开发有限公司 | 全自动端子机|刺破式端子压接机|全自动双头沾锡机|全自动插胶壳端子机-东莞市傅氏兄弟机械设备有限公司 | 双相钢_双相不锈钢_双相钢圆钢棒_双相不锈钢报价「海新双相钢」 双能x射线骨密度检测仪_dxa骨密度仪_双能x线骨密度仪_品牌厂家【品源医疗】 | VOC检测仪-甲醛检测仪-气体报警器-气体检测仪厂家-深恒安科技有限公司 | 塑胶跑道_学校塑胶跑道_塑胶球场_运动场材料厂家_中国塑胶跑道十大生产厂家_混合型塑胶跑道_透气型塑胶跑道-广东绿晨体育设施有限公司 | 发电机价格|发电机组价格|柴油发电机价格|柴油发电机组价格网 | 电脑刺绣_绣花厂家_绣花章仔_织唛厂家-[源欣刺绣]潮牌刺绣打版定制绣花加工厂家 | PSI渗透压仪,TPS酸度计,美国CHAI PCR仪,渗透压仪厂家_价格,微生物快速检测仪-华泰和合(北京)商贸有限公司 | 防堵吹扫装置-防堵风压测量装置-电动操作显示器-兴洲仪器 | 千斤顶,液压千斤顶-力良企业,专业的液压千斤顶制造商,shliliang.com | PC构件-PC预制构件-构件设计-建筑预制构件-PC构件厂-锦萧新材料科技(浙江)股份有限公司 | 二手光谱仪维修-德国OBLF光谱仪|进口斯派克光谱仪-热电ARL光谱仪-意大利GNR光谱仪-永晖检测 | [品牌官网]贵州遵义双宁口腔连锁_贵州遵义牙科医院哪家好_种植牙_牙齿矫正_原华美口腔 | 专业广州网站建设,微信小程序开发,一物一码和NFC应用开发、物联网、外贸商城、定制系统和APP开发【致茂网络】 | 【北京写字楼出租_写字楼租赁_办公室出租网/出售】-远行地产官网 | 10吨无线拉力计-2吨拉力计价格-上海佳宜电子科技有限公司 | 烟台条码打印机_烟台条码扫描器_烟台碳带_烟台数据采集终端_烟台斑马打印机-金鹏电子-金鹏电子 | 潍坊大集网-潍坊信息港-潍坊信息网 | 跨境物流_美国卡派_中大件运输_尾程派送_海外仓一件代发 - 广州环至美供应链平台 | 楼承板-开闭口楼承板-无锡海逵楼承板 | 单螺旋速冻机-双螺旋-流态化-隧道式-食品速冻机厂家-广州冰泉制冷 | 电机保护器-电动机综合保护器-浙江开民 | 环比机械 | 艾默生变频器,艾默生ct,变频器,ct驱动器,广州艾默生变频器,供水专用变频器,风机变频器,电梯变频器,艾默生变频器代理-广州市盟雄贸易有限公司官方网站-艾默生变频器应用解决方案服务商 | 齿轮减速马达一体式_蜗轮蜗杆减速机配电机-德国BOSERL齿轮减速电动机生产厂家 | 二氧化碳/活性炭投加系统,次氯酸钠发生器,紫外线消毒设备|广州新奥 | 武汉高低温试验机-现货恒温恒湿试验箱-高低温湿热交变箱价格-湖北高天试验设备 | bng防爆挠性连接管-定做金属防爆挠性管-依客思防爆科技 |