Engine.java 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. package me.hammerle.snuviengine.api;
  2. import org.lwjgl.glfw.*;
  3. import org.lwjgl.opengl.*;
  4. import static org.lwjgl.glfw.Callbacks.*;
  5. import static org.lwjgl.glfw.GLFW.*;
  6. import static org.lwjgl.opengl.GL11.*;
  7. import static org.lwjgl.system.MemoryUtil.*;
  8. public abstract class Engine
  9. {
  10. private static final String VERSION = "0.0.1";
  11. public static final int TILE_SIZE = 16;
  12. public static final float SCALE = 2.0f;
  13. private long window;
  14. private int fpsIndex = 0;
  15. private final long[] fps = new long[200];
  16. private long fpsSum = 0;
  17. private double currentFps = 0;
  18. private int tpsIndex = 0;
  19. private final long[] tps = new long[200];
  20. private long tpsSum = 0;
  21. private double currentTps = 0;
  22. private long nanosPerTick = 10_000_000;
  23. private final int maxTicksPerFrame = 20;
  24. public Engine()
  25. {
  26. }
  27. public final void run()
  28. {
  29. initGLFW();
  30. loop();
  31. glfwFreeCallbacks(window);
  32. glfwDestroyWindow(window);
  33. glfwTerminate();
  34. GLFWErrorCallback error = glfwSetErrorCallback(null);
  35. if(error != null)
  36. {
  37. error.free();
  38. }
  39. }
  40. private void initGLFW()
  41. {
  42. GLFWErrorCallback.createPrint(System.err).set();
  43. if(!glfwInit())
  44. {
  45. throw new IllegalStateException("Unable to initialize GLFW");
  46. }
  47. glfwDefaultWindowHints();
  48. glfwWindowHint(GLFW_VISIBLE, GLFW_FALSE);
  49. glfwWindowHint(GLFW_RESIZABLE, GLFW_TRUE);
  50. window = glfwCreateWindow(Shader.getViewWidth(), Shader.getViewHeight(), "SnuviEngine " + VERSION, NULL, NULL);
  51. if(window == NULL)
  52. {
  53. throw new RuntimeException("Failed to create the GLFW window");
  54. }
  55. glfwSetKeyCallback(window, (w, key, scancode, action, mods) ->
  56. {
  57. if(action == GLFW_RELEASE)
  58. {
  59. KeyHandler.onKeyUpEvent(key);
  60. if(key == GLFW_KEY_ESCAPE)
  61. {
  62. glfwSetWindowShouldClose(window, true);
  63. }
  64. }
  65. else if(action == GLFW_PRESS)
  66. {
  67. KeyHandler.onKeyDownEvent(key);
  68. }
  69. });
  70. glfwSetFramebufferSizeCallback(window, (w, fwidth, fheight) ->
  71. {
  72. glViewport(0, 0, fwidth, fheight);
  73. Shader.setViewPort(fwidth, fheight);
  74. });
  75. //glfwSetWindowAspectRatio(window, 16, 9);
  76. glfwMakeContextCurrent(window);
  77. glfwSwapInterval(1);
  78. glfwShowWindow(window);
  79. }
  80. private void loop()
  81. {
  82. GL.createCapabilities();
  83. long lastFrame = System.nanoTime();
  84. long lastTick = System.nanoTime();
  85. long time;
  86. long lag = 0;
  87. Shader.init();
  88. init();
  89. while(!glfwWindowShouldClose(window))
  90. {
  91. glfwSwapBuffers(window);
  92. glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
  93. time = System.nanoTime();
  94. fpsSum -= fps[fpsIndex];
  95. fps[fpsIndex] = time - lastFrame;
  96. fpsSum += fps[fpsIndex];
  97. lag += fps[fpsIndex];
  98. fpsIndex = (fpsIndex + 1) % fps.length;
  99. lastFrame = time;
  100. currentFps = (1_000_000_000.0 * fps.length) / fpsSum;
  101. int ticksPerFrame = 0;
  102. while(lag >= nanosPerTick)
  103. {
  104. lag -= nanosPerTick;
  105. time = System.nanoTime();
  106. tpsSum -= tps[tpsIndex];
  107. tps[tpsIndex] = time - lastTick;
  108. tpsSum += tps[tpsIndex];
  109. tpsIndex = (tpsIndex + 1) % tps.length;
  110. lastTick = time;
  111. currentTps = (1_000_000_000.0 * tps.length) / tpsSum;
  112. KeyHandler.tick();
  113. tick();
  114. ticksPerFrame++;
  115. if(ticksPerFrame >= maxTicksPerFrame)
  116. {
  117. long skip = lag / nanosPerTick;
  118. lag -= skip * nanosPerTick;
  119. if(skip > 0)
  120. {
  121. System.out.println("skipped " + skip + " game ticks " + lag);
  122. }
  123. break;
  124. }
  125. }
  126. Shader.doTasks();
  127. renderTick((double) lag / nanosPerTick);
  128. glfwPollEvents();
  129. }
  130. }
  131. public final void setNanosPerTick(long nanos)
  132. {
  133. nanosPerTick = nanos;
  134. }
  135. public final double getTps()
  136. {
  137. return currentTps;
  138. }
  139. public final double getFps()
  140. {
  141. return currentFps;
  142. }
  143. public abstract void init();
  144. public abstract void tick();
  145. public abstract void renderTick(double lag);
  146. }