package com.jme3.app;

import java.util.logging.Logger;

import android.app.Activity;
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.content.pm.ActivityInfo;
import android.opengl.GLSurfaceView;
import android.os.Bundle;
import android.view.Display;
import android.view.SurfaceView;
import android.view.View;
import android.view.Window;
import android.view.WindowManager;
import android.widget.TextView;

import com.jme3.app.Application;
import com.jme3.input.TouchInput;
import com.jme3.input.android.AndroidInput;
import com.jme3.input.controls.TouchListener;
import com.jme3.input.controls.TouchTrigger;
import com.jme3.input.event.TouchEvent;
import com.jme3.system.AppSettings;
import com.jme3.system.JmeSystem;
import com.jme3.system.android.OGLESContext;
import com.jme3.system.android.AndroidConfigChooser.ConfigType;


/**
 * <code>AndroidHarness</code> wraps a jme application object and runs it on Android
 * @author Kirill
 * @author larynx
 */
public class AndroidHarness extends Activity implements TouchListener, DialogInterface.OnClickListener
{
    protected final static Logger logger = Logger.getLogger(AndroidHarness.class.getName());
    
    /**
     * The application class to start
     */
    protected String appClass = "jme3test.android.Test";
    
    /**
     * The jme3 application object
     */
    protected Application app = null;
    
    /**
     * ConfigType.FASTEST is RGB565, GLSurfaceView default
     * ConfigType.BEST is RGBA8888 or better if supported by the hardware
     */
    protected ConfigType eglConfigType = ConfigType.FASTEST;
    
    /**
     * If true all valid and not valid egl configs are logged
     */
    protected boolean eglConfigVerboseLogging = false;

    /**
     * If true MouseEvents are generated from TouchEvents
     */
    protected boolean mouseEventsEnabled = true;    
    /**
     * Flip X axis
     */
    protected boolean mouseEventsInvertX = true;
    /**
     * Flip Y axis
     */
    protected boolean mouseEventsInvertY = true;
        
    /**
     * Title of the exit dialog, default is "Do you want to exit?"
     */
    protected String exitDialogTitle = "Do you want to exit?";
    /**
     * Message of the exit dialog, default is "Use your home key to bring this app into the background or exit to terminate it."
     */
    protected String exitDialogMessage = "Use your home key to bring this app into the background or exit to terminate it.";

    /**
     * Set the screen orientation, default is SENSOR
     * ActivityInfo.SCREEN_ORIENTATION_* constants
     * package android.content.pm.ActivityInfo
     *  
     *   SCREEN_ORIENTATION_UNSPECIFIED
     *   SCREEN_ORIENTATION_LANDSCAPE
     *   SCREEN_ORIENTATION_PORTRAIT
     *   SCREEN_ORIENTATION_USER
     *   SCREEN_ORIENTATION_BEHIND
     *   SCREEN_ORIENTATION_SENSOR (default)
     *   SCREEN_ORIENTATION_NOSENSOR
     */
    protected int screenOrientation = ActivityInfo.SCREEN_ORIENTATION_SENSOR;
          
    protected OGLESContext ctx;
    protected GLSurfaceView view;
    protected boolean isGLThreadPaused = true;
    final private String ESCAPE_EVENT = "TouchEscape"; 

    @Override
    public void onCreate(Bundle savedInstanceState) 
    {
        super.onCreate(savedInstanceState);

        JmeSystem.setResources(getResources());
        JmeSystem.setActivity(this);

        requestWindowFeature(Window.FEATURE_NO_TITLE);
        getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
        WindowManager.LayoutParams.FLAG_FULLSCREEN);
        
        setRequestedOrientation(screenOrientation);
        
        // Create Settings
        AppSettings settings = new AppSettings(true);
                        
        // Create the input class
        AndroidInput input = new AndroidInput(this);        
        input.setMouseEventsInvertX(mouseEventsInvertX);
        input.setMouseEventsInvertY(mouseEventsInvertY);
        input.setMouseEventsEnabled(mouseEventsEnabled);
                
