0

Xóa nền ảnh với Grabcut

Việc tách vùng ảnh (segmentation) là một bài toán lâu đời trong Computer Vision. Nó có thể được sử dụng để giúp tìm kiếm những vùng được quan tâm (Region of interest) hoặc loại bỏ những vùng chứa ít thông tin. Có nhiều phương pháp tiếp cận với những bài toán tách vùng, có thể kể đến như Watershed, Graph-based cuts,... Sau đây tôi xin trình bày cách phân vùng foreground và background sử dụng Grabcut (thuộc Graph-based cuts) trên Android:

Ở bài trước, tôi đã hướng dẫn các bạn việc cài đặt OpenCV trong Android (cùng vói jni) . GrabCut được xây dựng sẵn trên OpenCV:

cv::Mat result;
cv::Mat bgModel,fgModel;
// GrabCut segmentation
cv::grabCut(image,    // input image
        result,      // segmentation result
        rectangle,   // rectangle containing foreground
        bgModel,fgModel, // models
        iterCount,           // number of iterations
        mode); // use rectangle

Trong đó:

  • iterCount: số vòng lặp mà giải thuật sẽ chạy cho đến khi nhận được kết quả.
  • mode:
    • GC_INIT_WITH_RECT: Hàm sẽ khởi tạo mask và trạng thái theo rectangle argument.
    • GC_INIT_WITH_MASK: Hàm sẽ khởi tạo trạng thái theo mask argument.

Dưới đây là hàm jni của tôi giúp trả về foreground và background từ một ảnh:

JNIEXPORT void JNICALL Java_com_hci_hidefaces_MainActivity_removeBackground(JNIEnv *, jobject obj, jlong src, jlong background, jlong foreground) {
	Mat& image = *(Mat *)src;
	Mat& background_out = *(Mat *)background;
	Mat& foreground_out = *(Mat *)foreground;

	Rect rectangle(10, 10, image.cols -20, image.rows -20);

	Mat result;
	Mat bgModel, fgModel;
	Mat downsampledImage;

	if (image.cols >= MAX_INPUT_IMAGE_WIDTH || image.rows >= MAX_INPUT_IMAGE_HEIGHT) {
		cv::pyrDown(image, downsampledImage, cv::Size(image.cols/2, image.rows/2));
	} else {
		downsampledImage = image.clone();
	}
	// GrabCut segmentation
	grabCut(downsampledImage, result, rectangle, bgModel, fgModel, 1, GC_INIT_WITH_RECT);
	cv::compare(result,cv::GC_PR_FGD,result,cv::CMP_EQ);

	// Separate foreground and background images
	cv::Mat _foreground(downsampledImage.size(),CV_8UC3,cv::Scalar(0,0,0));
	downsampledImage.copyTo(_foreground,result); // bg pixels not copied

	Mat whiteImage(Mat(downsampledImage.rows, downsampledImage.cols, CV_8U));
	whiteImage = Scalar(255);
	Mat complementResult = whiteImage - result;

	cv::Mat _background(downsampledImage.size(),CV_8UC3,cv::Scalar(0,0,0));
	downsampledImage.copyTo(_background, complementResult);
	background_out = _background;
	foreground_out = _foreground;
}

Ở hàm MainActivity, tôi định nghĩa hàm tương ứng chiếu đến hàm jni đó.

	public native void removeBackground(long src, long background, long foreground);

Tôi tạo thêm hàm ApplyFilter để làm mờ, hoặc contourize background để giúp phân biệt 2 vùng bằng mắt dễ hơn như sau:

JNIEXPORT void JNICALL Java_com_hci_hidefaces_MainActivity_applyFilter(JNIEnv *, jobject obj, jlong background, jlong foreground, jlong dst, jint type) {
	Mat& _background = *(Mat *)background;
	Mat& _foreground = *(Mat *)foreground;
	Mat& image_out = *(Mat *)dst;
	Mat _image_out;
	Mat contourBackground;
	Mat blurBackground;
	// Post processing
	switch(type) {
	case TYPE_CONTOUR:
		contourBackground = contourImage(_background);
		_image_out = contourBackground + _foreground;
		break;
	case TYPE_BLUR:
		blurBackground = blurImage(_background, 25);
		_image_out = blurBackground + _foreground;
		break;
	default:
		contourBackground = contourImage(_background);
		_image_out = contourBackground + _foreground;
		break;
	}

	//convert to RGB
	cvtColor(_image_out, image_out, CV_BGR2RGB);
}

Dưới đây là kết quả của chương trình: Mui.jpg

