davidmigloz/go-bees

View on GitHub
app/src/main/java/com/davidmiguel/gobees/monitoring/algorithm/processors/Morphology.java

Summary

Maintainability
A
1 hr
Test Coverage
/*
 * GoBees
 * Copyright (c) 2016 - 2017 David Miguel Lozano
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program. If not, see <https://www.gnu.org/licenses/gpl-3.0.txt>.
 */

package com.davidmiguel.gobees.monitoring.algorithm.processors;

import android.support.annotation.NonNull;

import com.davidmiguel.gobees.logging.Log;

import org.opencv.core.Mat;
import org.opencv.core.Size;
import org.opencv.imgproc.Imgproc;

/**
 * Perform morphological transformations (erode/dilate) to improve the bees contours.
 */
public class Morphology implements VideoProcessor {


    private static final int ELEMENT_SHAPE = Imgproc.CV_SHAPE_ELLIPSE;
    private static final int REPETITIONS_DILATE = 3;
    private static final int REPETITIONS_ERODE = 3;
    private static final Mat KERNEL1 = Imgproc.getStructuringElement(ELEMENT_SHAPE, new Size(1, 1));
    private static final Mat KERNEL2 = Imgproc.getStructuringElement(ELEMENT_SHAPE, new Size(2, 2));
    private static final Mat KERNEL3 = Imgproc.getStructuringElement(ELEMENT_SHAPE, new Size(3, 3));

    private Mat kernelErode;
    private Mat kernelDilate;

    /**
     * Default constructor. Initializes the kernels.
     */
    public Morphology() {
        kernelErode = KERNEL3;
        kernelDilate = KERNEL3;
    }

    @Override
    public Mat process(@NonNull Mat frame) {
        if (frame.empty()) {
            Log.e("Invalid input frame.");
            return null;
        }
        Mat tmp = frame.clone();
        // Step 1: erode to remove legs
        Imgproc.erode(tmp, tmp, KERNEL3);
        // Step 2: dilate to join bodies and heads
        Imgproc.dilate(tmp, tmp, KERNEL2);
        for (int i = 0; i < REPETITIONS_DILATE; i++) {
            Imgproc.dilate(tmp, tmp, kernelDilate);
        }
        // Step 3: erode to recover original size
        Imgproc.erode(tmp, tmp, KERNEL1);
        for (int i = 0; i < REPETITIONS_ERODE; i++) {
            Imgproc.erode(tmp, tmp, kernelErode);
        }
        return tmp;
    }

    /**
     * Set erode kernel size.
     *
     * @param size size (1, 2 or 3. Default 3).
     */
    public void setErodeKernel(int size) {
        switch (size) {
            case 1:
                kernelErode = KERNEL1;
                break;
            case 2:
                kernelErode = KERNEL2;
                break;
            case 3:
            default:
                kernelErode = KERNEL3;
        }
    }

    /**
     * Set dilate kernel size.
     *
     * @param size size (1, 2 or 3. Default 3).
     */
    public void setDilateKernel(int size) {
        switch (size) {
            case 1:
                kernelDilate = KERNEL1;
                break;
            case 2:
                kernelDilate = KERNEL2;
                break;
            case 3:
            default:
                kernelDilate = KERNEL3;
        }
    }
}