/*
* Copyright (C) 2016 Google Inc. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.google.android.apps.santatracker.doodles.pursuit;
import android.content.Context;
import android.graphics.Point;
import android.graphics.Typeface;
import android.graphics.drawable.Drawable;
import android.os.Bundle;
import android.support.v4.content.ContextCompat;
import android.util.Log;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.FrameLayout;
import android.widget.LinearLayout;
import android.widget.TextView;
import com.google.android.apps.santatracker.doodles.R;
import com.google.android.apps.santatracker.doodles.pursuit.PursuitModel.ScoreListener;
import com.google.android.apps.santatracker.doodles.pursuit.PursuitModel.State;
import com.google.android.apps.santatracker.doodles.pursuit.PursuitModel.StateChangedListener;
import com.google.android.apps.santatracker.doodles.shared.AndroidUtils;
import com.google.android.apps.santatracker.doodles.shared.DoodleConfig;
import com.google.android.apps.santatracker.doodles.shared.EventBus;
import com.google.android.apps.santatracker.doodles.shared.EventBus.EventBusListener;
import com.google.android.apps.santatracker.doodles.shared.GameFragment;
import com.google.android.apps.santatracker.doodles.shared.GameType;
import com.google.android.apps.santatracker.doodles.shared.HistoryManager;
import com.google.android.apps.santatracker.doodles.shared.PineappleLogger;
import com.google.android.apps.santatracker.doodles.shared.UIUtil;
import com.google.android.apps.santatracker.doodles.shared.sound.SoundManager;
import com.google.android.apps.santatracker.util.MeasurementManager;
import com.google.firebase.analytics.FirebaseAnalytics;
import java.text.NumberFormat;
import java.util.Locale;
import static com.google.android.apps.santatracker.doodles.shared.PineappleLogEvent.RUNNING_GAME_TYPE;
/**
* Fragment for the second version of the running game.
* Manages input & threads, delegates to PursuitModel & PursuitView for the rest.
*/
public class PursuitFragment extends GameFragment implements EventBusListener {
private static final String TAG = PursuitFragment.class.getSimpleName();
private static final long GAME_OVER_DELAY_MILLISECONDS = 1000;
private static final long GAME_OVER_SHORT_DELAY_MILLISECONDS = 150;
private PursuitView gameView;
private TextView countdownView;
private PursuitModel model;
private boolean mIsGameOver = false;
public PursuitFragment() {
super();
}
public PursuitFragment(Context context, DoodleConfig doodleConfig, PineappleLogger logger) {
super(context, doodleConfig, logger);
}
@Override
public void update(float deltaMs) {
if (!isPaused) {
model.update(deltaMs);
}
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
if (context == null) {
return null;
}
historyManager = new HistoryManager(context, new HistoryManager.HistoryListener() {
@Override
public void onFinishedLoading() {
}
@Override
public void onFinishedSaving() {
}
});
wrapper = new FrameLayout(context);
titleView = getTitleView(R.drawable.snowballrunner_loadingscreen, R.string.running);
wrapper.addView(titleView);
getActivity().runOnUiThread(new Runnable() {
@Override
public void run() {
loadGame();
}
});
return wrapper;
}
@Override
protected void firstPassLoadOnUiThread() {
wrapper.setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
return onTouchEvent(event);
}
});
countdownView = new TextView(context);
countdownView.setGravity(Gravity.CENTER);
countdownView.setTextColor(context.getResources().getColor(R.color.ui_text_yellow));
countdownView.setTypeface(Typeface.DEFAULT_BOLD);
countdownView.setText("0");
countdownView.setVisibility(View.INVISIBLE);
Locale locale = context.getResources().getConfiguration().locale;
countdownView.setText(NumberFormat.getInstance(locale).format(3));
Point screenDimens = AndroidUtils.getScreenSize();
UIUtil.fitToBounds(countdownView, screenDimens.x / 10, screenDimens.y / 10);
final FrameLayout.LayoutParams lp = new FrameLayout.LayoutParams(
LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.MATCH_PARENT);
gameView = new PursuitView(context);
wrapper.addView(gameView, 0, lp);
wrapper.addView(countdownView, 1, lp);
scoreView = getScoreView();
wrapper.addView(scoreView, 2, lp);
pauseView = getPauseView();
pauseView.hidePauseButton();
wrapper.addView(pauseView, 3, lp);
}
@Override
protected void secondPassLoadOnBackgroundThread() {
super.secondPassLoadOnBackgroundThread();
EventBus.getInstance().register(this);
model = new PursuitModel(context.getResources(), getActivity().getApplicationContext());
gameView.setModel(model);
model.setStateListener(new StateChangedListener() {
@Override
public void onStateChanged(State state) {
if (state == State.SETUP) {
getActivity().runOnUiThread(new Runnable() {
@Override
public void run() {
hideTitle();
pauseView.showPauseButton();
}
});
} else if (state == State.SUCCESS) {
getActivity().runOnUiThread(new Runnable() {
@Override
public void run() {
if (model == null) {
return;
}
mIsGameOver = true;
float timeInSeconds = model.getScore();
int starCount = 4 - model.getFinishingPlace();
long delay = GAME_OVER_SHORT_DELAY_MILLISECONDS;
if (model.getFinishingPlace() == 1) {
delay = GAME_OVER_DELAY_MILLISECONDS;
}
Double bestTimeInSeconds = historyManager.getBestScore(GameType.PURSUIT);
Integer bestStarCount = historyManager.getBestStarCount(GameType.PURSUIT);
if (bestTimeInSeconds == null || timeInSeconds < bestTimeInSeconds) {
bestTimeInSeconds = (double) timeInSeconds;
historyManager.setBestScore(GameType.PURSUIT, bestTimeInSeconds);
historyManager.save();
}
if (bestStarCount == null || starCount > bestStarCount) {
bestStarCount = starCount;
historyManager.setBestStarCount(GameType.PURSUIT, bestStarCount);
historyManager.save();
}
pauseView.hidePauseButton();
final int finalStarCount = starCount;
final double finalBestTimeInSeconds = bestTimeInSeconds;
scoreView.postDelayed(new Runnable() {
@Override
public void run() {
if (finalStarCount >= 1) {
scoreView.addStar();
}
if (finalStarCount >= 2) {
scoreView.addStar();
}
if (finalStarCount >= 3) {
scoreView.addStar();
}
scoreView.updateBestScore(AndroidUtils.getText(context.getResources(),
R.string.pursuit_score, finalBestTimeInSeconds));
scoreView.setShareDrawable(getShareImageDrawable(finalStarCount));
updateShareText();
scoreView.animateToEndState();
}
}, delay);
}
});
// [ANALYTICS]
FirebaseAnalytics analytics = FirebaseAnalytics.getInstance(getActivity());
MeasurementManager.recordRunningEnd(analytics,
4 - model.getFinishingPlace(),
(int) model.getScore());
} else if (state == State.FAIL) {
getActivity().runOnUiThread(new Runnable() {
@Override
public void run() {
pauseView.hidePauseButton();
mIsGameOver = true;
scoreView.postDelayed(new Runnable() {
@Override
public void run() {
scoreView.setHeaderText(null);
updateShareText();
scoreView.setShareDrawable(getShareImageDrawable(0));
scoreView.animateToEndState();
}
}, GAME_OVER_DELAY_MILLISECONDS);
}});
// [ANALYTICS]
FirebaseAnalytics analytics = FirebaseAnalytics.getInstance(getActivity());
MeasurementManager.recordRunningEnd(analytics, 0, 0);
} else if (state == State.READY) {
getActivity().runOnUiThread(new Runnable() {
@Override
public void run() {
mIsGameOver = false;
pauseView.showPauseButton();
}
});
}
}
});
model.setScoreListener(new ScoreListener() {
@Override
public void newScore(final float timeInSeconds) {
getActivity().runOnUiThread(new Runnable() {
@Override
public void run() {
scoreView.updateCurrentScore(
AndroidUtils.getText(context.getResources(),
R.string.pursuit_score, timeInSeconds), false);
}
});
}
});
model.setCountdownView(countdownView);
}
@Override
protected void finalPassLoadOnUiThread() {
soundManager = SoundManager.getInstance();
loadSounds();
onFinishedLoading();
startHandlers();
}
@Override
protected void loadSounds() {
super.loadSounds();
soundManager.loadShortSound(context, R.raw.bmx_cheering);
soundManager.loadShortSound(context, R.raw.jumping_jump);
soundManager.loadShortSound(context, R.raw.present_throw_character_appear);
soundManager.loadShortSound(context, R.raw.running_foot_loop_fast, true, 2f);
soundManager.loadShortSound(context, R.raw.running_foot_power_up);
soundManager.loadShortSound(context, R.raw.running_foot_power_up_fast, false, 0.4f);
soundManager.loadShortSound(context, R.raw.running_foot_power_squish);
soundManager.loadShortSound(context, R.raw.present_throw_block);
}
@Override
protected void replay() {
super.replay();
model.gameReplay();
}
private Drawable getShareImageDrawable(int starCount) {
return ContextCompat.getDrawable(getActivity(), R.drawable.winner);
}
@Override
protected void onDestroyHelper() {
if (gameView != null) {
gameView.setModel(null);
}
model = null;
}
@Override
public void onEventReceived(int type, Object data) {
if (isDestroyed) {
return;
}
if (type == EventBus.PLAY_SOUND && soundManager != null) {
int resId = (int) data;
soundManager.play(resId);
} else if (type == EventBus.PAUSE_SOUND && soundManager != null) {
int resId = (int) data;
soundManager.pause(resId);
} else if (type == EventBus.MUTE_SOUNDS && soundManager != null) {
boolean shouldMute = (boolean) data;
if (shouldMute) {
soundManager.mute();
} else {
soundManager.unmute();
}
} else if (type == EventBus.GAME_LOADED) {
mIsGameOver = false;
long loadTimeMs = (long) data;
model.titleDurationMs = Math.max(0, model.titleDurationMs - loadTimeMs);
Log.d(TAG, "Waiting " + model.titleDurationMs + "ms and then hiding title.");
}
}
public boolean onTouchEvent(MotionEvent event) {
if (model != null) {
model.touch(event);
return true;
}
return false;
}
@Override
protected void resume() {
super.resume();
if (uiRefreshHandler != null) {
uiRefreshHandler.start(gameView);
}
}
private void updateShareText() {
}
@Override
protected String getGameType() {
return RUNNING_GAME_TYPE;
}
@Override
protected float getScore() {
if (model == null) {
return 0;
}
return model.getScore();
}
@Override
protected int getShareImageId() {
return scoreView.getStarCount();
}
public boolean isGameOver() {
return mIsGameOver;
}
}