package gameEngine.views;
import gameEngine.ConnectionManager;
import gameEngine.GameEditor;
import gameEngine.SwingGameEditor;
import gameEngine.Time;
import gameEngine.entites.Entity;
import gameEngine.entites.GameObject;
import gameEngine.entites.gameComponents.*;
import gameEngine.input.*;
import gameEngine.scenes.*;
import org.lwjgl.*;
import org.lwjgl.glfw.*;
import org.lwjgl.opengl.*;
import org.lwjgl.system.*;
import java.nio.*;
import java.util.HashMap;
import static org.lwjgl.glfw.Callbacks.*;
import static org.lwjgl.glfw.GLFW.*;
import static org.lwjgl.opengl.GL11.*;
import static org.lwjgl.system.MemoryStack.*;
import static org.lwjgl.system.MemoryUtil.*;
public class Window {
    private static Window window;
    private static Scene currentScene;
    public int width;
    public int height;
    private String title;
    private long glfwWindow;
    private static GameEditor gameEditor;
    public ConnectionManager connectionManager = new ConnectionManager();
    private static HashMap<String , Entity> editorEntities;
    private Window() {
        this.width = 1200;
        this.height = 900;
        this.title = "HelloWorld";
        init();
    }
    public static void changeScene(int newScene) {
        switch (newScene) {
            case 0:  // EditorScene
                if (editorEntities == null) {
                    currentScene = new EditorScene();
                } else {
                    currentScene = new EditorScene(editorEntities);
                    gameEditor.setScene(currentScene);
                }
                initializeSceneEntities();
                break;
            case 1:  // GameScene
                EditorScene editorScene = (EditorScene) currentScene;
                editorEntities = editorScene.editorEntities;
                HashMap<String, Entity> gameSceneEntities = new HashMap<>(editorEntities);
                currentScene = new GameScene(gameSceneEntities);
                initializeSceneEntities();
                break;
            default:
                assert false : "Unknown Scene [" + newScene + "]";
                break;
        }
    }
    private static void initializeSceneEntities() {
        for (Entity entity : currentScene.entities.values()) {
            if (entity instanceof GameObject) {
                GameObject gameObject = (GameObject) entity;
                gameObject.initComponents();  // コンポーネントの初期化
            }
        }
    }
    public static Window get() {
        if (window == null) {
            window = new Window();
        }
        return window;
    }
    public void runWithEditor() {
        //swingGameEditor = new SwingGameEditor();
        run();
    }
    public void run() {
        System.out.println("Hello LWJGL " + Version.getVersion() + "!");
        Window.changeScene(0);
        gameEditor = new GameEditor(currentScene);
        loop();
        glfwFreeCallbacks(glfwWindow);
        glfwDestroyWindow(glfwWindow);
        glfwTerminate();
        glfwSetErrorCallback(null).free();
    }
    private void init() {
        GLFWErrorCallback.createPrint(System.err).set();
        if ( !glfwInit() ) {
            throw new IllegalStateException("Unable to initialize GLFW");
        }
        //Configure GLFW
        glfwDefaultWindowHints();
        glfwWindowHint(GLFW_VISIBLE, GLFW_FALSE);
        glfwWindowHint(GLFW_RESIZABLE, GLFW_FALSE);
        glfwWindowHint(GLFW_MAXIMIZED, GLFW_FALSE);
        glfwWindow = glfwCreateWindow(this.width, this.height, this.title, NULL, NULL);
        if ( glfwWindow == NULL ) {
            throw new RuntimeException("Failed to create the GLFW window");
        }
        glfwSetCursorPosCallback(glfwWindow, MouseInput::mousePosCallback);
        glfwSetMouseButtonCallback(glfwWindow, MouseInput::mouseButtonCallback);
        glfwSetScrollCallback(glfwWindow, MouseInput::mouseScrollCallBack);
        glfwSetKeyCallback(glfwWindow, KeyInput::keyCallback);
        // Get the thread stack and push a new frame
        try ( MemoryStack stack = stackPush() ) {
            IntBuffer pWidth = stack.mallocInt(1); // int*
            IntBuffer pHeight = stack.mallocInt(1); // int*
            // Get the window size passed to glfwCreateWindow
            glfwGetWindowSize(glfwWindow, pWidth, pHeight);
            // Get the resolution of the primary monitor
            GLFWVidMode vidmode = glfwGetVideoMode(glfwGetPrimaryMonitor());
            // Center the window
            glfwSetWindowPos(
                    glfwWindow,
                    (vidmode.width() - pWidth.get(0)) / 2,
                    (vidmode.height() - pHeight.get(0)) / 2
            );
        }
        glfwMakeContextCurrent(glfwWindow);
        glfwSwapInterval(1);
        glfwShowWindow(glfwWindow);
        GL.createCapabilities();
    }
    private void loop() {
        while (!glfwWindowShouldClose(glfwWindow)) {
            glfwPollEvents();
            glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
            float dt = Time.deltaTime;
            if (dt >= 0) {
                resetScene();
                // シーン全体の update を呼び出す
                currentScene.update(dt);
                currentScene.processTasks();
                if(currentScene instanceof EditorScene) {
                    connectionManager.update();
                    for (Entity entity : ((EditorScene) currentScene).editorEntities.values()) {
                        if (entity instanceof GameObject) {
                            GameObject gameObject = (GameObject) entity;
                            gameObject.updateComponents();
                        }
                    }
                    gameEditor.update();
                }
                else {
                    for (Entity entity : currentScene.entities.values()) {
                        if (entity instanceof GameObject) {
                            GameObject gameObject = (GameObject) entity;
                            if(gameObject.getComponent(EntityView.class) != null) continue;
                            if(gameObject.getComponent(ComponentView.class) != null)  continue;
                            if(gameObject.getComponent(TextMesh.class) != null) continue;
                            gameObject.updateComponents();
                        }
                    }
                }
                MouseInput.endFrame();
                KeyInput.endFrame();
            }
            glfwSwapBuffers(glfwWindow);
            Time.update();
        }
    }
    //Gameシーン開始時にタイマーをリセット、Editorシーン戻ったときにフラグをfalseにする
    private boolean startGameScene = false;
    private void resetScene() {
        boolean isGameScene = currentScene instanceof GameScene;
        if (isGameScene && !startGameScene) {
            Time.reset();
            startGameScene = true;
        }
        if (!isGameScene) startGameScene = false;
    }
    public Scene getScene(){
        return currentScene;
    }
}