Score bar, level introductions and new interactive menu
This commit is contained in:
parent
cecd9e20f6
commit
d711a93fbf
11 changed files with 612 additions and 40 deletions
|
@ -37,7 +37,7 @@
|
||||||
<ConfirmationsSetting value="0" id="Add" />
|
<ConfirmationsSetting value="0" id="Add" />
|
||||||
<ConfirmationsSetting value="0" id="Remove" />
|
<ConfirmationsSetting value="0" id="Remove" />
|
||||||
</component>
|
</component>
|
||||||
<component name="ProjectRootManager" version="2" languageLevel="JDK_1_8" default="true" assert-keyword="true" jdk-15="true" project-jdk-name="1.8" project-jdk-type="JavaSDK">
|
<component name="ProjectRootManager" version="2" languageLevel="JDK_1_7" default="true" assert-keyword="true" jdk-15="true" project-jdk-name="1.8" project-jdk-type="JavaSDK">
|
||||||
<output url="file://$PROJECT_DIR$/build/classes" />
|
<output url="file://$PROJECT_DIR$/build/classes" />
|
||||||
</component>
|
</component>
|
||||||
<component name="ProjectType">
|
<component name="ProjectType">
|
||||||
|
|
|
@ -51,9 +51,10 @@
|
||||||
<activity
|
<activity
|
||||||
android:name=".CreditsActivity"
|
android:name=".CreditsActivity"
|
||||||
android:theme="@style/FullscreenTheme" />
|
android:theme="@style/FullscreenTheme" />
|
||||||
|
<activity
|
||||||
<activity android:name=".ReviewActivity"
|
android:name=".ReviewActivity"
|
||||||
android:theme="@style/AppTheme.NoActionBar"></activity>
|
android:theme="@style/AppTheme.NoActionBar" />
|
||||||
|
<activity android:name=".MirrorMenuActivity"></activity>
|
||||||
</application>
|
</application>
|
||||||
|
|
||||||
</manifest>
|
</manifest>
|
|
@ -2,6 +2,8 @@ package com.rubenvandeven.emotionhero;
|
||||||
|
|
||||||
import android.Manifest;
|
import android.Manifest;
|
||||||
import android.annotation.TargetApi;
|
import android.annotation.TargetApi;
|
||||||
|
import android.app.Dialog;
|
||||||
|
import android.content.DialogInterface;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.pm.PackageManager;
|
import android.content.pm.PackageManager;
|
||||||
import android.media.AudioAttributes;
|
import android.media.AudioAttributes;
|
||||||
|
@ -9,6 +11,8 @@ import android.media.AudioManager;
|
||||||
import android.media.SoundPool;
|
import android.media.SoundPool;
|
||||||
import android.os.Build;
|
import android.os.Build;
|
||||||
import android.support.v4.app.ActivityCompat;
|
import android.support.v4.app.ActivityCompat;
|
||||||
|
import android.support.v4.app.DialogFragment;
|
||||||
|
import android.support.v7.app.AlertDialog;
|
||||||
import android.support.v7.app.AppCompatActivity;
|
import android.support.v7.app.AppCompatActivity;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
@ -63,6 +67,8 @@ public class GamingActivity extends AppCompatActivity implements Detector.ImageL
|
||||||
|
|
||||||
protected Player player;
|
protected Player player;
|
||||||
|
|
||||||
|
protected boolean isStarted = false;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onCreate(Bundle savedInstanceState) {
|
protected void onCreate(Bundle savedInstanceState) {
|
||||||
super.onCreate(savedInstanceState);
|
super.onCreate(savedInstanceState);
|
||||||
|
@ -88,6 +94,8 @@ public class GamingActivity extends AppCompatActivity implements Detector.ImageL
|
||||||
public void onMeasure(int widthSpec, int heightSpec) {
|
public void onMeasure(int widthSpec, int heightSpec) {
|
||||||
int measureWidth = MeasureSpec.getSize(widthSpec);
|
int measureWidth = MeasureSpec.getSize(widthSpec);
|
||||||
int measureHeight = MeasureSpec.getSize(heightSpec);
|
int measureHeight = MeasureSpec.getSize(heightSpec);
|
||||||
|
int maxWidth = GamingActivity.this.getWindow().getDecorView().getWidth();
|
||||||
|
int maxHeight = GamingActivity.this.getWindow().getDecorView().getHeight();
|
||||||
int width;
|
int width;
|
||||||
int height;
|
int height;
|
||||||
if (previewHeight == 0 || previewWidth == 0) {
|
if (previewHeight == 0 || previewWidth == 0) {
|
||||||
|
@ -105,6 +113,18 @@ public class GamingActivity extends AppCompatActivity implements Detector.ImageL
|
||||||
height = measureHeight;
|
height = measureHeight;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// make sure we fill the screen!
|
||||||
|
if(width < maxWidth) {
|
||||||
|
float ratio = (float)maxWidth/width;
|
||||||
|
width = maxWidth;
|
||||||
|
height = (int) (ratio * height);
|
||||||
|
}
|
||||||
|
if(height < maxHeight) {
|
||||||
|
float ratio = (float)maxHeight/height;
|
||||||
|
width = (int) (ratio * width);
|
||||||
|
height = maxHeight;
|
||||||
|
}
|
||||||
setMeasuredDimension(width,height);
|
setMeasuredDimension(width,height);
|
||||||
// setMeasuredDimension(1,1); // this DOES increase performance....
|
// setMeasuredDimension(1,1); // this DOES increase performance....
|
||||||
}
|
}
|
||||||
|
@ -113,8 +133,8 @@ public class GamingActivity extends AppCompatActivity implements Detector.ImageL
|
||||||
// RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(1,1);
|
// RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(1,1);
|
||||||
params.addRule(RelativeLayout.CENTER_IN_PARENT,RelativeLayout.TRUE);
|
params.addRule(RelativeLayout.CENTER_IN_PARENT,RelativeLayout.TRUE);
|
||||||
// 10% margin:
|
// 10% margin:
|
||||||
params.leftMargin = (int) 120;
|
// params.leftMargin = (int) 120;
|
||||||
params.rightMargin = params.leftMargin;
|
// params.rightMargin = params.leftMargin;
|
||||||
cameraPreview.setLayoutParams(params);
|
cameraPreview.setLayoutParams(params);
|
||||||
videoLayout.addView(cameraPreview,0);
|
videoLayout.addView(cameraPreview,0);
|
||||||
|
|
||||||
|
@ -125,18 +145,28 @@ public class GamingActivity extends AppCompatActivity implements Detector.ImageL
|
||||||
currentScenario.init(); // "start the clock"...
|
currentScenario.init(); // "start the clock"...
|
||||||
|
|
||||||
scenarioView = new ScenarioView(this, currentScenario);
|
scenarioView = new ScenarioView(this, currentScenario);
|
||||||
|
scenarioView.drawOverlay = true;
|
||||||
RelativeLayout.LayoutParams scenarioViewParams = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
|
RelativeLayout.LayoutParams scenarioViewParams = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
|
||||||
videoLayout.addView(scenarioView, 1, scenarioViewParams);
|
videoLayout.addView(scenarioView, 1, scenarioViewParams);
|
||||||
|
|
||||||
createSoundPool(); // instantiate SoundPool in sound
|
createSoundPool(); // instantiate SoundPool in sound
|
||||||
soundIds.put(SOUND_SCORE, sound.load(this, R.raw.score2, 1));
|
soundIds.put(SOUND_SCORE, sound.load(this, R.raw.score2, 1));
|
||||||
|
|
||||||
|
StoryDialogFragment storyDialog = new StoryDialogFragment();
|
||||||
|
storyDialog.show(getSupportFragmentManager(), "StoryDialog");
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void startGame() {
|
||||||
|
isStarted = true;
|
||||||
|
startDetector();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onResume() {
|
protected void onResume() {
|
||||||
super.onResume();
|
super.onResume();
|
||||||
startDetector();
|
if(isStarted)
|
||||||
|
startDetector();
|
||||||
// should resume 'on face detection started'
|
// should resume 'on face detection started'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -172,10 +202,10 @@ public class GamingActivity extends AppCompatActivity implements Detector.ImageL
|
||||||
detector.setLicensePath("emotionhero_dev.license");
|
detector.setLicensePath("emotionhero_dev.license");
|
||||||
|
|
||||||
detector.setDetectAllEmotions(true);
|
detector.setDetectAllEmotions(true);
|
||||||
detector.setDetectAllAppearances(false);
|
detector.setDetectAllAppearances(true);
|
||||||
detector.setDetectAllEmojis(false);
|
detector.setDetectAllEmojis(false);
|
||||||
detector.setDetectAllExpressions(true);
|
detector.setDetectAllExpressions(true);
|
||||||
detector.setMaxProcessRate(20);
|
detector.setMaxProcessRate(12);
|
||||||
|
|
||||||
detector.setImageListener(this);
|
detector.setImageListener(this);
|
||||||
detector.setOnCameraEventListener(this);
|
detector.setOnCameraEventListener(this);
|
||||||
|
@ -264,7 +294,9 @@ public class GamingActivity extends AppCompatActivity implements Detector.ImageL
|
||||||
public void onFaceDetectionStarted()
|
public void onFaceDetectionStarted()
|
||||||
{
|
{
|
||||||
setText("START!");
|
setText("START!");
|
||||||
currentScenario.start();
|
|
||||||
|
if(isStarted)
|
||||||
|
currentScenario.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -89,7 +89,7 @@ public class IntroActivity extends AppCompatActivity {
|
||||||
|
|
||||||
public void continueToMenu() {
|
public void continueToMenu() {
|
||||||
continueHandler.removeCallbacks(continueRunnable);
|
continueHandler.removeCallbacks(continueRunnable);
|
||||||
Intent intent = new Intent(this, MenuActivity.class);
|
Intent intent = new Intent(this, MirrorMenuActivity.class);
|
||||||
finish();
|
finish();
|
||||||
startActivity(intent);
|
startActivity(intent);
|
||||||
//disable distracting transition when going to menu
|
//disable distracting transition when going to menu
|
||||||
|
|
|
@ -87,7 +87,7 @@ public class MenuActivity extends AppCompatActivity {
|
||||||
public void onClick(View v) {
|
public void onClick(View v) {
|
||||||
loadGameDialog = ProgressDialog.show(MenuActivity.this, "",
|
loadGameDialog = ProgressDialog.show(MenuActivity.this, "",
|
||||||
MenuActivity.this.getResources().getString(R.string.load_game_activity), true);
|
MenuActivity.this.getResources().getString(R.string.load_game_activity), true);
|
||||||
Intent intent = new Intent(MenuActivity.this, GamingActivity.class);
|
Intent intent = new Intent(MenuActivity.this, MirrorMenuActivity.class);
|
||||||
startActivity(intent);
|
startActivity(intent);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,329 @@
|
||||||
|
package com.rubenvandeven.emotionhero;
|
||||||
|
|
||||||
|
import android.Manifest;
|
||||||
|
import android.animation.ArgbEvaluator;
|
||||||
|
import android.animation.ValueAnimator;
|
||||||
|
import android.app.ProgressDialog;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.content.pm.PackageManager;
|
||||||
|
import android.graphics.Color;
|
||||||
|
import android.graphics.Typeface;
|
||||||
|
import android.support.v4.app.ActivityCompat;
|
||||||
|
import android.support.v7.app.AppCompatActivity;
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.util.Log;
|
||||||
|
import android.view.SurfaceView;
|
||||||
|
import android.view.View;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
import android.view.WindowManager;
|
||||||
|
import android.view.animation.AlphaAnimation;
|
||||||
|
import android.view.animation.Animation;
|
||||||
|
import android.widget.ImageView;
|
||||||
|
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.detector.CameraDetector;
|
||||||
|
import com.affectiva.android.affdex.sdk.detector.Detector;
|
||||||
|
import com.affectiva.android.affdex.sdk.detector.Face;
|
||||||
|
|
||||||
|
import org.w3c.dom.Text;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public class MirrorMenuActivity extends AppCompatActivity implements Detector.ImageListener, CameraDetector.CameraEventListener, Detector.FaceListener {
|
||||||
|
|
||||||
|
final static String LOG_TAG = "EmotionHero-Mirror";
|
||||||
|
|
||||||
|
private CameraDetector detector;
|
||||||
|
|
||||||
|
|
||||||
|
final int PERMISSIONS_REQUEST_CAMERA = 1;
|
||||||
|
|
||||||
|
int previewWidth = 0;
|
||||||
|
int previewHeight = 0;
|
||||||
|
|
||||||
|
SurfaceView cameraPreview;
|
||||||
|
ScenarioView scenarioView;
|
||||||
|
TextView messageText;
|
||||||
|
|
||||||
|
boolean has_camera_permission = false;
|
||||||
|
|
||||||
|
ProgressDialog loadGameDialog;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onCreate(Bundle savedInstanceState) {
|
||||||
|
super.onCreate(savedInstanceState);
|
||||||
|
getWindow().addFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN);
|
||||||
|
if(getSupportActionBar() != null) {
|
||||||
|
getSupportActionBar().hide();
|
||||||
|
}
|
||||||
|
setContentView(R.layout.activity_mirror_menu);
|
||||||
|
|
||||||
|
RelativeLayout videoContentLayout = (RelativeLayout) findViewById(R.id.videoContent);
|
||||||
|
ImageView logoEmotionHero = (ImageView) findViewById(R.id.logoEmotionHero);
|
||||||
|
messageText = (TextView) findViewById(R.id.messageText);
|
||||||
|
final TextView startButton = (TextView) findViewById(R.id.startButton);
|
||||||
|
TextView highscoresButton = (TextView) findViewById(R.id.highscoresButton);
|
||||||
|
TextView creditsButton = (TextView) findViewById(R.id.creditsButton);
|
||||||
|
|
||||||
|
Typeface font = Typeface.createFromAsset(getAssets(), "unifont-9.0.02.ttf");
|
||||||
|
startButton .setTypeface(font);
|
||||||
|
highscoresButton .setTypeface(font);
|
||||||
|
creditsButton .setTypeface(font);
|
||||||
|
|
||||||
|
//We create a custom SurfaceView that resizes itself to match the aspect ratio of the incoming camera frames
|
||||||
|
cameraPreview = new SurfaceView(this) {
|
||||||
|
@Override
|
||||||
|
public void onMeasure(int widthSpec, int heightSpec) {
|
||||||
|
int measureWidth = MeasureSpec.getSize(widthSpec);
|
||||||
|
int measureHeight = MeasureSpec.getSize(heightSpec);
|
||||||
|
int maxWidth = MirrorMenuActivity.this.getWindow().getDecorView().getWidth();
|
||||||
|
int maxHeight = MirrorMenuActivity.this.getWindow().getDecorView().getHeight();
|
||||||
|
int width;
|
||||||
|
int height;
|
||||||
|
if (previewHeight == 0 || previewWidth == 0) {
|
||||||
|
width = measureWidth;
|
||||||
|
height = measureHeight;
|
||||||
|
} else {
|
||||||
|
float viewAspectRatio = (float)measureWidth/measureHeight;
|
||||||
|
float cameraPreviewAspectRatio = (float) previewWidth/previewHeight;
|
||||||
|
|
||||||
|
if (cameraPreviewAspectRatio > viewAspectRatio) {
|
||||||
|
width = measureWidth;
|
||||||
|
height =(int) (measureWidth / cameraPreviewAspectRatio);
|
||||||
|
} else {
|
||||||
|
width = (int) (measureHeight * cameraPreviewAspectRatio);
|
||||||
|
height = measureHeight;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// make sure we fill the screen!
|
||||||
|
if(width < maxWidth) {
|
||||||
|
float ratio = (float)maxWidth/width;
|
||||||
|
width = maxWidth;
|
||||||
|
height = (int) (ratio * height);
|
||||||
|
}
|
||||||
|
if(height < maxHeight) {
|
||||||
|
float ratio = (float)maxHeight/height;
|
||||||
|
width = (int) (ratio * width);
|
||||||
|
height = maxHeight;
|
||||||
|
}
|
||||||
|
|
||||||
|
setMeasuredDimension(width,height);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);
|
||||||
|
// RelativeLayout.LayoutParams params = new RelativeLayout.LayoutParams(1,1);
|
||||||
|
params.addRule(RelativeLayout.CENTER_IN_PARENT,RelativeLayout.TRUE);
|
||||||
|
// 10% margin:
|
||||||
|
// params.leftMargin = (int) 120;
|
||||||
|
// params.rightMargin = params.leftMargin;
|
||||||
|
cameraPreview.setLayoutParams(params);
|
||||||
|
videoContentLayout.addView(cameraPreview,0);
|
||||||
|
|
||||||
|
scenarioView = new ScenarioView(this, null);
|
||||||
|
RelativeLayout.LayoutParams scenarioViewParams = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
|
||||||
|
videoContentLayout.addView(scenarioView, 1, scenarioViewParams);
|
||||||
|
|
||||||
|
scenarioView.drawOverlay = true;
|
||||||
|
scenarioView.noFace = true;
|
||||||
|
|
||||||
|
Animation anim = new AlphaAnimation(0.0f, 1.0f);
|
||||||
|
anim.setDuration(1000); //You can manage the blinking time with this parameter
|
||||||
|
anim.setStartOffset(20);
|
||||||
|
anim.setRepeatMode(Animation.REVERSE);
|
||||||
|
anim.setRepeatCount(Animation.INFINITE);
|
||||||
|
logoEmotionHero.startAnimation(anim);
|
||||||
|
|
||||||
|
Integer colorFrom = getResources().getColor(R.color.textPrimary);
|
||||||
|
Integer colorTo = getResources().getColor(R.color.textHighlight);
|
||||||
|
ValueAnimator colorAnimation = ValueAnimator.ofObject(new ArgbEvaluator(), colorFrom, colorTo);
|
||||||
|
colorAnimation.setDuration(1000);
|
||||||
|
colorAnimation.setRepeatCount(ValueAnimator.INFINITE);
|
||||||
|
colorAnimation.setRepeatMode(ValueAnimator.REVERSE);
|
||||||
|
colorAnimation.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onAnimationUpdate(ValueAnimator animator) {
|
||||||
|
startButton.setTextColor((Integer)animator.getAnimatedValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
colorAnimation.start();
|
||||||
|
|
||||||
|
|
||||||
|
startButton.setOnClickListener(new View.OnClickListener() {
|
||||||
|
@Override
|
||||||
|
public void onClick(View v) {
|
||||||
|
loadGameDialog = ProgressDialog.show(MirrorMenuActivity.this, "",
|
||||||
|
MirrorMenuActivity.this.getResources().getString(R.string.load_game_activity), true);
|
||||||
|
Intent intent = new Intent(MirrorMenuActivity.this, GamingActivity.class);
|
||||||
|
startActivity(intent);
|
||||||
|
detector.stop();
|
||||||
|
detector = null;
|
||||||
|
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
highscoresButton.setOnClickListener(new View.OnClickListener() {
|
||||||
|
@Override
|
||||||
|
public void onClick(View v) {
|
||||||
|
Intent intent = new Intent(MirrorMenuActivity.this, HighscoreActivity.class);
|
||||||
|
// intent.putExtra(HighscoreActivity.INTENT_EXTRA_LEVEL, player.getPlayerInfo().reachedLevelId);
|
||||||
|
startActivity(intent);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
creditsButton.setOnClickListener(new View.OnClickListener() {
|
||||||
|
@Override
|
||||||
|
public void onClick(View v) {
|
||||||
|
Intent intent = new Intent(MirrorMenuActivity.this, CreditsActivity.class);
|
||||||
|
startActivity(intent);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onResume() {
|
||||||
|
super.onResume();
|
||||||
|
startDetector();
|
||||||
|
|
||||||
|
if(loadGameDialog != null)
|
||||||
|
loadGameDialog.dismiss();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onPause() {
|
||||||
|
super.onPause();
|
||||||
|
stopDetector();
|
||||||
|
}
|
||||||
|
|
||||||
|
void startDetector() {
|
||||||
|
if (detector == null || !detector.isRunning()) {
|
||||||
|
// check permission
|
||||||
|
String permission = "android.permission.CAMERA";
|
||||||
|
int res = getApplicationContext().checkCallingOrSelfPermission(permission);
|
||||||
|
if (res == PackageManager.PERMISSION_GRANTED) {
|
||||||
|
Log.i(LOG_TAG, "Has camera permission");
|
||||||
|
has_camera_permission = true;
|
||||||
|
} else {
|
||||||
|
Log.i(LOG_TAG, "No camera permission");
|
||||||
|
ActivityCompat.requestPermissions(this,
|
||||||
|
new String[]{Manifest.permission.CAMERA},
|
||||||
|
PERMISSIONS_REQUEST_CAMERA);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(has_camera_permission)
|
||||||
|
{
|
||||||
|
if(detector == null)
|
||||||
|
{
|
||||||
|
// SurfaceView surfaceView = (SurfaceView) findViewById(R.id.surfaceView);
|
||||||
|
|
||||||
|
detector = new CameraDetector(this, CameraDetector.CameraType.CAMERA_FRONT, cameraPreview, 1, Detector.FaceDetectorMode.LARGE_FACES);
|
||||||
|
detector.setLicensePath("emotionhero_dev.license");
|
||||||
|
|
||||||
|
detector.setDetectAllEmotions(true);
|
||||||
|
detector.setDetectAllAppearances(false);
|
||||||
|
detector.setDetectAllEmojis(false);
|
||||||
|
detector.setDetectAllExpressions(true);
|
||||||
|
detector.setMaxProcessRate(20);
|
||||||
|
|
||||||
|
detector.setImageListener(this);
|
||||||
|
detector.setOnCameraEventListener(this);
|
||||||
|
detector.setFaceListener(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
detector.start();
|
||||||
|
setText("STARTING...");
|
||||||
|
Log.d(LOG_TAG, Boolean.toString(detector.isRunning()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void stopDetector() {
|
||||||
|
if (detector != null && detector.isRunning()) {
|
||||||
|
detector.stop();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
/**
|
||||||
|
* Detector callback gives the faces found so we can match their hits to the given scenario.
|
||||||
|
*/
|
||||||
|
public void onImageResults(List<Face> list, Frame frame, float timestamp) {if (list == null)
|
||||||
|
return;
|
||||||
|
if (list.size() == 0) {
|
||||||
|
setText("Show me your face");
|
||||||
|
} else {
|
||||||
|
hideText();
|
||||||
|
Face face = list.get(0);
|
||||||
|
scenarioView.setCurrentAttributeScoresForFace(face);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
/**
|
||||||
|
* For CameraDetector.CameraEventListener
|
||||||
|
* Used to scale video preview output
|
||||||
|
*/
|
||||||
|
public void onCameraSizeSelected(int width, int height, Frame.ROTATE rotate) {
|
||||||
|
if (rotate == Frame.ROTATE.BY_90_CCW || rotate == Frame.ROTATE.BY_90_CW) {
|
||||||
|
previewWidth = height;
|
||||||
|
previewHeight = width;
|
||||||
|
} else {
|
||||||
|
previewHeight = height;
|
||||||
|
previewWidth = width;
|
||||||
|
}
|
||||||
|
cameraPreview.requestLayout();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onRequestPermissionsResult(int requestCode,
|
||||||
|
String permissions[], int[] grantResults) {
|
||||||
|
switch (requestCode) {
|
||||||
|
case PERMISSIONS_REQUEST_CAMERA: {
|
||||||
|
// If request is cancelled, the result arrays are empty.
|
||||||
|
if (grantResults.length > 0
|
||||||
|
&& grantResults[0] == PackageManager.PERMISSION_GRANTED) {
|
||||||
|
has_camera_permission = true;
|
||||||
|
startDetector();
|
||||||
|
} else {
|
||||||
|
has_camera_permission = false;
|
||||||
|
Toast errorMsg = Toast.makeText(this, R.string.camera_required, Toast.LENGTH_LONG);
|
||||||
|
errorMsg.show();
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// other 'case' lines to check for other
|
||||||
|
// permissions this app might request
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onFaceDetectionStarted()
|
||||||
|
{
|
||||||
|
hideText();
|
||||||
|
scenarioView.noFace = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onFaceDetectionStopped()
|
||||||
|
{
|
||||||
|
scenarioView.noFace = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setText(String text)
|
||||||
|
{
|
||||||
|
messageText.setVisibility(View.VISIBLE);
|
||||||
|
messageText.setText(text);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void hideText()
|
||||||
|
{
|
||||||
|
messageText.setVisibility(View.GONE);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -70,6 +70,8 @@ public class Scenario {
|
||||||
|
|
||||||
private GamingActivity _activity;
|
private GamingActivity _activity;
|
||||||
|
|
||||||
|
int maxScore = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The scorres in this moment, as to draw them on the screen.
|
* The scorres in this moment, as to draw them on the screen.
|
||||||
* Indexes are Emotion ordinals
|
* Indexes are Emotion ordinals
|
||||||
|
@ -177,6 +179,7 @@ public class Scenario {
|
||||||
target.emotion = emotion;
|
target.emotion = emotion;
|
||||||
target.index = targets.size() + 1;
|
target.index = targets.size() + 1;
|
||||||
targets.add(target);
|
targets.add(target);
|
||||||
|
maxScore = getMaxScore();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -205,6 +208,23 @@ public class Scenario {
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public float getHitTotalMisValue()
|
||||||
|
{
|
||||||
|
float value = 0;
|
||||||
|
for (Hit hit : game.hits.values()) {
|
||||||
|
value += 100-hit.score;
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public float getHitPercentage() {
|
||||||
|
return (getHitTotalValue()/maxScore) * 100;
|
||||||
|
}
|
||||||
|
|
||||||
|
public float getMissedPercentage() {
|
||||||
|
return (getHitTotalMisValue()/maxScore) * 100;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check whether given timestamp is within duration of the scenario
|
* Check whether given timestamp is within duration of the scenario
|
||||||
* @param timestamp
|
* @param timestamp
|
||||||
|
|
|
@ -8,6 +8,7 @@ import android.graphics.Path;
|
||||||
import android.graphics.PixelFormat;
|
import android.graphics.PixelFormat;
|
||||||
import android.graphics.Rect;
|
import android.graphics.Rect;
|
||||||
import android.graphics.Typeface;
|
import android.graphics.Typeface;
|
||||||
|
import android.util.Log;
|
||||||
import android.view.SurfaceHolder;
|
import android.view.SurfaceHolder;
|
||||||
import android.view.SurfaceView;
|
import android.view.SurfaceView;
|
||||||
|
|
||||||
|
@ -38,10 +39,15 @@ public class ScenarioView extends SurfaceView implements SurfaceHolder.Callback
|
||||||
private Map<Emotion, Paint> emoScoredPaints = new HashMap<>();
|
private Map<Emotion, Paint> emoScoredPaints = new HashMap<>();
|
||||||
private Paint mainPaint = new Paint();
|
private Paint mainPaint = new Paint();
|
||||||
private Paint attrScorePaint = new Paint();
|
private Paint attrScorePaint = new Paint();
|
||||||
|
private Paint attrScoreLinePaint = new Paint();
|
||||||
private Paint linePaint = new Paint();
|
private Paint linePaint = new Paint();
|
||||||
private Paint scorePaint = new Paint();
|
private Paint scorePaint = new Paint();
|
||||||
private Paint scoreFinishedPaint = new Paint();
|
private Paint scoreFinishedPaint = new Paint();
|
||||||
|
|
||||||
|
public boolean drawOverlay = false;
|
||||||
|
|
||||||
|
public boolean noFace = false;
|
||||||
|
|
||||||
// see: http://blog.danielnadeau.io/2012/01/android-canvas-beginners-tutorial.html
|
// see: http://blog.danielnadeau.io/2012/01/android-canvas-beginners-tutorial.html
|
||||||
class PanelThread extends Thread {
|
class PanelThread extends Thread {
|
||||||
private SurfaceHolder _surfaceHolder;
|
private SurfaceHolder _surfaceHolder;
|
||||||
|
@ -103,6 +109,9 @@ public class ScenarioView extends SurfaceView implements SurfaceHolder.Callback
|
||||||
linePaint.setStrokeWidth(5);
|
linePaint.setStrokeWidth(5);
|
||||||
attrScorePaint.setColor(Color.DKGRAY);
|
attrScorePaint.setColor(Color.DKGRAY);
|
||||||
|
|
||||||
|
attrScoreLinePaint.setColor(Color.DKGRAY);
|
||||||
|
attrScoreLinePaint.setStrokeWidth(2);
|
||||||
|
|
||||||
for(Emotion emotion: Emotion.values()) {
|
for(Emotion emotion: Emotion.values()) {
|
||||||
Paint emoPaint = new Paint();
|
Paint emoPaint = new Paint();
|
||||||
emoPaint.setTextSize(22);
|
emoPaint.setTextSize(22);
|
||||||
|
@ -135,9 +144,14 @@ public class ScenarioView extends SurfaceView implements SurfaceHolder.Callback
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onDraw(Canvas canvas) {
|
public void onDraw(Canvas canvas) {
|
||||||
// canvas.drawColor(0x00AAAAAA);
|
if(drawOverlay) {
|
||||||
|
canvas.drawColor(0x990000FF);
|
||||||
|
}
|
||||||
//do drawing stuff here.
|
//do drawing stuff here.
|
||||||
|
|
||||||
|
if (noFace)
|
||||||
|
return;
|
||||||
|
|
||||||
float height = canvas.getHeight();
|
float height = canvas.getHeight();
|
||||||
float width = canvas.getWidth();
|
float width = canvas.getWidth();
|
||||||
|
|
||||||
|
@ -187,42 +201,62 @@ public class ScenarioView extends SurfaceView implements SurfaceHolder.Callback
|
||||||
// Draw targets:
|
// Draw targets:
|
||||||
float sec_height = height * (float) 0.2; // each second is 20% of canvas height
|
float sec_height = height * (float) 0.2; // each second is 20% of canvas height
|
||||||
// current moved position
|
// current moved position
|
||||||
float diff_y = sec_height * _scenario.getTime();
|
if(_scenario != null) {
|
||||||
|
|
||||||
for(Scenario.Target target: _scenario.getTargets()) {
|
float diff_y = sec_height * _scenario.getTime();
|
||||||
// Paint targetPaint = (target.hit == null) ? emoPaints.get(target.emotion) : emoScoredPaints.get(target.emotion);
|
|
||||||
float cy = bottomline_height - (target.timestamp * sec_height) + diff_y;
|
|
||||||
|
|
||||||
float cx = padding_left + (step_y * target.emotion.ordinal() + step_y / 2);
|
for(Scenario.Target target: _scenario.getTargets()) {
|
||||||
|
|
||||||
canvas.drawCircle(cx, cy, max_ball_radius * target.value/100, emoPaints.get(target.emotion));
|
// don't draw hit targets
|
||||||
|
if(target.isHit){
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if(target.isHit) {
|
// Paint targetPaint = (target.hit == null) ? emoPaints.get(target.emotion) : emoScoredPaints.get(target.emotion);
|
||||||
Hit hit = _scenario.getHitForTarget(target);
|
float cy = bottomline_height - (target.timestamp * sec_height) + diff_y;
|
||||||
canvas.drawText(Integer.toString(Math.round(hit.score)), cx, cy, emoPaints.get(target.emotion));
|
|
||||||
canvas.drawCircle(cx, cy, max_ball_radius * hit.score/100, emoScoredPaints.get(target.emotion));
|
|
||||||
|
|
||||||
// TODO: Use target.hit.face to set Rect src.
|
float cx = padding_left + (step_y * target.emotion.ordinal() + step_y / 2);
|
||||||
// if(target.hit.image != null) {
|
|
||||||
// Rect rect = new Rect((int) cx-10, (int) cy-10, (int) cx+10, (int) cy+10);
|
canvas.drawCircle(cx, cy, max_ball_radius * target.value/100, emoPaints.get(target.emotion));
|
||||||
// canvas.drawBitmap(target.hit.image, null, rect, null);
|
|
||||||
// }
|
if(target.isHit) {
|
||||||
|
Hit hit = _scenario.getHitForTarget(target);
|
||||||
|
canvas.drawText(Integer.toString(Math.round(hit.score)), cx, cy, emoPaints.get(target.emotion));
|
||||||
|
canvas.drawCircle(cx, cy, max_ball_radius * hit.score/100, emoScoredPaints.get(target.emotion));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// String target_text = target.emotion.toString() + " " + target.value + "%";
|
||||||
|
// canvas.drawText(target_text, cx, y_pos + diff_y , emoPaint);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// draw the hit middle bottom;
|
||||||
|
// Paint usedScorePaint = _scenario.isFinished() ? scoreFinishedPaint : scorePaint;
|
||||||
|
// String scoreText = String.format("Total: %1$.0f", _scenario.getHitTotalValue());
|
||||||
|
// Rect scoreTextBounds = new Rect();
|
||||||
|
// usedScorePaint.getTextBounds(scoreText, 0, scoreText.length(), scoreTextBounds);
|
||||||
|
|
||||||
|
// canvas.drawText(scoreText, (width - scoreTextBounds.width()) / 2, height - 10, usedScorePaint);
|
||||||
|
|
||||||
|
|
||||||
|
float hitWidth = width*(_scenario.getHitPercentage()/100);
|
||||||
|
float misWidth = width*(_scenario.getMissedPercentage()/100);
|
||||||
|
|
||||||
|
String scoreText = String.format("%1$.0f", _scenario.getHitTotalValue());
|
||||||
|
Rect scoreTextBounds = new Rect();
|
||||||
|
scorePaint.getTextBounds(scoreText, 0, scoreText.length(), scoreTextBounds);
|
||||||
|
canvas.drawText(scoreText, hitWidth, height*0.95f-scoreTextBounds.height()-5, scorePaint);
|
||||||
|
|
||||||
|
canvas.drawRect(0, height*0.95f, width, height, mainPaint);
|
||||||
|
float stepSize = width / _scenario.targets.size();
|
||||||
|
for(int i = _scenario.targets.size(); i > 0; i--) {
|
||||||
|
canvas.drawLine(stepSize*i, height*0.95f, stepSize*i, height, attrScoreLinePaint);
|
||||||
|
}
|
||||||
|
canvas.drawRect(0, height*0.95f, hitWidth, height, emoPaints.get(Emotion.JOY));
|
||||||
|
canvas.drawRect(hitWidth, height*0.95f, hitWidth+misWidth, height, attrScorePaint);
|
||||||
|
|
||||||
// String target_text = target.emotion.toString() + " " + target.value + "%";
|
|
||||||
// canvas.drawText(target_text, cx, y_pos + diff_y , emoPaint);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// draw the hit middle bottom;
|
|
||||||
Paint usedScorePaint = _scenario.isFinished() ? scoreFinishedPaint : scorePaint;
|
|
||||||
String scoreText = String.format("Total: %1$.0f", _scenario.getHitTotalValue());
|
|
||||||
Rect scoreTextBounds = new Rect();
|
|
||||||
usedScorePaint.getTextBounds(scoreText, 0, scoreText.length(), scoreTextBounds);
|
|
||||||
|
|
||||||
canvas.drawText(scoreText, (width - scoreTextBounds.width()) / 2, height - 10, usedScorePaint);
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,54 @@
|
||||||
|
package com.rubenvandeven.emotionhero;
|
||||||
|
|
||||||
|
import android.annotation.SuppressLint;
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.app.Dialog;
|
||||||
|
import android.content.DialogInterface;
|
||||||
|
import android.os.Bundle;
|
||||||
|
import android.support.v7.app.AlertDialog;
|
||||||
|
import android.support.v7.app.AppCompatDialogFragment;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by ruben on 07/09/16.
|
||||||
|
*/
|
||||||
|
|
||||||
|
public class StoryDialogFragment extends AppCompatDialogFragment {
|
||||||
|
|
||||||
|
GamingActivity activity;
|
||||||
|
|
||||||
|
// Override the Fragment.onAttach() method to instantiate the NoticeDialogListener
|
||||||
|
@Override
|
||||||
|
public void onAttach(Activity activity) {
|
||||||
|
super.onAttach(activity);
|
||||||
|
// Verify that the host activity implements the callback interface
|
||||||
|
try {
|
||||||
|
// Instantiate the NoticeDialogListener so we can send events to the host
|
||||||
|
this.activity = (GamingActivity) activity;
|
||||||
|
} catch (ClassCastException e) {
|
||||||
|
// The activity doesn't implement the interface, throw exception
|
||||||
|
throw new ClassCastException(activity.toString()
|
||||||
|
+ " must implement StoryDialogFragmentListener");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Dialog onCreateDialog(Bundle savedInstanceState) {
|
||||||
|
|
||||||
|
// Use the Builder class for convenient dialog construction
|
||||||
|
AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
|
||||||
|
builder.setMessage(activity.currentScenario.getDescription()).setTitle(activity.currentScenario.toString())
|
||||||
|
.setNeutralButton("Start!", new DialogInterface.OnClickListener() {
|
||||||
|
public void onClick(DialogInterface dialog, int id) {
|
||||||
|
activity.startGame();
|
||||||
|
}
|
||||||
|
}).setCancelable(false).setOnDismissListener(new DialogInterface.OnDismissListener() {
|
||||||
|
@Override
|
||||||
|
public void onDismiss(DialogInterface dialog) {
|
||||||
|
activity.startGame();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// Create the AlertDialog object and return it
|
||||||
|
return builder.create();
|
||||||
|
}
|
||||||
|
}
|
102
app/src/main/res/layout/activity_mirror_menu.xml
Normal file
102
app/src/main/res/layout/activity_mirror_menu.xml
Normal file
|
@ -0,0 +1,102 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<RelativeLayout 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:id="@+id/activity_mirror_menu"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="match_parent"
|
||||||
|
tools:context="com.rubenvandeven.emotionhero.MirrorMenuActivity">
|
||||||
|
|
||||||
|
<RelativeLayout
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="match_parent"
|
||||||
|
android:id="@+id/videoContent">
|
||||||
|
|
||||||
|
</RelativeLayout>
|
||||||
|
|
||||||
|
<LinearLayout
|
||||||
|
android:orientation="vertical"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="match_parent"
|
||||||
|
android:layout_below="@+id/logoEmotionHero"
|
||||||
|
android:layout_above="@+id/messageText"
|
||||||
|
android:layout_centerHorizontal="true"
|
||||||
|
android:layout_marginTop="19dp">
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:textAlignment="center"
|
||||||
|
android:textSize="30dp"
|
||||||
|
android:textColor="@color/textPrimary"
|
||||||
|
android:textAllCaps="true"
|
||||||
|
android:layout_marginBottom="@dimen/fab_margin"
|
||||||
|
android:text="@string/start"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:id="@+id/startButton"
|
||||||
|
android:fontFamily="sans-serif" />
|
||||||
|
<TextView
|
||||||
|
android:textAlignment="center"
|
||||||
|
android:textSize="30dp"
|
||||||
|
android:textColor="@color/textPrimary"
|
||||||
|
android:textAllCaps="true"
|
||||||
|
android:layout_marginBottom="@dimen/fab_margin"
|
||||||
|
android:text="@string/highscores"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:id="@+id/highscoresButton"
|
||||||
|
android:fontFamily="sans-serif" />
|
||||||
|
<TextView
|
||||||
|
android:textAlignment="center"
|
||||||
|
android:textSize="30dp"
|
||||||
|
android:textColor="@color/textPrimary"
|
||||||
|
android:textAllCaps="true"
|
||||||
|
android:layout_marginBottom="@dimen/fab_margin"
|
||||||
|
android:text="@string/credits"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:id="@+id/creditsButton"
|
||||||
|
android:fontFamily="sans-serif" />
|
||||||
|
|
||||||
|
</LinearLayout>
|
||||||
|
|
||||||
|
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/messageText"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:gravity="bottom"
|
||||||
|
android:textAlignment="center"
|
||||||
|
android:keepScreenOn="true"
|
||||||
|
android:text="@string/dummy_content"
|
||||||
|
android:textColor="@color/textPrimary"
|
||||||
|
android:textSize="36sp"
|
||||||
|
android:textStyle="bold"
|
||||||
|
android:paddingBottom="30dp"
|
||||||
|
android:shadowColor="@android:color/black"
|
||||||
|
android:shadowDx="0"
|
||||||
|
android:shadowDy="0"
|
||||||
|
|
||||||
|
android:layout_alignParentBottom="true"
|
||||||
|
android:shadowRadius="20"
|
||||||
|
android:typeface="normal"
|
||||||
|
android:fontFamily="sans-serif-condensed" />
|
||||||
|
|
||||||
|
|
||||||
|
<ImageView
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
app:srcCompat="@drawable/emotionhero_logo"
|
||||||
|
android:id="@+id/logoEmotionHero"
|
||||||
|
android:cropToPadding="false"
|
||||||
|
android:adjustViewBounds="true"
|
||||||
|
android:layout_marginTop="58dp"
|
||||||
|
android:paddingBottom="@dimen/activity_vertical_margin"
|
||||||
|
android:paddingLeft="@dimen/activity_horizontal_margin"
|
||||||
|
android:paddingRight="@dimen/activity_horizontal_margin"
|
||||||
|
android:paddingTop="@dimen/activity_vertical_margin"
|
||||||
|
android:layout_alignParentTop="true"
|
||||||
|
android:layout_alignParentLeft="true"
|
||||||
|
android:layout_alignParentStart="true"
|
||||||
|
android:contentDescription="@string/app_name" />
|
||||||
|
|
||||||
|
</RelativeLayout>
|
|
@ -8,7 +8,7 @@
|
||||||
<string name="nextLvl">Next Level!</string>
|
<string name="nextLvl">Next Level!</string>
|
||||||
<string name="start">Start</string>
|
<string name="start">Start</string>
|
||||||
<string name="continueBtn">Continue</string>
|
<string name="continueBtn">Continue</string>
|
||||||
<string name="highscores">Levels</string>
|
<string name="highscores">Progress</string>
|
||||||
<string name="credits">Credits</string>
|
<string name="credits">Credits</string>
|
||||||
<string name="title_activity_highscore">HighscoreActivity</string>
|
<string name="title_activity_highscore">HighscoreActivity</string>
|
||||||
<string name="action_settings">Settings</string>
|
<string name="action_settings">Settings</string>
|
||||||
|
|
Loading…
Reference in a new issue