Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Create demo.java #257

Open
wants to merge 2 commits into
base: main
Choose a base branch
from
Open
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
143 changes: 143 additions & 0 deletions models/face_recognition_sface/demo.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,143 @@
import nu.pattern.OpenCV;
import org.opencv.core.Mat;
import org.opencv.core.Size;
import org.opencv.imgcodecs.Imgcodecs;
import org.opencv.objdetect.FaceDetectorYN;
import org.opencv.objdetect.FaceRecognizerSF;

/**
* the java demo of FaceRecognizerSF
*
* you need the dependencies of maven:
*
* *OpenCV Java bindings packaged with native libraries,
* seamlessly delivered as a turn-key Maven dependency.
* You don't need to download OpenCV to support the relevant features.
* doc:https://github.com/openpnp/opencv?tab=readme-ov-file
*
* https://mvnrepository.com/artifact/org.openpnp/opencv
* <dependency>
* <groupId>org.openpnp</groupId>
* <artifactId>opencv</artifactId>
* <version>4.9.0-0</version>
* </dependency>
*/
public class FaceRecognizer {

private static double cosine_similar_threshold = 0.363;

private static double l2norm_similar_threshold = 1.128;

// Your full path of yunet model
private static String faceDetectModelPath = "/face_detection_yunet_2023mar.onnx";
// state faceDetector
private static FaceDetectorYN faceDetector = null;

// Your full path of sface model
private static String faceRecognizModelPath = "/face_recognition_sface_2021dec.onnx";
// state faceRecognizer
private static FaceRecognizerSF faceRecognizer = null;

public static void main(String[] args) {
// You need to use the full path of img please
boolean b = faceRecognizer("imgPathA", "imgPathB");
System.out.println(b);
}

public static boolean faceRecognizer(String imgPathA, String imgPathB) {
// Load for opencv
OpenCV.loadLocally();
// Load for faceDetector
loadFaceDetector();
// Load for faceRecognizer
loadFaceRecognizer();

return faceRecognizerUtil(imgPathA, imgPathB);
}

// Load of faceDetector
private static void loadFaceDetector() {
if (faceDetector != null) {
return;
}
// You could use the full path for faceDetect model instead to get the resource
faceDetector = FaceDetectorYN.create(faceDetectModelPath, "", new Size());
}

// Load for faceRecognizer
private static void loadFaceRecognizer() {
if (faceRecognizer != null) {
return;
}
// You could use the full path for faceRecogniz model instead to get the resource
faceRecognizer = FaceRecognizerSF.create(faceRecognizModelPath, "");
}

/**
* FaceRecogniz. Calculating the distance between two face features
*
* @param imgPathA the path of imgA
* @param imgPathB the path of imgB
*/
private static boolean faceRecognizerUtil(String imgPathA, String imgPathB) {
// 1.Read img convert to a mat
Mat imgA = Imgcodecs.imread(imgPathA);
Mat imgB = Imgcodecs.imread(imgPathB);

// 2.Detect face from given image
Mat faceA = new Mat();
faceDetector.setInputSize(imgA.size());
faceDetector.detect(imgA, faceA);
Mat faceB = new Mat();
faceDetector.setInputSize(imgB.size());
faceDetector.detect(imgB, faceB);

/*
// 3.Draw face info in img
// <a href="https://docs.opencv.org/4.8.0/df/d20/classcv_1_1FaceDetectorYN.html">FaceDetectorYN</a>
// Draw a face frame in imgA
Imgproc.rectangle(imgA, new Rect((int) (faceA.get(0, 0)[0]), (int) (faceA.get(0, 1)[0]), (int) (faceA.get(0, 2)[0]), (int) (faceA.get(0, 3)[0])), new Scalar(0, 255, 0), 2);
// Draw eyes/nose/mouth points on imgB
Imgproc.circle(imgA, new Point(faceA.get(0, 4)[0], faceA.get(0, 5)[0]), 2, new Scalar(255, 0, 0), 2);
Imgproc.circle(imgA, new Point(faceA.get(0, 6)[0], faceA.get(0, 7)[0]), 2, new Scalar(0, 0, 255), 2);
Imgproc.circle(imgA, new Point(faceA.get(0, 8)[0], faceA.get(0, 9)[0]), 2, new Scalar(0, 255, 0), 2);
Imgproc.circle(imgA, new Point(faceA.get(0, 10)[0], faceA.get(0, 11)[0]), 2, new Scalar(255, 0, 255), 2);
Imgproc.circle(imgA, new Point(faceA.get(0, 12)[0], faceA.get(0, 13)[0]), 2, new Scalar(0, 255, 255), 2);
// Imgcodecs.imwrite("", imgA);

// Draw a face frame in imgB
Imgproc.rectangle(imgB, new Rect((int) (faceB.get(0, 0)[0]), (int) (faceB.get(0, 1)[0]), (int) (faceB.get(0, 2)[0]), (int) (faceB.get(0, 3)[0])), new Scalar(0, 255, 0), 2);
// Draw eyes/nose/mouth in imgB
Imgproc.circle(imgB, new Point(faceB.get(0, 4)[0], faceB.get(0, 5)[0]), 2, new Scalar(255, 0, 0), 2);
Imgproc.circle(imgB, new Point(faceB.get(0, 6)[0], faceB.get(0, 7)[0]), 2, new Scalar(0, 0, 255), 2);
Imgproc.circle(imgB, new Point(faceB.get(0, 8)[0], faceB.get(0, 9)[0]), 2, new Scalar(0, 255, 0), 2);
Imgproc.circle(imgB, new Point(faceB.get(0, 10)[0], faceB.get(0, 11)[0]), 2, new Scalar(255, 0, 255), 2);
Imgproc.circle(imgB, new Point(faceB.get(0, 12)[0], faceB.get(0, 13)[0]), 2, new Scalar(0, 255, 255), 2);
// Imgcodecs.imwrite("", imgB);*/
12341234-yangboran marked this conversation as resolved.
Show resolved Hide resolved

// 4.Aligning image to put face on the standard position
Mat alignFaceA = new Mat();
faceRecognizer.alignCrop(imgA, faceA.row(0), alignFaceA);
Mat alignFaceB = new Mat();
faceRecognizer.alignCrop(imgB, faceB.row(0), alignFaceB);

// 5.Extracting face feature from aligned image
Mat featureA = new Mat();
faceRecognizer.feature(alignFaceA, featureA);
featureA = featureA.clone();
Mat featureB = new Mat();
faceRecognizer.feature(alignFaceB, featureB);
featureB = featureB.clone();

// 6.FaceRecogniz. Calculating the distance between two face features. If the condition is met, it returns true
// Get cosine similar
double match1 = faceRecognizer.match(featureA, featureB, FaceRecognizerSF.FR_COSINE);
// Get l2norm similar
double match2 = faceRecognizer.match(featureA, featureB, FaceRecognizerSF.FR_NORM_L2);
if (match1 >= cosine_similar_threshold && match2 <= l2norm_similar_threshold) {
return true;
} else {
return false;
}
}
}