顏色矩陣
android中能夠經過顏色矩陣(ColorMatrix類)方面的操做顏色,顏色矩陣是一個4X5 的矩陣(如圖1.1)java
能夠用來方面的修改圖片中RGBA各份量的值,顏色矩陣以一維數組的方式存儲以下:
[ a, b, c, d, e, f, g, h, i, j, k, l, m, n, o, p, q, r, s, t ]
他經過RGBA四個通道來直接操做對應顏色,若是會使用Photoshop就會知道有時處理圖片經過控制RGBA各顏色通道來作出特殊的效果。android
這個矩陣對顏色的做用計算方式如1.3示:
矩陣的運算規則是矩陣A的一行乘以矩陣C的一列做爲矩陣R的一行,編程
C矩陣是圖片中包含的ARGB信息,R矩陣是用顏色矩陣應用於C以後的新的顏色份量,運算結果以下:
R' = a*R + b*G + c*B + d*A + e;
G' = f*R + g*G + h*B + i*A + j;
B' = k*R + l*G + m*B + n*A + o;
A' = p*R + q*G + r*B + s*A + t;
顏色矩陣並非看上去那麼深奧,其實須要使用的參數不多,並且頗有規律第一行決定紅色第二行決定綠色canvas
第三行決定藍色,第四行決定了透明度,第五列是顏色的偏移量。下面是一個實際中使用的顏色矩陣。
若是把這個矩陣做用於各顏色份量的話,R=A*C,計算後會發現,各個顏色份量實際上沒有任何的改變(R'=R G'=G B'=B A'=A)。
圖1.5所示矩陣計算後會發現紅色份量增長100,綠色份量增長100,小程序
這樣的效果就是圖片偏黃,由於紅色和綠色混合後獲得黃色,黃色增長了100,圖片固然就偏黃了。
改變各顏色份量不只能夠經過修改第5列的顏色偏移量也可如上面矩陣所示將對應的顏色值乘以一個倍數,直接放大。數組
上圖1.6是將綠色份量乘以2變爲原來的2倍。相信讀者至此已經明白瞭如何經過顏色矩陣來改變各顏色份量。ide
下面編寫一段代碼來,經過調整顏色矩陣來得到不一樣的顏色效果,JavaCode以下:函數
[java] view plaincopy學習
CMatrix類: spa
public class CMatrix extends Activity {
private Button change;
private EditText [] et=new EditText[20];
private float []carray=new float[20];
private MyImage sv;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
change=(Button)findViewById(R.id.set);
sv=(MyImage)findViewById(R.id.MyImage);
for(int i=0;i<20;i++){
et[i]=(EditText)findViewById(R.id.indexa+i);
carray[i]=Float.valueOf(et[i].getText().toString());
}
change.setOnClickListener(l);
}
private Button.OnClickListener l=new Button.OnClickListener(){
@Override
public void onClick(View arg0) {
// TODO Auto-generated method stub
getValues();
sv.setValues(carray);
sv.invalidate();
}
};
public void getValues(){
for(int i=0;i<20;i++){
carray[i]=Float.valueOf(et[i].getText().toString());
}
}
}
MyImage類繼承自View類:
public class MyImage extends View {
private Paint mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
private Bitmap mBitmap;
private float [] array=new float[20];
private float mAngle;
public MyImage(Context context,AttributeSet attrs) {
super(context,attrs);
mBitmap = BitmapFactory.decodeResource(context.getResources(),
R.drawable.test);
invalidate();
}
public void setValues(float [] a){
for(int i=0;i<20;i++){
array[i]=a[i];
}
}
@Override protected void onDraw(Canvas canvas) {
Paint paint = mPaint;
paint.setColorFilter(null);
canvas.drawBitmap(mBitmap, 0, 0, paint);
ColorMatrix cm = new ColorMatrix();
//設置顏色矩陣
cm.set(array);
//顏色濾鏡,將顏色矩陣應用於圖片
paint.setColorFilter(new ColorMatrixColorFilter(cm));
//繪圖
canvas.drawBitmap(mBitmap, 0, 0, paint);
Log.i("CMatrix", "--------->onDraw");
}
}
[java] view plaincopy
CMatrix類主要負責接收顏色矩陣的設置和重繪,沒有要說的。MyImage類中進行繪圖工做,首先設置顏色矩陣cm.set(..)從一維數組中讀取數據20個數據給顏色矩陣賦值,paint.setColorFilter(..)設置顏色濾鏡,而後繪圖,效果就出來了(這個過程和PS差很少)以下:
看到這裏,相信你們對顏色矩陣的做用已經有了一個直觀的感覺,如今也能夠嘗試作一個照片特效的軟件。
可是各類效果並不能讓用戶手動調節顏色矩陣,這裏須要計算公式,因爲本人並非作圖形軟件的也不能提供,能夠參考這個連接:
http://www.adobe.com/devnet/flash/articles/matrix_transformations/ColorMatrixDemo.swf
座標變換矩陣
座標變換矩陣是一個3*3的矩陣如圖2.1,用來對圖形進行座標變化,將原來的座標點轉移到新的座標點,
由於一個圖片是有點陣和每一點上的顏色信息組成的,因此對座標的變換,就是對每一點進行搬移造成新的圖片。
具體的說圖形的放大縮小,移動,旋轉,透視,扭曲這些效果均可以用此矩陣來完成。
這個矩陣的做用是對座標x,y進行變換計算結果以下:
x'=a*x+b*y+c
y'=d*x+e*y+f
一般狀況下g=h=0,這樣使1=0*x+0*y+1恆成立。和顏色矩陣同樣,座標變換矩陣真正使用的參數不多也頗有規律。
上圖就是一個座標變換矩陣的簡單例子,計算後發現x'=x+50,y'=y+50.
可見圖片的每一點都在x和y方向上平移到了(50,50)點處,這種效果就是平移效果,將圖片轉移到了(50,50)處。
計算上面得矩陣x'=2*x,y‘=2*y.通過顏色矩陣和上面轉移效果學習,相信讀者能夠明白這個矩陣的做用了,這個矩陣對圖片進行了放大,具體的說是放大了二倍。
下面將介紹幾種經常使用的變換矩陣:
1. 旋轉
繞原點逆時針旋轉θ度角的變換公式是 x' = xcosθ − ysinθ 與 y。' = xsinθ + ycosθ
2. 縮放
變換後長寬分別放大x'=scale*x;y'=scale*y.
3. 切變
4. 反射
( , )單位向量
5. 正投影
( , )單位向量
上面的各類效果也能夠疊加在一塊兒,既矩陣的組合變換,能夠用矩陣乘法實現之,如:R=B(A*C)=(B*A)C,注意一點就是B*A和A*B通常是不等的。
下面將編一個小程序,經過控制座標變換矩陣來達到控制圖形的目的,JavaCode以下:
[java] view plaincopy
CooMatrix類:
public class CooMatrix extends Activity {
private Button change;
private EditText [] et=new EditText[9];
private float []carray=new float[9];
private MyImage sv;
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
change=(Button)findViewById(R.id.set);
sv=(MyImage)findViewById(R.id.MyImage);
for(int i=0;i<9;i++){
et[i]=(EditText)findViewById(R.id.indexa+i);
carray[i]=Float.valueOf(et[i].getText().toString());
}
change.setOnClickListener(l);
}
private Button.OnClickListener l=new Button.OnClickListener(){
@Override
public void onClick(View arg0) {
// TODO Auto-generated method stub
getValues();
sv.setValues(carray);
sv.invalidate();
}
};
public void getValues(){
for(int i=0;i<9;i++){
carray[i]=Float.valueOf(et[i].getText().toString());
}
}
}
MyImage類繼承自View類:
public class MyImage extends View {
private Paint mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
private Bitmap mBitmap;
private float [] array=new float[9];
public MyImage(Context context,AttributeSet attrs) {
super(context,attrs);
mBitmap = BitmapFactory.decodeResource(context.getResources(),
R.drawable.ic_launcher_android);
invalidate();
}
public void setValues(float [] a){
for(int i=0;i<9;i++){
array[i]=a[i];
}
}
@Override protected void onDraw(Canvas canvas) {
Paint paint = mPaint;
canvas.drawBitmap(mBitmap, 0, 0, paint);
//new 一個座標變換矩陣
Matrix cm = new Matrix();
//爲座標變換矩陣設置響應的值
cm.setValues(array);
//按照座標變換矩陣的描述繪圖
canvas.drawBitmap(mBitmap, cm, paint);
Log.i("CMatrix", "--------->onDraw");
}
上面的代碼中類CooMatrix用於接收用戶輸入的座標變換矩陣參數,類MyImage接收參數,經過setValues()設置矩陣參數,而後Canvas調用drawBitmap繪圖。效果以下:
上面給出了用座標變換矩陣作出的各類效果,用座標變換矩陣能夠方面的調節圖形的各類效果,
可是咱們看看Matrix類就能夠發現,實際上,matrix類自己已經提供了許多相似的方法,咱們只要調用,就能夠了。
setScale(float sx, float sy, float px, float py) 放大
setSkew(float kx, float ky, float px, float py) 斜切
setTranslate(float dx, float dy) 平移
setRotate(float degrees, float px, float py) 旋轉
上面的函數提供了基本的變換平移,放大,旋轉,斜切。爲了作出更復雜的變換,同時沒必要親手去改動座標變換矩陣,
Matrix類提供了許多Map方法,將原圖形映射到目標點構成新的圖形,
下面簡述setPolyToPoly(float[] src, int srcIndex, float[] dst, int dstIndex, int pointCount) 的用法,但願起到觸類旁通的做用。
參數src和dst是分別存儲了原圖像的點和和指定的目標點的一維數組,數組中存儲的座標格式以下:
[x0, y0, x1, y1, x2,y2,...]
這個個函數將src中的座標映射到dst中的座標,實現圖像的變換。
具體的例子能夠參考APIDemos裏的PolyToPoly,我在這裏就再也不貼代碼了,只講一下函數是怎麼變換圖片的。下面是效果:
圖中寫1的是原圖,寫有2,3,4的是變換後的圖形。如今分析2是怎麼變換來的,變換的原座標點和目的座標點以下:
src=new float[] { 32, 32, 64, 32 }
dst=new float[] { 32, 32, 64, 48 }
從上圖標示出的座標看出原圖的(32,32)映射到原圖的(32,32),(64,32)映射到原圖(64,48)這樣的效果是圖像放大了並且發生了旋轉。這樣的過程至關於(32,32)點不動,而後拉住圖形(64,32)點並拉到(64,48)點處,這樣圖形必然會被拉伸放大而且發生旋轉。最後用一個平移將圖形移動到右邊如今的位置。但願可以好好理解這一過程,下面的3,4圖是一樣的道理。