內(nèi)容如題顷蟆,文末有demo
核心功能使用rxffmpeg實(shí)現(xiàn)
感謝開源sdk-rxffmpeg
今日帐偎,公司有個需求,就是對視頻實(shí)現(xiàn)剪輯豁生,和加水印漫贞。先上圖:
視頻加水印效果
視頻剪輯
使用方法
VideoEditFun.getInstance().init(this)
VideoEditFun.getInstance().setOnPicCallback(object : VideoEditFunPicCallback {
override fun loadImage(path: String?, view: ImageView?) {
ImageManager.getInstance().loadNormalPic(path, view)
}
override fun loadImage(resId: Int, view: ImageView?) {
ImageManager.getInstance().loadLocalPic(resId, view)
}
})
VideoEditFun.getInstance().setOnFunCallback(object : VideoEditFunCallback {
override fun addTxt(path: String?) {
KtLogUtil.d("視頻信息addTxt:${path}")
}
override fun error(message: String?) {
KtLogUtil.d("視頻信息error:${message}")
}
override fun cropPath(path: String?) {
KtLogUtil.d("視頻信息cropPath:${path}")
}
})
val path = Environment.getExternalStorageDirectory().getPath() + "/DCIM/Camera/share_d4d8c1b6646faa1527eac44c252dd121.mp4"
// val path = Environment.getExternalStorageDirectory().getPath() + "/DCIM/Camera/VID_20220321_213020.mp4"
// val path = Environment.getExternalStorageDirectory().getPath() + "/DCIM/Camera/VID_20220322_172210.mp4"
// val path = Environment.getExternalStorageDirectory().getPath() +
// "/Screenrecorder-2022-03-03-14-43-16-450.mp4"
VideoEditFun.getInstance().cropVideo(this, path)
// VideoEditFun.getInstance().addText(this, path)
實(shí)現(xiàn)講解
視頻剪輯:
怎樣獲取剪裁位置芍殖?
首先腦海里面應(yīng)該有個思路豌骏,我視頻的長度,換算為什么東西窃躲,才能進(jìn)行等比例的剪裁,就是這個剪裁的范圍值躁倒,是怎樣算出來的樱溉?
這里纬凤,我的實(shí)現(xiàn)思路,就是把視頻剪裁的范圍挖帘,換算為屏幕的寬度恋技。
設(shè)想一下?
視頻假如10s骄崩,換算為一個1080px的寬度的分辨率薄辅。那么,就能得出一個px對應(yīng)多少s視頻長度了吧脱惰。
說到這里
應(yīng)該突然懂了吧窿春,那就直接劃到底部吧。
看不懂就繼續(xù)蔚润。
怎樣實(shí)現(xiàn)尺栖?
首先,視頻剪裁本質(zhì)上货徙,是有一個層疊的滑動選擇控件皮胡,一個是視頻長度可以滑動(如果超過范圍),再一個蠢棱,是視頻的范圍選擇控件可以滑動甩栈。
對于第一個,我們使用recyclerview實(shí)現(xiàn)即可玉转,設(shè)定每個item的固定長度究抓,通過layoutmanager獲取首尾可以item袭灯,計算出視頻可見范圍。
對于第二個橘茉,使用一個可拖動的自定義view姨丈,基于第一個可見范圍的基礎(chǔ)上,再進(jìn)行范圍的選取构挤。
最終我們的范圍取值,是基于第二個控件的唐础,第一個控件提供了視頻原始范圍矾飞,第二個控件提供了我們的剪裁范圍。
最后豹绪,有了剪裁范圍,就別扯這么多了瞒津,直接調(diào)用ffmpeg進(jìn)行剪裁。
說到這里巷蚪,不懂的話屁柏,直接滑動到底部,下載demo自己看淌喻。
視頻加水印:
首先八拱,加水印烁落,我這里選擇的方法是在視頻上,覆蓋上一個圖片灯萍。
什么?圖片旦棉?是不是明白了什么绑洛?
沒錯童本,核心就是一個控件的圖片獲取,核心代碼如下:
public static void viewSnapshot(View view, ViewSnapListener listener) {
//使控件可以進(jìn)行緩存
view.setDrawingCacheEnabled(true);
//獲取緩存的 Bitmap
Bitmap drawingCache = view.getDrawingCache();
//復(fù)制獲取的 Bitmap
drawingCache = Bitmap.createBitmap(drawingCache);
//關(guān)閉視圖的緩存
view.setDrawingCacheEnabled(false);
if (drawingCache != null) {
if (listener != null) {
listener.success(drawingCache);
}
} else {
if (listener != null) {
listener.failed();
}
}
}
說重點(diǎn)绑蔫,怎樣加個文字在這個控件上面泵额,又怎樣實(shí)現(xiàn)文字的顏色,文字的拖動篓叶。
那就自定義一個文字控件,可以改變顏色左敌,實(shí)現(xiàn)拖動嗦董,放大縮小的那種。
沒錯,bitmap是操作核心幸斥,那你懂了吧
顏色
這個不用講了吧,原生屬性
拖動
GestureDetector實(shí)現(xiàn)坐標(biāo)獲取廊勃,然后控件進(jìn)行位置
放大縮小
就是對控件的拖動操作经窖,判斷是放大還是縮小,然后進(jìn)行繪制
然后獲取到一個view的bitmap冰悠,然后合成圖片配乱,然后合成視頻。
注意桑寨,圖片寬高和視頻寬高要等比縮放忿檩,不然合成會有拉伸。
獲取視頻寬高的時候沙咏,也要注意旋轉(zhuǎn)角度的適配
說完了兽掰,是不是一臉懵?
最后附上demo
that's all----------------------------------------------------------------------------------------