/** * Copyright (c) 2016 Affectiva Inc. * See the file license.txt for copying permission. */ package com.affectiva.affdexme; import android.Manifest; import android.app.ActivityManager; import android.content.Context; import android.content.DialogInterface; import android.content.Intent; import android.content.SharedPreferences; import android.content.pm.PackageManager; import android.graphics.Bitmap; import android.graphics.Canvas; import android.graphics.Paint; import android.graphics.Rect; import android.graphics.Typeface; import android.media.MediaPlayer; import android.os.Bundle; import android.os.Environment; import android.os.SystemClock; import android.preference.PreferenceManager; import android.support.annotation.NonNull; import android.support.v4.app.ActivityCompat; import android.support.v4.content.ContextCompat; import android.app.AlertDialog; import android.support.v7.app.AppCompatActivity; import android.text.format.DateFormat; import android.util.DisplayMetrics; import android.util.Log; import android.view.KeyEvent; import android.view.MotionEvent; import android.view.SurfaceView; import android.view.View; import android.view.ViewGroup; import android.view.WindowManager; import android.widget.Button; import android.widget.ImageButton; import android.widget.LinearLayout; import android.widget.ProgressBar; import android.widget.RelativeLayout; import android.widget.TextView; import android.widget.Toast; import com.affectiva.android.affdex.sdk.Frame; import com.affectiva.android.affdex.sdk.Frame.ROTATE; import com.affectiva.android.affdex.sdk.detector.CameraDetector; import com.affectiva.android.affdex.sdk.detector.Detector; import com.affectiva.android.affdex.sdk.detector.Face; import java.io.File; import java.io.IOException; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.util.Date; import java.util.List; import java.util.Locale; /* * AffdexMe is an app that demonstrates the use of the Affectiva Android SDK. It uses the * front-facing camera on your Android device to view, process and analyze live video of your face. * Start the app and you will see your own face on the screen and metrics describing your * expressions. * * Tapping the screen will bring up a menu with options to display the Processed Frames Per Second metric, * display facial tracking points, and control the rate at which frames are processed by the SDK. * * Most of the methods in this file control the application's UI. Therefore, if you are just interested in learning how the Affectiva SDK works, * you will find the calls relevant to the use of the SDK in the initializeCameraDetector(), startCamera(), stopCamera(), * and onImageResults() methods. * * This class implements the Detector.ImageListener interface, allowing it to receive the onImageResults() event. * This class implements the Detector.FaceListener interface, allowing it to receive the onFaceDetectionStarted() and * onFaceDetectionStopped() events. * This class implements the CameraDetector.CameraSurfaceViewListener interface, allowing it to receive * onSurfaceViewAspectRatioChanged() events. * * In order to use this project, you will need to: * - Obtain the SDK from Affectiva (visit http://www.affdex.com/mobile-sdk) * - Copy the SDK assets folder contents into this project's assets folder under AffdexMe/app/src/main/assets * - Copy the contents of the SDK's libs folder into this project's libs folder under AffdexMe/app/lib * - Copy the armeabi-v7a folder (found in the SDK libs folder) into this project's jniLibs folder under AffdexMe/app/src/main/jniLibs * - Add your license file to the assets/Affdex folder and rename to license.txt. * (Note: if you name the license file something else you will need to update the licensePath in the initializeCameraDetector() method in MainActivity) * - Build the project * - Run the app on an Android device with a front-facing camera */ public class MainActivity extends AppCompatActivity implements Detector.FaceListener, Detector.ImageListener, CameraDetector.CameraEventListener, View.OnTouchListener, ActivityCompat.OnRequestPermissionsResultCallback, DrawingView.DrawingThreadEventListener { public static final int MAX_SUPPORTED_FACES = 3; public static final boolean STORE_RAW_SCREENSHOTS = false; // setting to enable saving the raw images when taking screenshots public static final int NUM_METRICS_DISPLAYED = 7; private static final String LOG_TAG = "AffdexMe"; private static final int CAMERA_PERMISSIONS_REQUEST = 42; //value is arbitrary (between 0 and 255) private static final int EXTERNAL_STORAGE_PERMISSIONS_REQUEST = 73; int cameraPreviewWidth = 0; int cameraPreviewHeight = 0; CameraDetector.CameraType cameraType; boolean mirrorPoints = false; private boolean cameraPermissionsAvailable = false; private boolean storagePermissionsAvailable = false; private CameraDetector detector = null; private RelativeLayout metricViewLayout; private LinearLayout leftMetricsLayout; private LinearLayout rightMetricsLayout; private TextView[] metricDisplays; private TextView[] metricNames; private TextView fpsName; private TextView fpsPct; private TextView pleaseWaitTextView; private ProgressBar progressBar; private RelativeLayout mainLayout; //layout, to be resized, containing all UI elements private RelativeLayout progressBarLayout; //layout used to show progress circle while camera is starting private LinearLayout permissionsUnavailableLayout; //layout used to notify the user that not enough permissions have been granted to use the app private SurfaceView cameraView; //SurfaceView used to display camera images private DrawingView drawingView; //SurfaceView containing its own thread, used to draw facial tracking dots private Frame mostRecentFrame; private boolean isMenuVisible = false; private boolean isFPSVisible = false; private boolean isMenuShowingForFirstTime = true; private long firstSystemTime = 0; private float numberOfFrames = 0; private long timeToUpdate = 0; private boolean isFrontFacingCameraDetected = true; private boolean isBackFacingCameraDetected = true; private boolean multiFaceModeEnabled = false; private TextView showFaceText; private MediaPlayer mp; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); getWindow().addFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN); //To maximize UI space, we declare our app to be full-screen preproccessMetricImages(); setContentView(R.layout.activity_main); initializeUI(); checkForCameraPermissions(); determineCameraAvailability(); initializeCameraDetector(); mp = MediaPlayer.create(this, R.raw.beyond_sleep_vo); } /** * Only load the files onto disk the first time the app opens */ private void preproccessMetricImages() { Context context = getBaseContext(); for (Face.EMOJI emoji : Face.EMOJI.values()) { if (emoji.equals(Face.EMOJI.UNKNOWN)) { continue; } String emojiResourceName = emoji.name().trim().replace(' ', '_').toLowerCase(Locale.US).concat("_emoji"); String emojiFileName = emojiResourceName + ".png"; ImageHelper.preproccessImageIfNecessary(context, emojiFileName, emojiResourceName); } ImageHelper.preproccessImageIfNecessary(context, "female_glasses.png", "female_glasses"); ImageHelper.preproccessImageIfNecessary(context, "female_noglasses.png", "female_noglasses"); ImageHelper.preproccessImageIfNecessary(context, "male_glasses.png", "male_glasses"); ImageHelper.preproccessImageIfNecessary(context, "male_noglasses.png", "male_noglasses"); ImageHelper.preproccessImageIfNecessary(context, "unknown_glasses.png", "unknown_glasses"); ImageHelper.preproccessImageIfNecessary(context, "unknown_noglasses.png", "unknown_noglasses"); } private void checkForCameraPermissions() { cameraPermissionsAvailable = ContextCompat.checkSelfPermission( getApplicationContext(), Manifest.permission.CAMERA) == PackageManager.PERMISSION_GRANTED; if (!cameraPermissionsAvailable) { // Should we show an explanation? if (ActivityCompat.shouldShowRequestPermissionRationale(this, Manifest.permission.CAMERA)) { // Show an explanation to the user *asynchronously* -- don't block // this thread waiting for the user's response! After the user // sees the explanation, try again to request the permission. showPermissionExplanationDialog(CAMERA_PERMISSIONS_REQUEST); } else { // No explanation needed, we can request the permission. requestCameraPermissions(); } } } private void checkForStoragePermissions() { storagePermissionsAvailable = ContextCompat.checkSelfPermission( getApplicationContext(), Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED; if (!storagePermissionsAvailable) { // Should we show an explanation? if (ActivityCompat.shouldShowRequestPermissionRationale(this, Manifest.permission.WRITE_EXTERNAL_STORAGE)) { // Show an explanation to the user *asynchronously* -- don't block // this thread waiting for the user's response! After the user // sees the explanation, try again to request the permission. showPermissionExplanationDialog(EXTERNAL_STORAGE_PERMISSIONS_REQUEST); } else { // No explanation needed, we can request the permission. requestStoragePermissions(); } } else { } } private void requestStoragePermissions() { if (!storagePermissionsAvailable) { ActivityCompat.requestPermissions( this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, EXTERNAL_STORAGE_PERMISSIONS_REQUEST); // EXTERNAL_STORAGE_PERMISSIONS_REQUEST is an app-defined int constant that must be between 0 and 255. // The callback method gets the result of the request. } } private void requestCameraPermissions() { if (!cameraPermissionsAvailable) { ActivityCompat.requestPermissions( this, new String[]{Manifest.permission.CAMERA}, CAMERA_PERMISSIONS_REQUEST); // CAMERA_PERMISSIONS_REQUEST is an app-defined int constant that must be between 0 and 255. // The callback method gets the result of the request. } } @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { super.onRequestPermissionsResult(requestCode, permissions, grantResults); if (requestCode == CAMERA_PERMISSIONS_REQUEST) { for (int i = 0; i < permissions.length; i++) { String permission = permissions[i]; int grantResult = grantResults[i]; if (permission.equals(Manifest.permission.CAMERA)) { cameraPermissionsAvailable = (grantResult == PackageManager.PERMISSION_GRANTED); } } if (!cameraPermissionsAvailable) { permissionsUnavailableLayout.setVisibility(View.VISIBLE); } else { permissionsUnavailableLayout.setVisibility(View.GONE); } } if (requestCode == EXTERNAL_STORAGE_PERMISSIONS_REQUEST) { for (int i = 0; i < permissions.length; i++) { String permission = permissions[i]; int grantResult = grantResults[i]; if (permission.equals(Manifest.permission.WRITE_EXTERNAL_STORAGE)) { storagePermissionsAvailable = (grantResult == PackageManager.PERMISSION_GRANTED); } } if (storagePermissionsAvailable) { // resume taking the screenshot } } } private void showPermissionExplanationDialog(int requestCode) { final AlertDialog.Builder alertDialogBuilder = new AlertDialog.Builder( MainActivity.this); // set title alertDialogBuilder.setTitle(getResources().getString(R.string.insufficient_permissions)); // set dialog message if (requestCode == CAMERA_PERMISSIONS_REQUEST) { alertDialogBuilder .setMessage(getResources().getString(R.string.permissions_camera_needed_explanation)) .setCancelable(false) .setPositiveButton(getResources().getString(R.string.understood), new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int id) { dialog.cancel(); requestCameraPermissions(); } }); } else if (requestCode == EXTERNAL_STORAGE_PERMISSIONS_REQUEST) { alertDialogBuilder .setMessage(getResources().getString(R.string.permissions_storage_needed_explanation)) .setCancelable(false) .setPositiveButton(getResources().getString(R.string.understood), new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int id) { dialog.cancel(); requestStoragePermissions(); } }); } // create alert dialog AlertDialog alertDialog = alertDialogBuilder.create(); // show it alertDialog.show(); } /** * We check to make sure the device has a front-facing camera. * If it does not, we obscure the app with a notice informing the user they cannot * use the app. */ void determineCameraAvailability() { PackageManager manager = getPackageManager(); isFrontFacingCameraDetected = manager.hasSystemFeature(PackageManager.FEATURE_CAMERA_FRONT); isBackFacingCameraDetected = manager.hasSystemFeature(PackageManager.FEATURE_CAMERA); if (!isFrontFacingCameraDetected && !isBackFacingCameraDetected) { progressBar.setVisibility(View.INVISIBLE); pleaseWaitTextView.setVisibility(View.INVISIBLE); TextView notFoundTextView = (TextView) findViewById(R.id.not_found_textview); notFoundTextView.setVisibility(View.VISIBLE); } //set default camera settings SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(this); //restore the camera type settings String cameraTypeName = sharedPreferences.getString("cameraType", CameraDetector.CameraType.CAMERA_FRONT.name()); if (cameraTypeName.equals(CameraDetector.CameraType.CAMERA_FRONT.name())) { cameraType = CameraDetector.CameraType.CAMERA_FRONT; mirrorPoints = true; } else { cameraType = CameraDetector.CameraType.CAMERA_BACK; mirrorPoints = false; } } void initializeUI() { //Get handles to UI objects ViewGroup activityLayout = (ViewGroup) findViewById(android.R.id.content); progressBarLayout = (RelativeLayout) findViewById(R.id.progress_bar_cover); permissionsUnavailableLayout = (LinearLayout) findViewById(R.id.permissionsUnavialableLayout); metricViewLayout = (RelativeLayout) findViewById(R.id.metric_view_group); leftMetricsLayout = (LinearLayout) findViewById(R.id.left_metrics); rightMetricsLayout = (LinearLayout) findViewById(R.id.right_metrics); mainLayout = (RelativeLayout) findViewById(R.id.main_layout); // fpsPct = (TextView) findViewById(R.id.fps_value); // fpsName = (TextView) findViewById(R.id.fps_name); cameraView = (SurfaceView) findViewById(R.id.camera_preview); drawingView = (DrawingView) findViewById(R.id.drawing_view); progressBar = (ProgressBar) findViewById(R.id.progress_bar); pleaseWaitTextView = (TextView) findViewById(R.id.please_wait_textview); Button retryPermissionsButton = (Button) findViewById(R.id.retryPermissionsButton); //Initialize views to display metrics metricNames = new TextView[NUM_METRICS_DISPLAYED]; metricNames[0] = (TextView) findViewById(R.id.metric_name_0); metricNames[1] = (TextView) findViewById(R.id.metric_name_1); metricNames[2] = (TextView) findViewById(R.id.metric_name_2); metricNames[3] = (TextView) findViewById(R.id.metric_name_3); metricNames[4] = (TextView) findViewById(R.id.metric_name_4); metricNames[5] = (TextView) findViewById(R.id.metric_name_5); metricNames[6] = (TextView) findViewById(R.id.metric_name_6); metricDisplays = new TextView[NUM_METRICS_DISPLAYED]; metricDisplays[0] = (TextView) findViewById(R.id.metric_pct_0); metricDisplays[1] = (TextView) findViewById(R.id.metric_pct_1); metricDisplays[2] = (TextView) findViewById(R.id.metric_pct_2); metricDisplays[3] = (TextView) findViewById(R.id.metric_pct_3); metricDisplays[4] = (TextView) findViewById(R.id.metric_pct_4); metricDisplays[5] = (TextView) findViewById(R.id.metric_pct_5); metricDisplays[6] = (TextView) findViewById(R.id.metric_pct_6); showFaceText = (TextView) findViewById(R.id.no_face_found); //Load Application Font and set UI Elements to use it Typeface face = Typeface.createFromAsset(getAssets(), "fonts/Square.ttf"); // for (TextView textView : metricNames) { // textView.setTypeface(face); // } // for (TextView metricDisplay : metricDisplays) { // metricDisplay.setTypeface(face); // } // fpsPct.setTypeface(face); // fpsName.setTypeface(face); drawingView.setTypeface(face); pleaseWaitTextView.setTypeface(face); //Hide left and right metrics by default (will be made visible when face detection starts) leftMetricsLayout.setAlpha(0); rightMetricsLayout.setAlpha(0); /** * This app uses two SurfaceView objects: one to display the camera image and the other to draw facial tracking dots. * Since we want the tracking dots to appear over the camera image, we use SurfaceView.setZOrderMediaOverlay() to indicate that * cameraView represents our 'media', and drawingView represents our 'overlay', so that Android will render them in the * correct order. */ drawingView.setZOrderMediaOverlay(true); cameraView.setZOrderMediaOverlay(false); //Attach event listeners to the menu and edit box activityLayout.setOnTouchListener(this); //Attach event listerner to drawing view drawingView.setEventListener(this); /* * This app sets the View.SYSTEM_UI_FLAG_HIDE_NAVIGATION flag. Unfortunately, this flag causes * Android to steal the first touch event after the navigation bar has been hidden, a touch event * which should be used to make our menu visible again. Therefore, we attach a listener to be notified * when the navigation bar has been made visible again, which corresponds to the touch event that Android * steals. If the menu bar was not visible, we make it visible. */ activityLayout.setOnSystemUiVisibilityChangeListener(new View.OnSystemUiVisibilityChangeListener() { @Override public void onSystemUiVisibilityChange(int uiCode) { if ((uiCode == 0) && (!isMenuVisible)) { setMenuVisible(true); } } }); retryPermissionsButton.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { requestCameraPermissions(); } }); } void initializeCameraDetector() { /* Put the SDK in camera mode by using this constructor. The SDK will be in control of * the camera. If a SurfaceView is passed in as the last argument to the constructor, * that view will be painted with what the camera sees. */ detector = new CameraDetector(this, cameraType, cameraView, (multiFaceModeEnabled ? MAX_SUPPORTED_FACES : 1), Detector.FaceDetectorMode.LARGE_FACES); detector.setImageListener(this); detector.setFaceListener(this); detector.setOnCameraEventListener(this); } /* * We use onResume() to restore application settings using the SharedPreferences object */ @Override public void onResume() { super.onResume(); checkForCameraPermissions(); restoreApplicationSettings(); setMenuVisible(false); isMenuShowingForFirstTime = true; } private void setMultiFaceModeEnabled(boolean isEnabled) { //if setting change is necessary if (isEnabled != multiFaceModeEnabled) { // change the setting, stop the detector, and reinitialize it to change the setting multiFaceModeEnabled = isEnabled; stopDetector(); initializeCameraDetector(); } } /* * We use the Shared Preferences object to restore application settings. */ public void restoreApplicationSettings() { SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(this); //restore the camera type settings String cameraTypeName = sharedPreferences.getString("cameraType", CameraDetector.CameraType.CAMERA_FRONT.name()); if (cameraTypeName.equals(CameraDetector.CameraType.CAMERA_FRONT.name())) { setCameraType(CameraDetector.CameraType.CAMERA_FRONT); } else { setCameraType(CameraDetector.CameraType.CAMERA_BACK); } //restore the multiface mode setting to reset the detector if necessary if (sharedPreferences.getBoolean("multiface", false)) { // default to false setMultiFaceModeEnabled(true); } else { setMultiFaceModeEnabled(false); } //restore camera processing rate int detectorProcessRate = PreferencesUtils.getFrameProcessingRate(sharedPreferences); detector.setMaxProcessRate(detectorProcessRate); drawingView.invalidateDimensions(); setFPSVisible(false); setTrackPoints(true); detector.setDetectAllAppearances(false); setShowAppearance(false); detector.setDetectAllEmojis(false); setShowEmoji(false); //populate metric displays // for (int n = 0; n < NUM_METRICS_DISPLAYED; n++) { // activateMetric(n, PreferencesUtils.getMetricFromPrefs(sharedPreferences, n)); // } detector.setDetectAllEmotions(true); int i=0; // for(MetricsManager.Emotions emotion: MetricsManager.Emotions.values()) { // if(emotion.equals(MetricsManager.Emotions.ENGAGEMENT) || emotion.equals(MetricsManager.Emotions.VALENCE)) { // continue; // } // activateMetric(i, emotion); // i++; // } //if we are in multiface mode, we need to enable the detection of all emotions // if (multiFaceModeEnabled) { // detector.setDetectAllEmotions(true); // } } /** * Populates a TextView to display a metric name and readies a MetricDisplay to display the value. * Uses reflection to: * -enable the corresponding metric in the Detector object by calling Detector.setDetect() * -save the Method object that will be invoked on the Face object received in onImageResults() to get the metric score */ void activateMetric(int index, MetricsManager.Emotions metric) { Method getFaceScoreMethod = null; //The method that will be used to get a metric score try { // metricNames[index].setText(MetricsManager.getUpperCaseName(metric)); getFaceScoreMethod = Face.Emotions.class.getMethod("get" + MetricsManager.getCamelCase(metric)); //The MetricDisplay for Valence is unique; it shades it color depending on the metric value // if (metric == MetricsManager.Emotions.VALENCE) { // metricDisplays[index].setIsShadedMetricView(true); // } else { // metricDisplays[index].setIsShadedMetricView(false); // } } catch (NoSuchMethodException e) { Log.e(LOG_TAG, String.format("No such method while using reflection to generate methods for %s", metric.toString()), e); } // metricDisplays[index].setMetricToDisplay(metric, getFaceScoreMethod); } /** * Reset the variables used to calculate processed frames per second. **/ public void resetFPSCalculations() { firstSystemTime = SystemClock.elapsedRealtime(); timeToUpdate = firstSystemTime + 1000L; numberOfFrames = 0; } /** * We want to start the camera as late as possible, so it does not freeze the application before it has been visually resumed. * We thus post a runnable that will take care of starting the camera. * We also reset variables used to calculate the Processed Frames Per Second. */ @Override public void onWindowFocusChanged(boolean hasFocus) { if (hasFocus && isFrontFacingCameraDetected) { cameraView.post(new Runnable() { @Override public void run() { mainWindowResumedTasks(); } }); } if(!hasFocus) { // Close every kind of system dialog // Intent closeDialog = new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS); // sendBroadcast(closeDialog); } } void mainWindowResumedTasks() { //Notify the user that they can't use the app without authorizing these permissions. if (!cameraPermissionsAvailable) { permissionsUnavailableLayout.setVisibility(View.VISIBLE); return; } startDetector(); if (!drawingView.isDimensionsNeeded()) { progressBarLayout.setVisibility(View.GONE); } resetFPSCalculations(); cameraView.postDelayed(new Runnable() { @Override public void run() { if (isMenuShowingForFirstTime) { setMenuVisible(false); } } }, 5000); } void startDetector() { if (!isBackFacingCameraDetected && !isFrontFacingCameraDetected) return; //without any cameras detected, we cannot proceed detector.setDetectValence(false); //this app will always detect valence if (!detector.isRunning()) { try { detector.start(); } catch (Exception e) { Log.e(LOG_TAG, e.getMessage()); } } } @Override public void onFaceDetectionStarted() { leftMetricsLayout.animate().alpha(1); //make left and right metrics appear rightMetricsLayout.animate().alpha(1); showFaceText.setVisibility(View.GONE); resetFPSCalculations(); //Since the FPS may be different whether a face is being tracked or not, reset variables. if(mp != null && !mp.isPlaying()) { mp.start(); } } @Override public void onFaceDetectionStopped() { showFaceText.setVisibility(View.VISIBLE); performFaceDetectionStoppedTasks(); } void performFaceDetectionStoppedTasks() { leftMetricsLayout.animate().alpha(0); //make left and right metrics disappear rightMetricsLayout.animate().alpha(0); resetFPSCalculations(); //Since the FPS may be different whether a face is being tracked or not, reset variables. } /** * This event is received every time the SDK processes a frame. */ @Override public void onImageResults(List faces, Frame image, float timeStamp) { mostRecentFrame = image; //If the faces object is null, we received an unprocessed frame if (faces == null) { return; } //At this point, we know the frame received was processed, so we perform our processed frames per second calculations performFPSCalculations(); //If faces.size() is 0, we received a frame in which no face was detected if (faces.size() <= 0) { drawingView.invalidatePoints(); } else if (faces.size() == 1) { metricViewLayout.setVisibility(View.VISIBLE); //update metrics with latest face information. The metrics are displayed on a MetricView, a custom view with a .setScore() method. int i = 0; for (TextView metricDisplay : metricDisplays) { float score = 0; switch (i){ case 4: score = faces.get(0).emotions.getAnger(); break; case 5: score = faces.get(0).emotions.getContempt(); break; case 6: score = faces.get(0).emotions.getDisgust(); break; case 0: score = faces.get(0).emotions.getFear(); break; case 1: score = faces.get(0).emotions.getJoy(); break; case 2: score = faces.get(0).emotions.getSadness(); break; case 3: score = faces.get(0).emotions.getSurprise(); break; } i++; metricDisplay.setText(String.format(Locale.US, "%.0f%%", score)); // updateMetricScore(metricDisplay, faces.get(0)); } /** * If the user has selected to have any facial attributes drawn, we use face.getFacePoints() to send those points * to our drawing thread and also inform the thread what the valence score was, as that will determine the color * of the bounding box. */ if (drawingView.getDrawPointsEnabled() || drawingView.getDrawAppearanceMarkersEnabled() || drawingView.getDrawEmojiMarkersEnabled()) { drawingView.updatePoints(faces, mirrorPoints); } } else { // metrics overlay is hidden in multi face mode metricViewLayout.setVisibility(View.GONE); // always update points in multi face mode drawingView.updatePoints(faces, mirrorPoints); } } public void takeScreenshot(View view) { // Check the permissions to see if we are allowed to save the screenshot if (!storagePermissionsAvailable) { checkForStoragePermissions(); return; } drawingView.requestBitmap(); /** * A screenshot of the drawing view is generated and processing continues via the callback * onBitmapGenerated() which calls processScreenshot(). */ } private void processScreenshot(Bitmap drawingViewBitmap, boolean alsoSaveRaw) { if (mostRecentFrame == null) { Toast.makeText(getApplicationContext(), "No frame detected, aborting screenshot", Toast.LENGTH_SHORT).show(); return; } if (!storagePermissionsAvailable) { checkForStoragePermissions(); return; } Bitmap faceBitmap = ImageHelper.getBitmapFromFrame(mostRecentFrame); if (faceBitmap == null) { Log.e(LOG_TAG, "Unable to generate bitmap for frame, aborting screenshot"); return; } metricViewLayout.setDrawingCacheEnabled(true); Bitmap metricsBitmap = Bitmap.createBitmap(metricViewLayout.getDrawingCache()); metricViewLayout.setDrawingCacheEnabled(false); Bitmap finalScreenshot = Bitmap.createBitmap(faceBitmap.getWidth(), faceBitmap.getHeight(), Bitmap.Config.ARGB_8888); Canvas canvas = new Canvas(finalScreenshot); Paint paint = new Paint(Paint.FILTER_BITMAP_FLAG); canvas.drawBitmap(faceBitmap, 0, 0, paint); float scaleFactor = ((float) faceBitmap.getWidth()) / ((float) drawingViewBitmap.getWidth()); int scaledHeight = Math.round(drawingViewBitmap.getHeight() * scaleFactor); canvas.drawBitmap(drawingViewBitmap, null, new Rect(0, 0, faceBitmap.getWidth(), scaledHeight), paint); scaleFactor = ((float) faceBitmap.getWidth()) / ((float) metricsBitmap.getWidth()); scaledHeight = Math.round(metricsBitmap.getHeight() * scaleFactor); canvas.drawBitmap(metricsBitmap, null, new Rect(0, 0, faceBitmap.getWidth(), scaledHeight), paint); metricsBitmap.recycle(); drawingViewBitmap.recycle(); Date now = new Date(); String timestamp = DateFormat.format("yyyy-MM-dd_hh-mm-ss", now).toString(); File pictureFolder = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES), "AffdexMe"); if (!pictureFolder.exists()) { if (!pictureFolder.mkdir()) { Log.e(LOG_TAG, "Unable to create directory: " + pictureFolder.getAbsolutePath()); return; } } String screenshotFileName = timestamp + ".png"; File screenshotFile = new File(pictureFolder, screenshotFileName); try { ImageHelper.saveBitmapToFileAsPng(finalScreenshot, screenshotFile); } catch (IOException e) { String msg = "Unable to save screenshot"; Toast.makeText(getApplicationContext(), msg, Toast.LENGTH_SHORT).show(); Log.e(LOG_TAG, msg, e); return; } ImageHelper.addPngToGallery(getApplicationContext(), screenshotFile); if (alsoSaveRaw) { String rawScreenshotFileName = timestamp + "_raw.png"; File rawScreenshotFile = new File(pictureFolder, rawScreenshotFileName); try { ImageHelper.saveBitmapToFileAsPng(faceBitmap, rawScreenshotFile); } catch (IOException e) { String msg = "Unable to save screenshot"; Log.e(LOG_TAG, msg, e); } ImageHelper.addPngToGallery(getApplicationContext(), rawScreenshotFile); } faceBitmap.recycle(); finalScreenshot.recycle(); String fileSavedMessage = "Screenshot saved to: " + screenshotFile.getPath(); Toast.makeText(getApplicationContext(), fileSavedMessage, Toast.LENGTH_SHORT).show(); Log.d(LOG_TAG, fileSavedMessage); } /** * Use the method that we saved in activateMetric() to get the metric score and display it */ void updateMetricScore(MetricDisplay metricDisplay, Face face) { MetricsManager.Metrics metric = metricDisplay.getMetricToDisplay(); float score = Float.NaN; try { switch (metric.getType()) { case Emotion: score = (Float) metricDisplay.getFaceScoreMethod().invoke(face.emotions); break; case Expression: score = (Float) metricDisplay.getFaceScoreMethod().invoke(face.expressions); break; case Emoji: score = (Float) metricDisplay.getFaceScoreMethod().invoke(face.emojis); break; default: throw new Exception("Unknown Metric Type: " + metric.getType().toString()); } } catch (Exception e) { Log.e(LOG_TAG, String.format("Error using reflecting to get %s score from face.", metric.toString())); } metricDisplay.setScore(score); } /** * FPS measurement simply uses SystemClock to measure how many frames were processed since * the FPS variables were last reset. * The constants 1000L and 1000f appear because .elapsedRealtime() measures time in milliseconds. * Note that if 20 frames per second are processed, this method could run for 1.5 years without being reset * before numberOfFrames overflows. */ void performFPSCalculations() { return; // numberOfFrames += 1; // long currentTime = SystemClock.elapsedRealtime(); // if (currentTime > timeToUpdate) { // float framesPerSecond = (numberOfFrames / (float) (currentTime - firstSystemTime)) * 1000f; // fpsPct.setText(String.format(" %.1f", framesPerSecond)); // timeToUpdate = currentTime + 1000L; // } } /** * Although we start the camera in onWindowFocusChanged(), we stop it in onPause(), and set detector to be null so that when onWindowFocusChanged() * is called it restarts the camera. We also set the Progress Bar to be visible, so the camera (which may need resizing when the app * is resumed) is obscured. */ @Override public void onPause() { super.onPause(); progressBarLayout.setVisibility(View.VISIBLE); ActivityManager activityManager = (ActivityManager) getApplicationContext() .getSystemService(Context.ACTIVITY_SERVICE); activityManager.moveTaskToFront(getTaskId(), 0); performFaceDetectionStoppedTasks(); stopDetector(); } void stopDetector() { if (detector.isRunning()) { try { detector.stop(); } catch (Exception e) { Log.e(LOG_TAG, e.getMessage()); } } detector.setDetectAllEmotions(false); detector.setDetectAllExpressions(false); detector.setDetectAllAppearances(false); detector.setDetectAllEmojis(false); } /** * When the user taps the screen, hide the menu if it is visible and show it if it is hidden. **/ void setMenuVisible(boolean b) { //We hide the navigation bar getWindow().getDecorView().setSystemUiVisibility( View.SYSTEM_UI_FLAG_LAYOUT_STABLE | View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN | View.SYSTEM_UI_FLAG_HIDE_NAVIGATION | View.SYSTEM_UI_FLAG_FULLSCREEN); } /** * If a user has a phone with a physical menu button, they may expect it to toggle * the menu, so we add that functionality. */ @Override public boolean onKeyDown(int keyCode, KeyEvent event) { if (keyCode == KeyEvent.KEYCODE_MENU) { setMenuVisible(!isMenuVisible); return true; } return super.onKeyDown(keyCode, event); } //If the user selects to have facial tracking dots drawn, inform our drawing thread. void setTrackPoints(boolean b) { drawingView.setDrawPointsEnabled(b); } void setShowAppearance(boolean b) { drawingView.setDrawAppearanceMarkersEnabled(b); } void setShowEmoji(boolean b) { drawingView.setDrawEmojiMarkersEnabled(b); } void setFPSVisible(boolean b) { return; // isFPSVisible = b; // if (b) { // fpsName.setVisibility(View.VISIBLE); // fpsPct.setVisibility(View.VISIBLE); // } else { // fpsName.setVisibility(View.INVISIBLE); // fpsPct.setVisibility(View.INVISIBLE); // } } @Override public boolean onTouch(View v, MotionEvent event) { if (event.getAction() == MotionEvent.ACTION_DOWN) { setMenuVisible(!isMenuVisible); } return false; } @SuppressWarnings("SuspiciousNameCombination") @Override public void onCameraSizeSelected(int cameraWidth, int cameraHeight, ROTATE rotation) { if (rotation == ROTATE.BY_90_CCW || rotation == ROTATE.BY_90_CW) { cameraPreviewWidth = cameraHeight; cameraPreviewHeight = cameraWidth; } else { cameraPreviewWidth = cameraWidth; cameraPreviewHeight = cameraHeight; } drawingView.setThickness((int) (cameraPreviewWidth / 100f)); mainLayout.post(new Runnable() { @Override public void run() { //Get the screen width and height, and calculate the new app width/height based on the surfaceview aspect ratio. DisplayMetrics displaymetrics = new DisplayMetrics(); getWindowManager().getDefaultDisplay().getMetrics(displaymetrics); int layoutWidth = displaymetrics.widthPixels; int layoutHeight = displaymetrics.heightPixels; if (cameraPreviewWidth == 0 || cameraPreviewHeight == 0 || layoutWidth == 0 || layoutHeight == 0) return; float layoutAspectRatio = (float) layoutWidth / layoutHeight; float cameraPreviewAspectRatio = (float) cameraPreviewWidth / cameraPreviewHeight; int newWidth; int newHeight; if (cameraPreviewAspectRatio > layoutAspectRatio) { newWidth = layoutWidth; newHeight = (int) (layoutWidth / cameraPreviewAspectRatio); } else { newWidth = (int) (layoutHeight * cameraPreviewAspectRatio); newHeight = layoutHeight; } drawingView.updateViewDimensions(newWidth, newHeight, cameraPreviewWidth, cameraPreviewHeight); ViewGroup.LayoutParams params = mainLayout.getLayoutParams(); params.height = newHeight; params.width = newWidth; mainLayout.setLayoutParams(params); //Now that our main layout has been resized, we can remove the progress bar that was obscuring the screen (its purpose was to obscure the resizing of the SurfaceView) progressBarLayout.setVisibility(View.GONE); } }); } private void setCameraType(CameraDetector.CameraType type) { SharedPreferences.Editor preferencesEditor = PreferenceManager.getDefaultSharedPreferences(this).edit(); //If a settings change is necessary if (cameraType != type) { switch (type) { case CAMERA_BACK: if (isBackFacingCameraDetected) { cameraType = CameraDetector.CameraType.CAMERA_BACK; mirrorPoints = false; } else { Toast.makeText(this, "No back-facing camera found", Toast.LENGTH_LONG).show(); return; } break; case CAMERA_FRONT: if (isFrontFacingCameraDetected) { cameraType = CameraDetector.CameraType.CAMERA_FRONT; mirrorPoints = true; } else { Toast.makeText(this, "No front-facing camera found", Toast.LENGTH_LONG).show(); return; } break; default: Log.e(LOG_TAG, "Unknown camera type selected"); } performFaceDetectionStoppedTasks(); detector.setCameraType(cameraType); preferencesEditor.putString("cameraType", cameraType.name()); preferencesEditor.apply(); } } @Override public void onBitmapGenerated(@NonNull final Bitmap bitmap) { runOnUiThread(new Runnable() { @Override public void run() { processScreenshot(bitmap, STORE_RAW_SCREENSHOTS); } }); } @Override public void onBackPressed() { // nothing to do here // … really } }