android:照片涂畫功能實現(xiàn)過程及原理詳解
這個功能可以幫你實現(xiàn),在圖片上進行隨意的涂抹,可以用于SNS產(chǎn)品。
繪圖本身很簡單,但是要實現(xiàn)在圖片上指定的部分精確(位置,縮放)的繪圖,就有點麻煩了。
下面講講實現(xiàn)過程及原理:
UI構圖
這個UI,看似簡單,還是有點復雜的,下面需要一個底圖,上面再來一個透明的圖片控件,畫圖的時候要同步兩個控件的變形。
UI層次簡圖
為什么,需要上面一個Canvas Image與Back保持一致?因為,Canvas Image會被畫到Canvas上,它是Canvas的宿主,即ImageView被變成了一塊畫布,原來ImageView上的內容會被擦除。如果只有Back Image那么一旦開啟畫布,你什么也看不到。
神奇的Matrix
ImageView控件是常用的Android控件之一,主要用于圖片展示?;旧纤械母呒壵Z言中,都有類似控件。但是,有一樣東西讓他化腐朽為神奇,那就是Matrix。有了Matrix我們就可以實現(xiàn)圖片平移、放大、旋轉、扭曲等常用的特效。Matrix本身是一個9*9的矩陣,里面存放的是平移坐標、放大系數(shù)、sin/cos角度值。我們可以通過getMatrix()來獲取一個IV的矩陣,或者通過setMatrix()來設置它的值。
上面的東西拿來有什么用?試想一下,當我們打開相冊,查看一張照片的時候,可以通過觸摸,平移或者放大圖片。我們,要在上面繪圖,先把Canvas Image 變?yōu)镃anvas,但是,Canvas Image沒有經(jīng)過變化。必須,至少確保兩個Img控件擁有相同的變形,否則無法對齊畫的坐標點。這個時候,要么當Back Image變的時候,Canvas Image立即同步操作,要么,最畫到canvas上的時候,同步變形。前一種方案是沒有必要的,果斷使用后一種。這個時候你就要問,怎么得到IV的變形參數(shù)?IV提供了一些方法來單獨的獲取和設置某種變形,當時折騰了很久,不但繁瑣,而且達不到效果。這時候,上面的Matrix就派上用場了。當時,可沒人這么愉快的告訴我這個。
坐標映射
上面完成了圖形的變換,現(xiàn)在終于可以再上面作畫了。但是一畫,你就會發(fā)現(xiàn)一個問題,畫不到指定位置上。這是什么問題呢?坐標系偏移。(0,0)點默認為屏幕的左上角。但是,想一下當我們的圖片不滿一個屏幕,很小的時候,Canvas的坐標系在什么位置?我在屏幕(0,0)坐標畫一個點,canvas上就會出現(xiàn)一個點,即使兩者的位置相差很遠。
這個時候,我們需要將兩個坐標系進行映射,通過偏移對齊坐標系。偏移多少?這時候該使用矩陣的translate值了。這樣我們就可以知道圖片在坐標系上的偏移了,隨邊你怎么移動坐標都能準確對齊。
private HashMap getImageViewIneerSize(ImageView iv){
HashMap size=new HashMap();
//獲得ImageView中Image的變換矩陣
Matrix m = iv.getImageMatrix();
float[] values = new float[10];
m.getValues(values);
//Image在繪制過程中的變換矩陣,從中獲得x和y方向的縮放系數(shù)
float sx = values[0];
float sy = values[4];
//計算Image在屏幕上實際繪制的寬高
size.put("scaleX", 1/sx);
size.put("scaleY", 1/sy);
size.put("offsetX", values[2]); //X軸的translate的值
size.put("offsetY",values[5]);
return size;
}
其中width=backImage.getDrawable().getBounds().width();
你會發(fā)現(xiàn)有個scaleX,這是干什么的?我們假設現(xiàn)在圖片經(jīng)過縮放后的寬度恰好等于屏幕寬度,圖片的實際寬度是960px。但是我們在X=480px的地方畫一個點,這個點應該顯示在圖片的什么地方呢?我們的意圖是要在圖片的最后面,即X=960px的地方畫一個點,但是現(xiàn)在卻跑到了480處,明顯不滿足要求。這時,就需要乘上上面的scaleX了。
畫線的最終代碼:
/*根據(jù)兩點坐標,繪制連線
*startX、stopX 為觸摸事件開始、結束的地方
*offsetX,為圖片在X軸的位移值
*scaleX,為圖片在X軸的縮放值的倒數(shù)
*/
if((startY-offsetY)>=0&&(stopY-offsetY)>=0)
canvas.drawLine((startX-offsetX)*scaleX, (startY-offsetY)*scaleY, (stopX-offsetX)*scaleX, (stopY-offsetY)*scaleY, pen);
【注】
ImageView的實際大小等于屏幕的大小,Canvas的實際大小由圖片實際大小決定。
ImageView的寬高很容易取得,但是它里面的圖片是變過形的,怎么獲取它的當前大小呢?用(原始大小*縮放系數(shù))。
合并
最后一步就是將兩個圖層合并為一張圖片。參考代碼如下:
/**
* 合并兩張bitmap為一張
* @param background
* @param foreground
* @return Bitmap
*/
public static Bitmap combineBitmap(Bitmap background, Bitmap foreground) {
if (background == null) {
return null;
}
int bgWidth = background.getWidth();
int bgHeight = background.getHeight();
int fgWidth = foreground.getWidth();
int fgHeight = foreground.getHeight();
Bitmap newmap = Bitmap.createBitmap(bgWidth, bgHeight, Config.ARGB_8888);
Canvas canvas = new Canvas(newmap);
canvas.drawBitmap(background, 0, 0, null);
canvas.drawBitmap(foreground, (bgWidth - fgWidth) / 2,
(bgHeight - fgHeight) / 2, null);
canvas.save(Canvas.ALL_SAVE_FLAG);
canvas.restore();
return newmap;
} //end of combineBitmap
通過Canvas來合并和改變Bitmap的大小,由于兩個圖層的大小、位置完全一致,故坐標對齊(0,0)點就可以了。
如果,沒有前面的工作,你是很難精確的進行圖片合并的。
- Android照片墻應用實現(xiàn) 再多的圖片也不怕崩潰
- Android瀑布流照片墻實現(xiàn) 體驗不規(guī)則排列的美感
- Android開發(fā)仿掃一掃實現(xiàn)拍攝框內的照片功能
- Android自定義View實現(xiàn)照片裁剪框與照片裁剪功能
- 完美解決Android三星手機從圖庫選擇照片旋轉問題
- Android應用中拍照后獲取照片路徑并上傳的實例分享
- Android仿微信照片選擇器實現(xiàn)預覽查看圖片
- android獲取照片的快照 思路及實現(xiàn)方法
- Android調用相機并將照片存儲到sd卡上實現(xiàn)方法
- Android 調用系統(tǒng)相機拍攝獲取照片的兩種方法實現(xiàn)實例
相關文章
Android中利用zxing實現(xiàn)自己的二維碼掃描識別詳解
這篇文章主要給大家介紹了關于Android中利用zxing實現(xiàn)自己的二維碼掃描識別的相關資料,文中通過圖文介紹的非常詳細,對大家學習或者使用zxing具有一定的參考學習價值,需要的朋友們下面來一起看看吧。2017-09-09使用newInstance()來實例化fragment并傳遞數(shù)據(jù)操作
這篇文章主要介紹了使用newInstance()來實例化fragment并傳遞數(shù)據(jù)操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-08-08Android不使用自定義布局情況下實現(xiàn)自定義通知欄圖標的方法
這篇文章主要介紹了Android不使用自定義布局情況下實現(xiàn)自定義通知欄圖標的方法,實例分析了Android通知欄圖標的創(chuàng)建技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-12-12