opencv+android 相片美颜

opencv+android 相片美颜

入门opencv 写个demo

  • **步骤
  • **代码

步骤

下载 opencv for android SDK https://opencv.org/releases.html
Android Studio 新建android工程
activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.example.ligang.myapplication.MainActivity">
    <LinearLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical"
        tools:context=".MainActivity">

        <ImageView
            android:id="@+id/imageView"
            android:layout_width="match_parent"
            android:layout_height="505dp"
            android:layout_marginLeft="10dp"
            android:layout_marginRight="10dp"
            android:contentDescription="图片"
            app:srcCompat="@mipmap/ic_launcher_round"
            tools:ignore="VectorDrawableCompat" />

        <SeekBar
            android:id="@+id/seekBar"
            android:layout_marginLeft="20dp"
            android:layout_marginRight="20dp"
            android:layout_marginTop="10dp"
            android:layout_marginBottom="10dp"
            android:layout_width="match_parent"
            android:layout_height="30dp"
            android:max="500"
            android:progress="0"
            android:secondaryProgress="1"
            tools:ignore="MissingConstraints"/>

    </LinearLayout>

</android.support.constraint.ConstraintLayout>
  1. 在 Android Studio New=> import Module=>找到OpenCV-android-sdk\sdk\java 这一层
  2. 拷贝OpenCV-android-sdk\sdk\native\libs 所有文件 到 MyApplication\app\libs下
    MainActivity.java
package com.example.ligang.myapplication;


import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.AsyncTask;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.ImageView;
import android.widget.SeekBar;

import org.opencv.android.OpenCVLoader;
import org.opencv.android.Utils;
import org.opencv.core.Core;
import org.opencv.core.CvType;
import org.opencv.core.Mat;
import org.opencv.core.Scalar;
import org.opencv.core.Size;
import org.opencv.imgproc.Imgproc;

import java.io.InputStream;

import static android.os.AsyncTask.Status.RUNNING;

public class MainActivity extends AppCompatActivity {
    private static final String TAG = "test";
    private double max_size = 1024;
    private int PICK_IMAGE_REQUEST = 1;
    private ImageView imageView;
    private Bitmap selectbp;
    private Uri uri;
    private BilateralFilterTask bilateralFilterTask;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        staticLoadCVLibraries();
        bilateralFilterTask = new BilateralFilterTask();
        imageView = (ImageView)findViewById(R.id.imageView);
        imageView.setScaleType(ImageView.ScaleType.FIT_CENTER);
        imageView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                selectImage();
            }
        });

        SeekBar seekBar = (SeekBar)findViewById(R.id.seekBar);
        seekBar.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener() {
            @Override
            public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {

            }

            @Override
            public void onStartTrackingTouch(SeekBar seekBar) {

            }

            @Override
            public void onStopTrackingTouch(SeekBar seekBar) {
                if (imageView != null) {
                    readFile();
                    convertGray(seekBar.getProgress() / 100);
                }
            }
        });
    }

    //OpenCV库静态加载并初始化
    private void staticLoadCVLibraries(){
        boolean load = OpenCVLoader.initDebug();
        if(load) {
            Log.i("CV", "Open CV Libraries loaded...");
        }
    }
    private class BilateralFilterTask extends AsyncTask<Float, Bitmap, Bitmap> {
        @Override
        protected void onPreExecute() {
            super.onPreExecute();
        }

        protected Bitmap doInBackground(Float... bilityTraversal) {
            Bitmap processbp = Bitmap.createBitmap(selectbp);

            if (bilityTraversal[0] > 0) {
// 磨皮美颜算法
                int dx = (int)bilityTraversal[0].floatValue() * 5; // 双边滤波参数之一
                double fc = bilityTraversal[0] * 12.5; // 双边滤波参数之一
                double p = 0.1f; // 透明度
                Mat image = new Mat(), dst = new Mat(), matBilFilter = new Mat(), matGaussSrc = new Mat(), matGaussDest = new Mat(), matTmpDest = new Mat(), matSubDest = new Mat(), matTmpSrc = new Mat();

                // 双边滤波
                Utils.bitmapToMat(processbp, image);
                Imgproc.cvtColor(image, image, Imgproc.COLOR_BGRA2BGR);
                Imgproc.bilateralFilter(image, matBilFilter, dx, fc, fc);

                Core.subtract(matBilFilter, image, matSubDest);
                Core.add(matSubDest, new Scalar(128, 128, 128, 128), matGaussSrc);
                // 高斯模糊
                Imgproc.GaussianBlur(matGaussSrc, matGaussDest, new Size(2 * bilityTraversal[0] - 1, 2 * bilityTraversal[0] - 1), 0, 0);
                matGaussDest.convertTo(matTmpSrc, matGaussDest.type(), 2, -255);
                Core.add(image, matTmpSrc, matTmpDest);
                Core.addWeighted(image, p, matTmpDest, 1 - p, 0.0, dst);

                Core.add(dst, new Scalar(10, 10, 10), dst);
                Utils.matToBitmap(dst, processbp);

            }

            return processbp;
        }

        @Override
        protected void onPostExecute(Bitmap bitmap) {
            imageView.setImageBitmap(bitmap);
        }
    }
    private void convertGray(float bilityTraversal) {
        bilateralFilterTask.cancel(true);
        bilateralFilterTask = new BilateralFilterTask();
        bilateralFilterTask.execute(bilityTraversal);
    }

    private void readFile() {
        try {
            Log.d("image-tag", "start to decode selected image now...");
            InputStream input = getContentResolver().openInputStream(uri);
            BitmapFactory.Options options = new BitmapFactory.Options();
            options.inJustDecodeBounds = true;
            BitmapFactory.decodeStream(input, null, options);
            int raw_width = options.outWidth;
            int raw_height = options.outHeight;
            int max = Math.max(raw_width, raw_height);
            int newWidth = raw_width;
            int newHeight = raw_height;
            int inSampleSize = 1;
            if(max > max_size) {
                newWidth = raw_width / 2;
                newHeight = raw_height / 2;
                while((newWidth/inSampleSize) > max_size || (newHeight/inSampleSize) > max_size) {
                    inSampleSize *=2;
                }
            }

            options.inSampleSize = inSampleSize;
            options.inJustDecodeBounds = false;
            options.inPreferredConfig = Bitmap.Config.ARGB_8888;
            selectbp = BitmapFactory.decodeStream(getContentResolver().openInputStream(uri), null, options);

        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if(requestCode == PICK_IMAGE_REQUEST && resultCode == RESULT_OK && data != null && data.getData() != null) {
            uri = data.getData();
            readFile();
            imageView.setImageBitmap(selectbp);
        }
    }

    private void selectImage() {
        Intent intent = new Intent();
        intent.setType("image/*");
        intent.setAction(Intent.ACTION_GET_CONTENT);
        startActivityForResult(Intent.createChooser(intent,"选择图像..."), PICK_IMAGE_REQUEST);
    }
}

点击空白添加图片 seekbar来设置美颜质量。
####点击中心区域在SD卡中找图
点击中心区域在SD卡中找图

####图不好找,网上盗去前人的源图。
图不好找,网上盗去前人的源图。

####美颜效果
美颜效果

感觉可以结合许多功能做出许多新的效果来。
####代码下载地址http://download.csdn.net/download/yiqiyihuiligang/10177914

####github https://github.com/liganggit/opencv-android.git

  • 0
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值