Kết luận

Dựa vào kết quả chúng ta thấy, 2 vùng nhận được chưa thực sự theo ý muốn, nhưng thuật toán GrabCut đã đưa ra kết quả đáng ghi nhận.


All rights reserved

Bình luận

Đăng nhập để bình luận
Avatar
@euclid
thg 5 27, 2015 6:46 CH

Có thời gian thì giải thích kỹ hơn các method vs command trong chương trình đi em, có sample thì các tốt :v Đọc vẫn ko hiểu lắm (yaoming)

Avatar
@dzung.votuan
thg 5 27, 2015 6:53 CH

Dạ vâng, em sẽ bổ sung ạ

Avatar
@sonlm_1993
thg 4 15, 2016 1:37 SA

Anh ơi, anh cho em hỏi vài vấn đề ạ VĐ 1: anh có thể giải thích rõ hơn các tham số và các lấy giá trị chuẩn cho các tham số đó được không ạ. VĐ 2: anh có thể trình bày demo về loại chạy grabcut sử dụng Mask không ạ. vì em thấy thời gian chạy grabcut sử dụng rect lâu ạ. VĐ 3: Đây là đoạn code của em. em chạy thì nó cut được object nhưng nó chưa hoàn hảo và còn một lỗi với output ảnh bị tăng độ sáng và tăng độ tương phản. do em chưa hiểu rõ các tham số nên chưa fix được. anh có thể chỉ cho em không ạ. Em cảm ơn anh. Scalar color = new Scalar(200, 0, 0, 255); Mat dst = new Mat();

    Rect rect = new Rect(p1, p2);
    //Rect rect = new Rect(50,30, 100,200);
    Log.e("LMS", "Rect: " + rect);

    Mat mask = new Mat();
    Log.e("LMS", "Mark type: " + mask.type());
    mask.setTo(new Scalar(125));

    Mat fgdModel = Mat.zeros(1, 13 * 5, CvType.CV_64FC1);
    Mat bgdModel = Mat.zeros(1, 13 * 5, CvType.CV_64FC1);

    Mat imgC3 = new Mat();
    Imgproc.cvtColor(src, imgC3, Imgproc.COLOR_RGBA2RGB);//Convert color rgba to rgb
    Log.e("LMS", "ImgC3: " + imgC3);

    Log.e("LMS", "Grabcut begins");
    long startTime = System.currentTimeMillis();
    Imgproc.grabCut(
            imgC3,/*Color converted*/
            mask,
            rect, /*box of foreground object*/
            bgdModel,
            fgdModel,
            1/*Numb running "grabcut" algorithm*/,
            Imgproc.GC_INIT_WITH_RECT
    );
    long stopTime = System.currentTimeMillis();
    Log.e("LMS", "Grabcut stop: " + (stopTime - startTime) / 1000.0f);
    Mat source = new Mat(1, 1, CvType.CV_8U, new Scalar(3.0));


    Core.compare(mask, source, mask, Core.CMP_EQ);
    Mat foreground = new Mat(src.size(), CvType.CV_8UC3, new Scalar(255, 255, 255));
    src.copyTo(foreground, mask);
    Imgproc.rectangle(src, p1, p2, color);

    Mat background = new Mat();
    try {
        background = Utils.loadResource(context,
                R.drawable.bg_1027);
    } catch (IOException e) {

        e.printStackTrace();
    }

    Mat tmp = new Mat();
    Imgproc.resize(background, tmp, src.size());

    background = tmp;

    Mat tempMask = new Mat(foreground.size(), CvType.CV_8UC1, new Scalar(255, 255, 255));
    Imgproc.cvtColor(foreground, tempMask, 6/* COLOR_BGR2GRAY */);
    Imgproc.threshold(tempMask, tempMask, 200, 255, Imgproc.ADAPTIVE_THRESH_GAUSSIAN_C /* THRESH_BINARY_INV */);

    Mat vals = new Mat(1, 1, CvType.CV_8UC4, new Scalar(0.0));
    dst = new Mat();
    background.setTo(vals, tempMask);
    Imgproc.resize(foreground, tmp, mask.size());
    foreground = tmp;
    Core.add(background, foreground, dst, tempMask);

    Bitmap bitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);
    //convert to Bitmap
    Log.e("LMS", "Convert to Bitmap");
    Utils.matToBitmap(dst, bitmap);
Avatar
0
Viblo
Hãy đăng ký một tài khoản Viblo để nhận được nhiều bài viết thú vị hơn.
Đăng kí