Android 将多个颜色过滤器应用于同一个可绘制图形
我想应用几个颜色过滤器在一个可绘制的链。可能吗?或者创建一个过滤器,它是我想要应用的过滤器的组合 例如,我想:Android 将多个颜色过滤器应用于同一个可绘制图形,android,chaining,android-drawable,colorfilter,Android,Chaining,Android Drawable,Colorfilter,我想应用几个颜色过滤器在一个可绘制的链。可能吗?或者创建一个过滤器,它是我想要应用的过滤器的组合 例如,我想: Drawable d = ...; d.setColorFilter(0x3F000000, Mode.OVERLAY).setColorFilter(0xFF2D2D2D, Mode.SCREEN) 这就是我最后使用的方法:在画布上操作可绘制的位图,并根据需要应用尽可能多的层,使用绘制,它不仅适用于颜色过滤器,还适用于任何类型的图像混合 ... Drawable myBackgro
Drawable d = ...;
d.setColorFilter(0x3F000000, Mode.OVERLAY).setColorFilter(0xFF2D2D2D, Mode.SCREEN)
这就是我最后使用的方法:在
画布上操作可绘制的
位图
,并根据需要应用尽可能多的层,使用绘制
,它不仅适用于颜色过滤器,还适用于任何类型的图像混合
...
Drawable myBackground = createBackground(getResources().getColor(R.color.Green));
setBackgroundDrawable(myBackground);
...
private Drawable createBackground(int color) {
Canvas canvas = new Canvas();
Bitmap buttonImage = BitmapFactory.decodeResource(getResources(), R.drawable.btn_image);
Bitmap buttonShadows = BitmapFactory.decodeResource(getResources(), R.drawable.btn_shadows);
Bitmap buttonHighLights = BitmapFactory.decodeResource(getResources(), R.drawable.btn_highlights);
Bitmap result = Bitmap.createBitmap(buttonImage.getWidth(), buttonImage.getHeight(), Bitmap.Config.ARGB_8888);
canvas.setBitmap(result);
Paint paint = new Paint();
paint.setFilterBitmap(false);
// Color
paint.setColorFilter(new PorterDuffColorFilter(color, Mode.MULTIPLY));
canvas.drawBitmap(buttonImage, 0, 0, paint);
paint.setColorFilter(null);
// Shadows
paint.setXfermode(new PorterDuffXfermode(Mode.MULTIPLY));
canvas.drawBitmap(buttonShadows, 0, 0, paint);
// HighLights
paint.setXfermode(new PorterDuffXfermode(Mode.SCREEN));
canvas.drawBitmap(buttonHighLights, 0, 0, paint);
paint.setXfermode(null);
return new BitmapDrawable(getResources(), result);
}
警告:
setBackgroundDrawable(Drawable d)
已被弃用,而setBackground(Drawable d)
仅从api 16开始提供,因此如果您像我一样有一个最小目标api-14最大目标api-17,您没有“干净”的方法将可绘制设置为背景。我坚持不赞成的电话。在搜索了很多之后,我没有找到一个完全满足我需要的答案。
然而,我在2018年偶然发现了一个名为“Android中的实用图像处理”的课程,它为我指明了正确的方向:ImageFilterView
类。你可以找到那个
看到他们在同一张图像上应用多个过滤器的方式非常有启发性,而且碰巧也不那么难。
它包括使用一个ColorMatrix
并通过调用ColorMatrix.postConcat
方法来关联多个其他ColorMatrix
。最后,结果包含您应用于它的所有过滤器
下面是更改图像亮度和饱和度的示例代码
package com.rewieer.imagefilters;
import androidx.appcompat.app.AppCompatActivity;
import android.graphics.ColorMatrix;
import android.graphics.ColorMatrixColorFilter;
import android.graphics.drawable.BitmapDrawable;
import android.os.Bundle;
import android.widget.ImageView;
import android.widget.SeekBar;
public class MainActivity extends AppCompatActivity {
private ImageView image;
private SeekBar brightnessSeekBar;
private SeekBar saturationSeekBar;
private BitmapDrawable defaultDrawable;
private int brightness = 0;
private int saturation = 0;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
image = findViewById(R.id.mainImage);
defaultDrawable = (BitmapDrawable) image.getDrawable();
brightnessSeekBar = findViewById(R.id.brightnessSeekBar);
saturationSeekBar = findViewById(R.id.saturationSeekBar);
brightnessSeekBar.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener() {
@Override
public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
brightness = progress;
redraw();
}
@Override
public void onStartTrackingTouch(SeekBar seekBar) {
}
@Override
public void onStopTrackingTouch(SeekBar seekBar) {
}
});
saturationSeekBar.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener() {
@Override
public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
saturation = progress;
redraw();
}
@Override
public void onStartTrackingTouch(SeekBar seekBar) {
}
@Override
public void onStopTrackingTouch(SeekBar seekBar) {
}
});
}
public void redraw() {
ColorMatrix matrix = new ColorMatrix();
matrix.postConcat(new ColorMatrix(new float[]{
1f, 0, 0, 0, brightness,
0, 1f, 0, 0, brightness,
0, 0, 1f, 0, brightness,
0, 0, 0, 1f, 0
}));
float MS = 1.0F - saturation;
float Rt = 0.2999F * MS;
float Gt = 0.587F * MS;
float Bt = 0.114F * MS;
matrix.postConcat(new ColorMatrix(new float[]{
Rt + saturation, Gt, Bt, 0, 0,
Rt, Gt + saturation, Bt, 0, 0,
Rt, Gt, Bt + saturation, 0, 0,
0, 0, 0, 1f, 0
}));
image.setColorFilter(new ColorMatrixColorFilter(matrix));
}
}
这里最重要的方法是重画。
通过查看ImageFilterView
源代码,我们还可以通过简单的复制粘贴算法,非常轻松地对图像应用温暖和收缩