android图像处理系列之三-- 图片色调饱和度、色相、亮度处理
原图:
处理后:
下面贴代码:
一、图片处理层:
- package com.jacp.tone.view;
- import java.util.ArrayList;
- import android.content.Context;
- import android.graphics.Bitmap;
- import android.graphics.Canvas;
- import android.graphics.ColorMatrix;
- import android.graphics.ColorMatrixColorFilter;
- import android.graphics.Paint;
- import android.view.Gravity;
- import android.view.View;
- import android.widget.LinearLayout;
- import android.widget.SeekBar;
- import android.widget.SeekBar.OnSeekBarChangeListener;
- import android.widget.TextView;
- import com.jacp.tone.R;
- /**
- * 图片调色处理
- * @author maylian7700@126.com
- *
- */
- public class ToneLayer {
- /**
- * 饱和度标识
- */
- public static final int FLAG_SATURATION = 0x0;
- /**
- * 亮度标识
- */
- public static final int FLAG_LUM = 0x1;
- /**
- * 色相标识
- */
- public static final int FLAG_HUE = 0x2;
- /**
- * 饱和度
- */
- private TextView mSaturation;
- private SeekBar mSaturationBar;
- /**
- * 色相
- */
- private TextView mHue;
- private SeekBar mHueBar;
- /**
- * 亮度
- */
- private TextView mLum;
- private SeekBar mLumBar;
- private float mDensity;
- private static final int TEXT_WIDTH = 50;
- private LinearLayout mParent;
- private ColorMatrix mLightnessMatrix;
- private ColorMatrix mSaturationMatrix;
- private ColorMatrix mHueMatrix;
- private ColorMatrix mAllMatrix;
- /**
- * 亮度
- */
- private float mLumValue = 1F;
- /**
- * 饱和度
- */
- private float mSaturationValue = 0F;
- /**
- * 色相
- */
- private float mHueValue = 0F;
- /**
- * SeekBar的中间值
- */
- private static final int MIDDLE_VALUE = 127;
- /**
- * SeekBar的最大值
- */
- private static final int MAX_VALUE = 255;
- private ArrayList<SeekBar> mSeekBars = new ArrayList<SeekBar>();
- public ToneLayer(Context context) {
- init(context);
- }
- private void init(Context context) {
- mDensity = context.getResources().getDisplayMetrics().density;
- mSaturation = new TextView(context);
- mSaturation.setText(R.string.saturation);
- mHue = new TextView(context);
- mHue.setText(R.string.contrast);
- mLum = new TextView(context);
- mLum.setText(R.string.lightness);
- mSaturationBar = new SeekBar(context);
- mHueBar = new SeekBar(context);
- mLumBar = new SeekBar(context);
- mSeekBars.add(mSaturationBar);
- mSeekBars.add(mHueBar);
- mSeekBars.add(mLumBar);
- for (int i = 0, size = mSeekBars.size(); i < size; i++) {
- SeekBar seekBar = mSeekBars.get(i);
- seekBar.setMax(MAX_VALUE);
- seekBar.setProgress(MIDDLE_VALUE);
- seekBar.setTag(i);
- }
- LinearLayout saturation = new LinearLayout(context);
- saturation.setOrientation(LinearLayout.HORIZONTAL);
- saturation.setLayoutParams(new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.WRAP_CONTENT));
- LinearLayout.LayoutParams txtLayoutparams = new LinearLayout.LayoutParams((int) (TEXT_WIDTH * mDensity), LinearLayout.LayoutParams.MATCH_PARENT);
- mSaturation.setGravity(Gravity.CENTER);
- saturation.addView(mSaturation, txtLayoutparams);
- LinearLayout.LayoutParams seekLayoutparams = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.WRAP_CONTENT);
- saturation.addView(mSaturationBar, seekLayoutparams);
- LinearLayout hue = new LinearLayout(context);
- hue.setOrientation(LinearLayout.HORIZONTAL);
- hue.setLayoutParams(new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.WRAP_CONTENT));
- mHue.setGravity(Gravity.CENTER);
- hue.addView(mHue, txtLayoutparams);
- hue.addView(mHueBar, seekLayoutparams);
- LinearLayout lum = new LinearLayout(context);
- lum.setOrientation(LinearLayout.HORIZONTAL);
- lum.setLayoutParams(new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.WRAP_CONTENT));
- mLum.setGravity(Gravity.CENTER);
- lum.addView(mLum, txtLayoutparams);
- lum.addView(mLumBar, seekLayoutparams);
- mParent = new LinearLayout(context);
- mParent.setOrientation(LinearLayout.VERTICAL);
- mParent.setLayoutParams(new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.WRAP_CONTENT));
- mParent.addView(saturation);
- mParent.addView(hue);
- mParent.addView(lum);
- }
- public View getParentView() {
- return mParent;
- }
- /**
- * 设置饱和度值
- * @param saturation
- */
- public void setSaturation(int saturation) {
- mSaturationValue = saturation * 1.0F / MIDDLE_VALUE;
- }
- /**
- * 设置色相值
- * @param hue
- */
- public void setHue(int hue) {
- mHueValue = hue * 1.0F / MIDDLE_VALUE;
- }
- /**
- * 设置亮度值
- * @param lum
- */
- public void setLum(int lum) {
- mLumValue = (lum - MIDDLE_VALUE) * 1.0F / MIDDLE_VALUE * 180;
- }
- public ArrayList<SeekBar> getSeekBars()
- {
- return mSeekBars;
- }
- /**
- *
- * @param flag
- * 比特位0 表示是否改变色相,比位1表示是否改变饱和度,比特位2表示是否改变明亮度
- */
- public Bitmap handleImage(Bitmap bm, int flag) {
- Bitmap bmp = Bitmap.createBitmap(bm.getWidth(), bm.getHeight(),
- Bitmap.Config.ARGB_8888);
- // 创建一个相同尺寸的可变的位图区,用于绘制调色后的图片
- Canvas canvas = new Canvas(bmp); // 得到画笔对象
- Paint paint = new Paint(); // 新建paint
- paint.setAntiAlias(true); // 设置抗锯齿,也即是边缘做平滑处理
- if (null == mAllMatrix) {
- mAllMatrix = new ColorMatrix();
- }
- if (null == mLightnessMatrix) {
- mLightnessMatrix = new ColorMatrix(); // 用于颜色变换的矩阵,android位图颜色变化处理主要是靠该对象完成
- }
- if (null == mSaturationMatrix) {
- mSaturationMatrix = new ColorMatrix();
- }
- if (null == mHueMatrix) {
- mHueMatrix = new ColorMatrix();
- }
- switch (flag) {
- case FLAG_HUE: // 需要改变色相
- mHueMatrix.reset();
- mHueMatrix.setScale(mHueValue, mHueValue, mHueValue, 1); // 红、绿、蓝三分量按相同的比例,最后一个参数1表示透明度不做变化,此函数详细说明参考
- // // android
- // doc
- break;
- case FLAG_SATURATION: // 需要改变饱和度
- // saturation 饱和度值,最小可设为0,此时对应的是灰度图(也就是俗话的“黑白图”),
- // 为1表示饱和度不变,设置大于1,就显示过饱和
- mSaturationMatrix.reset();
- mSaturationMatrix.setSaturation(mSaturationValue);
- break;
- case FLAG_LUM: // 亮度
- // hueColor就是色轮旋转的角度,正值表示顺时针旋转,负值表示逆时针旋转
- mLightnessMatrix.reset(); // 设为默认值
- mLightnessMatrix.setRotate(0, mLumValue); // 控制让红色区在色轮上旋转的角度
- mLightnessMatrix.setRotate(1, mLumValue); // 控制让绿红色区在色轮上旋转的角度
- mLightnessMatrix.setRotate(2, mLumValue); // 控制让蓝色区在色轮上旋转的角度
- // 这里相当于改变的是全图的色相
- break;
- }
- mAllMatrix.reset();
- mAllMatrix.postConcat(mHueMatrix);
- mAllMatrix.postConcat(mSaturationMatrix); // 效果叠加
- mAllMatrix.postConcat(mLightnessMatrix); // 效果叠加
- paint.setColorFilter(new ColorMatrixColorFilter(mAllMatrix));// 设置颜色变换效果
- canvas.drawBitmap(bm, 0, 0, paint); // 将颜色变化后的图片输出到新创建的位图区
- // 返回新的位图,也即调色处理后的图片
- return bmp;
- }
- }
二、主界面:
- package com.jacp.tone;
- import java.util.ArrayList;
- import android.app.Activity;
- import android.graphics.Bitmap;
- import android.graphics.BitmapFactory;
- import android.os.Bundle;
- import android.widget.ImageView;
- import android.widget.LinearLayout;
- import android.widget.SeekBar;
- import android.widget.SeekBar.OnSeekBarChangeListener;
- import com.jacp.tone.view.ToneLayer;
- /**
- * 启动的主界面
- * @author maylian7700@126.com
- *
- */
- public class ImageToneActivity extends Activity implements OnSeekBarChangeListener {
- private ToneLayer mToneLayer;
- private ImageView mImageView;
- private Bitmap mBitmap;
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- init();
- }
- private void init()
- {
- mToneLayer = new ToneLayer(this);
- mBitmap = BitmapFactory.decodeResource(getResources(), R.drawable.test);
- mImageView = (ImageView) findViewById(R.id.img_view);
- mImageView.setImageBitmap(mBitmap);
- ((LinearLayout) findViewById(R.id.tone_view)).addView(mToneLayer.getParentView());
- ArrayList<SeekBar> seekBars = mToneLayer.getSeekBars();
- for (int i = 0, size = seekBars.size(); i < size; i++)
- {
- seekBars.get(i).setOnSeekBarChangeListener(this);
- }
- }
- @Override
- public void onProgressChanged(SeekBar seekBar, int progress,
- boolean fromUser) {
- int flag = (Integer) seekBar.getTag();
- switch (flag)
- {
- case ToneLayer.FLAG_SATURATION:
- mToneLayer.setSaturation(progress);
- break;
- case ToneLayer.FLAG_LUM:
- mToneLayer.setLum(progress);
- break;
- case ToneLayer.FLAG_HUE:
- mToneLayer.setHue(progress);
- break;
- }
- mImageView.setImageBitmap(mToneLayer.handleImage(mBitmap, flag));
- }
- @Override
- public void onStartTrackingTouch(SeekBar seekBar) {
- }
- @Override
- public void onStopTrackingTouch(SeekBar seekBar) {
- }
- }
三、布局文件:
- <?xml version="1.0" encoding="utf-8"?>
- <ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
- android:layout_width="match_parent"
- android:layout_height="match_parent"
- >
- <LinearLayout
- android:layout_width="match_parent"
- android:layout_height="match_parent"
- android:orientation="vertical" >
- <ImageView
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_weight="1"
- android:id="@+id/img_view"
- android:layout_gravity="center"
- />
- <LinearLayout
- android:layout_width="match_parent"
- android:layout_height="wrap_content"
- android:id="@+id/tone_view"
- />
- </LinearLayout>
- </ScrollView>
android图像处理系列之三-- 图片色调饱和度、色相、亮度处理的更多相关文章
- android图像处理系列之三--图片色调饱和度、色相、亮度处理
原图: 处理后: 下面贴代码: 一.图片处理层: package com.jacp.tone.view; import java.util.ArrayList; import android.cont ...
- android图像处理系列之四-- 给图片添加边框(上)
图片处理时,有时需要为图片加一些边框,下面介绍一种为图片添加简单边框的方法. 基本思路是:将边框图片裁剪成八张小图片(图片大小最好一致,不然后面处理会很麻烦),分别对应左上角,左边,左下角,下边,右下 ...
- android图像处理系列之五-- 给图片添加边框(中)
前面一篇讲到给图片加边框的方式,只能给图片加一些有规则的边框,如果想加一些比较精美的效果,就有点麻烦了.下面就给出解决这个问题的思路. 思路是:一些比较精美的花边图片我们是很难用代码控制,就目前本人水 ...
- android图像处理系列之四--给图片添加边框(上)
图片处理时,有时需要为图片加一些边框,下面介绍一种为图片添加简单边框的方法. 基本思路是:将边框图片裁剪成八张小图片(图片大小最好一致,不然后面处理会很麻烦),分别对应左上角,左边,左下角,下边,右下 ...
- android图像处理系列之五--给图片添加边框(中)
前面一篇讲到给图片加边框的方式,只能给图片加一些有规则的边框,如果想加一些比较精美的效果,就有点麻烦了.下面就给出解决这个问题的思路. 思路是:一些比较精美的花边图片我们是很难用代码控制,就目前本人水 ...
- android图像处理系列之七--图片涂鸦,水印-图片叠加
图片涂鸦和水印其实是一个功能,实现的方式是一样的,就是一张大图片和一张小点图片叠加即可.前面在android图像处理系列之六--给图片添加边框(下)-图片叠加中也讲到了图片叠加,里面实现的原理是直接操 ...
- android 图像处理系列合集
为了便于大家对滤镜算法的学习,以后发布的图像处理滤镜系列帖子会在这里汇总,本人第一次写合集,写得不好的地方大家请见谅,手头上虽然有一些滤镜的算法,但是大多不是android版的,教程里的代码大多是我借 ...
- iOS有关图片处理的总结 (四)------图片的饱和度,亮度,对照度。
在做图片处理的时候.会遇到调节图片的饱和度的问题,这里就要用到Core Image这个框架,Core Image是一个非常强大的框架. 它能够让你简单地应用各种滤镜来处理图像,比方改动鲜艳程度, 色泽 ...
- android图像处理系列之六-- 给图片添加边框(下)-图片叠加
前面介绍了一种用透明图片叠加的方式添加花边边框,下面将介绍另外一种图片叠加添加花边边框的方式.前面方法有一个缺点,就是做成PNG图片,图片体积会很大,不是一般的大,比同样的JPG大三倍多,如果项目可以 ...
随机推荐
- leetcode 生成杨辉三角形, 118 119 Pascal's Triangle 1,2
Given numRows, generate the first numRows of Pascal's triangle. For example, given numRows = 5, Retu ...
- Android框架-Volley(二)
1. ImageRequest的用法 前面我们已经学习过了StringRequest和JsonRequest的用法,并且总结出了它们的用法都是非常类似的,基本就是进行以下三步操作即可: 1. 创建一个 ...
- iview中 ...用法
1. 2. 3. 4.可以将divs转为数组解构 5. 解构 6.作为函数的参数 7.作为参数遍历
- orac
#!/bin/bash # Copyright (c) 2013, 2016, Liang Guojun. All rights reserved. # Program: # Check ...
- nginx直接连接操作memcache
原理: Nginx 要设定一个 key 去查 mem ,如果没查到,要有一个回调 php ,并且 key 传给 php. 配置方式: server{ ...
- TP5 上传文件
直接贴上一个完整的代码 /** * 图片上传方法 * @return [type] [description] */ /** * 1 获取到文件 * 2 验证文件的形状是不是符合上传的规则 * 3 i ...
- POJ1201Intervals(差分约束)
题意 给出数轴上的n个区间[ai,bi],每个区间都是连续的int区间. 现在要在数轴上任意取一堆元素,构成一个元素集合V 要求每个区间[ai,bi]和元素集合V的交集至少有ci不同的元素 求集合V最 ...
- 《Python生物信息学数据管理》中文PDF+英文PDF+代码
生物信息学经典资料,解决生物学问题,通过"编程技法"的形式,涵盖尽可能多的组织.分析.表现结果的策略.在每章结尾都会有为生物研究者设计的编程题目,适合教学和自学.由六部分组成:Py ...
- 紫书 习题 10-1UVa 111040(找规律)
通过观察可以得 图可以分成很多个上面一个,中间两个,下面三个的"模板" 这个时候最上面一个知道,最下面得左右知道 那么可以设下面中间为x,左边为a1, 右边为a2, a1a2已知 ...
- awk条件语句
条件语句用于在运行操作之前做一个測试.在前面的章节中,我们看到了模式匹配规则的一些演示样例. 模式匹配规则本质上就是影响输入循环的条件表达式. 在这一部分,我们主要就在action中所使用的条件语句进 ...