        // Create application instance
        try
        {
            app = null;
            view = null;
            
            @SuppressWarnings("unchecked")
            Class<? extends Application> clazz = (Class<? extends Application>) Class.forName(appClass);
            app = clazz.newInstance();
            
            app.setSettings(settings);
            app.start();    
            ctx = (OGLESContext) app.getContext();
            view = ctx.createView(input, eglConfigType, eglConfigVerboseLogging);
            setContentView(view);     
            
            // Set the screen reolution
            WindowManager wind = this.getWindowManager();
            Display disp = wind.getDefaultDisplay();
            ctx.getSettings().setResolution(disp.getWidth(), disp.getHeight());
            
            AppSettings s = ctx.getSettings();            
            logger.info("Settings: Width " + s.getWidth() + " Height " + s.getHeight());
            
            
        }
        catch (Exception ex)
        {
            handleError("Class " + appClass + " init failed", ex);
            setContentView(new TextView(this));
        }
        
    }


    @Override
    protected void onRestart()
    {
        super.onRestart(); 
        if (app != null)
            app.restart();
        logger.info("onRestart");
    }
    

    @Override
    protected void onStart()
    {
        super.onStart();
        logger.info("onStart");
    }
    
    @Override
    protected void onResume() 
    {
        super.onResume();
        if (view != null)
            view.onResume();
        isGLThreadPaused = false;
        logger.info("onResume");
    }

    @Override
    protected void onPause() 
    {
        super.onPause();
        if (view != null)
            view.onPause();
        isGLThreadPaused = true;
        logger.info("onPause");
    }
    
    @Override
    protected void onStop()
    {
        super.onStop();
        logger.info("onStop");
    }

    @Override
    protected void onDestroy()
    {
        if (app != null)
            app.stop(! isGLThreadPaused);
        super.onDestroy();                
        logger.info("onDestroy");
    }

    public Application getJmeApplication()
    {
        return app;
    }
    
    /**
     * Called when an error has occured. This is typically
     * invoked when an uncought exception is thrown in the render thread.
     * @param errorMsg The error message, if any, or null.
     * @param t Throwable object, or null.
     */
    public void handleError(final String errorMsg, final Throwable t)
    {
        
        String s = "";
        if (t != null && t.getStackTrace() != null)
        {
            for (StackTraceElement ste: t.getStackTrace())
            {
                s +=  ste.getClassName() + "." + ste.getMethodName() + "(" + + ste.getLineNumber() + ") ";
            }
        }                
        
        final String sTrace = s;
        
        logger.severe(t != null ? t.toString() : "OpenGL Exception");
        logger.severe((errorMsg != null ? errorMsg + ": " : "") + sTrace);
        
        this.runOnUiThread(new Runnable() {
            @Override
            public void run() 
            {                                
                AlertDialog dialog = new AlertDialog.Builder(AndroidHarness.this)
               // .setIcon(R.drawable.alert_dialog_icon)
                .setTitle(t != null ? (t.getMessage() != null ? (t.getMessage() + ": " + t.getClass().getName()) : t.getClass().getName()) : "OpenGL Exception")
                .setPositiveButton("Kill", AndroidHarness.this)
                .setMessage((errorMsg != null ? errorMsg + ": " : "") + sTrace)
                .create();    
                dialog.show();                
            }
        });
        
    }
    
    /**
     * Called by the android alert dialog, terminate the activity and OpenGL rendering
     * @param dialog
     * @param whichButton
     */
    public void onClick(DialogInterface dialog, int whichButton) 
    {        
        if (whichButton != -2)
        {
            if (app != null)
                app.stop(true);
            this.finish();
        }
    }
    
    /**
     * Gets called by the InputManager on all touch/drag/scale events
     */    
    @Override    
    public void onTouch(String name, TouchEvent evt, float tpf)  
    {
        if (name.equals(ESCAPE_EVENT))
        {
            switch(evt.getType())
            {                
                case KEY_UP:
                    this.runOnUiThread(new Runnable() 
                    {
                        @Override
                        public void run() 
                        {                                                
                            AlertDialog dialog = new AlertDialog.Builder(AndroidHarness.this)
                           // .setIcon(R.drawable.alert_dialog_icon)
                            .setTitle(exitDialogTitle)
                            .setPositiveButton("Yes", AndroidHarness.this)
                            .setNegativeButton("No", AndroidHarness.this)
                            .setMessage(exitDialogMessage)
                            .create();    
                            dialog.show();                
                        }
                    });
                            
                            
                    break;
                    
               default:
                   break;
            }
        }
                        
    }    
    
}