/**
* Copyright 2008 - 2015 The Loon Game Engine Authors
*
* 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.
*
* @project loon
* @author cping
* @email:javachenpeng@yahoo.com
* @version 0.5
*/
package loon.lwjgl;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.lang.reflect.Method;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
import org.lwjgl.glfw.GLFWErrorCallback;
import org.lwjgl.glfw.GLFWImage;
import org.lwjgl.glfw.GLFWVidMode;
import org.lwjgl.opengl.GL;
import org.lwjgl.opengl.GL11;
import static org.lwjgl.glfw.GLFW.*;
import static org.lwjgl.opengl.GL11.GL_FALSE;
import loon.*;
import loon.canvas.Image;
import loon.canvas.Pixmap;
import loon.event.KeyMake;
import loon.geom.Dimension;
import loon.jni.NativeSupport;
import loon.utils.Scale;
import loon.utils.reply.Port;
public class Lwjgl3Game extends LGame {
private Lwjgl3Sync sync;
private final long window;
public static class JavaSetting extends LSetting {
public boolean vSyncEnabled = true;
public String[] iconPaths = null;
public int synMode = Lwjgl3Sync.LWJGL_GLFW;
}
final static private boolean osIsLinux;
final static private boolean osIsUnix;
final static private boolean osIsMacOs;
final static private boolean osIsWindows;
final static private boolean osIsWindowsXP;
final static private boolean osIsWindows2003;
final static private boolean osBit64;
final static public String OS_NAME;
final static public int JAVA_13 = 0;
final static public int JAVA_14 = 1;
final static public int JAVA_15 = 2;
final static public int JAVA_16 = 3;
final static public int JAVA_17 = 4;
final static public int JAVA_18 = 5;
final static public int JAVA_19 = 6;
final static Support support = new Lwjgl3Support();
static {
OS_NAME = System.getProperty("os.name").toLowerCase();
osIsLinux = OS_NAME.indexOf("linux") != -1;
osIsUnix = OS_NAME.indexOf("nix") != -1 || OS_NAME.indexOf("nux") != 1;
osIsMacOs = OS_NAME.indexOf("mac") != -1;
osIsWindows = OS_NAME.indexOf("windows") != -1;
osIsWindowsXP = OS_NAME.startsWith("Windows")
&& (OS_NAME.compareTo("5.1") >= 0);
osIsWindows2003 = "windows 2003".equals(OS_NAME);
osBit64 = System.getProperty("os.arch").equals("amd64");
}
final private static Runtime systemRuntime = Runtime.getRuntime();
public static boolean isLinux() {
return osIsLinux;
}
public static boolean isMacOS() {
return osIsMacOs;
}
public static boolean isUnix() {
return osIsUnix;
}
public static boolean isWindows() {
return osIsWindows;
}
public static boolean isWindowsXP() {
return osIsWindowsXP;
}
public static boolean isWindows2003() {
return osIsWindows2003;
}
public static boolean isBit64() {
return osBit64;
}
public static boolean isSun() {
return System.getProperty("java.vm.vendor").indexOf("Sun") != -1
|| System.getProperty("java.vm.vendor").indexOf("Oracle") != -1;
}
public static boolean isApple() {
return System.getProperty("java.vm.vendor").indexOf("Apple") != -1;
}
public static boolean isHPUX() {
return System.getProperty("java.vm.vendor").indexOf(
"Hewlett-Packard Company") != -1;
}
public static boolean isIBM() {
return System.getProperty("java.vm.vendor").indexOf("IBM") != -1;
}
public static boolean isBlackdown() {
return System.getProperty("java.vm.vendor").indexOf("Blackdown") != -1;
}
private boolean active = true;
private final long start = System.nanoTime();
private final ExecutorService pool = Executors.newFixedThreadPool(4);
private final Lwjgl3Log log = new Lwjgl3Log();
private final Asyn asyn = new Lwjgl3Asyn(pool, log, frame);
private final Lwjgl3Accelerometer accelerometer = new Lwjgl3Accelerometer();
private final Lwjgl3Save save;
private final Lwjgl3ImplGraphics graphics;
private final Lwjgl3Input input;
private final Lwjgl3Assets assets = new Lwjgl3Assets(this);
public static class Headless extends Lwjgl3Game {
public Headless(Loon game, LSetting config) {
super(game, config);
}
@Override
public void setTitle(String title) {
}
@Override
protected void preInit() {
}
@Override
protected Lwjgl3ImplGraphics createGraphics() {
return new Lwjgl3ImplGraphics(this, null, Scale.ONE) {
{
setSize(game.setting.width, game.setting.height,
game.setting.fullscreen);
}
@Override
public void setSize(int width, int height, boolean fullscreen) {
updateViewport(Scale.ONE, width, height);
}
@Override
public Dimension screenSize() {
return new Dimension(game.setting.width,
game.setting.height);
}
@Override
protected void init() {
}
@Override
protected void upload(BufferedImage img, LTexture tex) {
}
};
}
@Override
protected Lwjgl3Input createInput() {
return new Lwjgl3Input(this);
}
}
private final GLFWErrorCallback errorCallback;
public Lwjgl3Game(final Loon game, final LSetting config) {
super(config, game);
this.preInit();
if (isMacOS()) {
System.setProperty("java.awt.headless", "true");
}
if (config instanceof JavaSetting) {
this.sync = new Lwjgl3Sync(((JavaSetting) config).synMode);
} else {
this.sync = new Lwjgl3Sync(Lwjgl3Sync.LWJGL_GLFW);
}
glfwSetErrorCallback(errorCallback = new GLFWErrorCallback() {
@Override
public void invoke(int error, long description) {
log().error(
"GL Error (" + error + "):"
+ getDescription(description));
}
});
if (!glfwInit())
throw new RuntimeException("Failed to init GLFW.");
long monitor = glfwGetPrimaryMonitor();
GLFWVidMode vidMode = glfwGetVideoMode(monitor);
int width = config.width, height = config.height;
if (config.fullscreen) {
width = vidMode.width();
height = vidMode.height();
} else {
monitor = 0;
}
glfwDefaultWindowHints();
glfwWindowHint(GLFW_VISIBLE, GL_FALSE);
glfwWindowHint(GLFW_RESIZABLE, GL_FALSE);
glfwWindowHint(GLFW_RED_BITS, 8);
glfwWindowHint(GLFW_GREEN_BITS, 8);
glfwWindowHint(GLFW_BLUE_BITS, 8);
glfwWindowHint(GLFW_ALPHA_BITS, 8);
glfwWindowHint(GLFW_STENCIL_BITS, 0);
glfwWindowHint(GLFW_DEPTH_BITS, 16);
glfwWindowHint(GLFW_SAMPLES, 0);
window = glfwCreateWindow(width, height, config.appName, monitor, 0);
if (window == 0) {
throw new RuntimeException(
"Failed to create window; see error log.");
}
this.graphics = createGraphics();
this.input = createInput();
glfwSetWindowPos(window, (vidMode.width() - width) / 2,
(vidMode.height() - height) / 2);
glfwMakeContextCurrent(window);
graphics.setSize(config.width, config.height, config.fullscreen);
if (config instanceof JavaSetting) {
glfwSwapInterval(((JavaSetting) config).vSyncEnabled ? 1 : 0);
} else {
glfwSwapInterval(0);
}
GL.createCapabilities();
this.save = new Lwjgl3Save(log, config.appName);
if (config.activationKey != -1) {
input.keyboardEvents.connect(new Port<KeyMake.Event>() {
public void onEmit(KeyMake.Event event) {
if (event instanceof KeyMake.KeyEvent) {
KeyMake.KeyEvent kevent = (KeyMake.KeyEvent) event;
if (kevent.keyCode == config.activationKey
&& kevent.down) {
toggleActivation();
}
}
}
});
}
this.setTitle(config.appName);
this.initProcess();
if (config instanceof JavaSetting) {
setIcon(window, ((JavaSetting) config).iconPaths);
}
this.graphics.init();
this.input.init();
glfwShowWindow(window);
for (int i = 0; i < 2; i++) {
GL11.glClearColor(0, 0, 0, 1);
GL11.glClear(GL11.GL_COLOR_BUFFER_BIT);
glfwSwapBuffers(window);
}
}
public void setTitle(String title) {
((Lwjgl3Graphics) graphics()).setTitle(title);
}
@Override
public double time() {
return System.currentTimeMillis();
}
@Override
public Type type() {
return Type.JAVASE;
}
@Override
public int tick() {
return (int) ((System.nanoTime() - start) / 1000000L);
}
@Override
public Lwjgl3Assets assets() {
return assets;
}
@Override
public Asyn asyn() {
return asyn;
}
@Override
public Lwjgl3ImplGraphics graphics() {
return graphics;
}
@Override
public Lwjgl3Input input() {
return input;
}
@Override
public Log log() {
return log;
}
@Override
public Save save() {
return save;
}
@Override
public Accelerometer accel() {
return accelerometer;
}
@Override
public Support support() {
return support;
}
private static void browse(String url) throws Exception {
if (isMacOS()) {
Class<?> fileMgr = Class.forName("com.apple.eio.FileManager");
Method openURL = fileMgr.getDeclaredMethod("openURL",
new Class[] { String.class });
openURL.invoke(null, new Object[] { url });
} else if (isWindows()) {
Runtime.getRuntime().exec(
"rundll32 url.dll,FileProtocolHandler " + url);
} else {
String[] browsers = { "google-chrome", "firefox", "mozilla",
"opera", "epiphany", "konqueror", "netscape", "links",
"lynx", "epiphany", "conkeror", "midori", "kazehakase", };
String browser = null;
for (int count = 0; count < browsers.length && browser == null; count++) {
if (Runtime.getRuntime()
.exec(new String[] { "which", browsers[count] })
.waitFor() == 0) {
browser = browsers[count];
}
}
if (browser == null) {
throw new Exception("Could not find web browser");
} else {
Runtime.getRuntime().exec(new String[] { browser, url });
}
}
}
@Override
public void openURL(String url) {
try {
java.net.URI uri = new java.net.URI(url);
java.awt.Desktop.getDesktop().browse(uri);
} catch (Throwable e) {
try {
browse(url);
} catch (Throwable err) {
try {
if (isWindows()) {
File iexplore = new File(
"C:\\Program Files\\Internet Explorer\\iexplore.exe");
if (iexplore.exists()) {
systemRuntime.exec(iexplore.getAbsolutePath()
+ " \"" + url + "\"");
} else {
File chrome = new File(
"C:\\Program Files (x86)\\Google\\Chrome\\Application\\chrome.exe");
if (chrome.exists()) {
systemRuntime.exec(chrome.getAbsolutePath()
+ " \"" + url + "\"");
return;
}
chrome = new File(
"C:\\Program Files\\Google\\Chrome\\Application\\chrome.exe");
if (chrome.exists()) {
systemRuntime.exec(chrome.getAbsolutePath()
+ " \"" + url + "\"");
return;
}
systemRuntime
.exec("rundll32 url.dll,FileProtocolHandler "
+ url);
}
} else if (isMacOS()) {
systemRuntime.exec("open " + url);
} else if (isUnix()) {
String[] browsers = { "google-chrome", "firefox",
"mozilla", "opera", "epiphany", "konqueror",
"netscape", "links", "lynx", "epiphany",
"conkeror", "midori", "kazehakase" };
StringBuffer cmd = new StringBuffer();
for (int i = 0; i < browsers.length; i++) {
cmd.append((i == 0 ? "" : " || ") + browsers[i]
+ " \"" + url + "\" ");
}
systemRuntime.exec(new String[] { "sh", "-c",
cmd.toString() });
}
} catch (IOException ex) {
e.printStackTrace();
}
}
}
}
protected void preInit() {
try {
NativeSupport.loadJNI("lplus");
} catch (Throwable exc) {
exc.printStackTrace();
}
}
protected Lwjgl3ImplGraphics createGraphics() {
return new Lwjgl3Graphics(this, window);
}
protected Lwjgl3Input createInput() {
return new Lwjgl3InputMake(this, window);
}
protected void shutdown() {
status.emit(Status.EXIT);
try {
pool.shutdown();
pool.awaitTermination(1, TimeUnit.SECONDS);
} catch (InterruptedException ie) {
}
System.exit(0);
}
protected void processFrame() {
try {
input.update();
} catch (Exception e) {
log.warn("your system input exception !", e);
}
emitFrame();
}
protected void toggleActivation() {
active = !active;
}
static void setIcon(long win, String[] imagePaths) {
if (isMacOS()) {
return;
}
if (imagePaths == null || imagePaths.length == 0) {
return;
}
Pixmap[] pixmaps = new Pixmap[imagePaths.length];
for (int i = 0; i < imagePaths.length; i++) {
pixmaps[i] = Image.createImage(imagePaths[i]).getPixmap();
}
setIcon(win, pixmaps);
for (Pixmap pixmap : pixmaps) {
pixmap.close();
}
}
static void setIcon(long win, Pixmap[] images) {
if (isMacOS()) {
return;
}
if (images == null || images.length == 0) {
return;
}
GLFWImage.Buffer buffer = GLFWImage.malloc(images.length);
Pixmap[] tmpPixmaps = new Pixmap[images.length];
for (int i = 0; i < images.length; i++) {
Pixmap pixmap = images[i];
GLFWImage icon = GLFWImage.malloc();
icon.set(pixmap.getWidth(), pixmap.getHeight(),
pixmap.convertPixmapToByteBuffer());
buffer.put(icon);
icon.free();
}
buffer.position(0);
glfwSetWindowIcon(win, buffer);
buffer.free();
for (Pixmap pixmap : tmpPixmaps) {
if (pixmap != null) {
pixmap.close();
}
}
}
public void reset() {
boolean wasActive = glfwGetWindowAttrib(window, GLFW_VISIBLE) > 0;
while (!glfwWindowShouldClose(window)) {
boolean newActive = glfwGetWindowAttrib(window, GLFW_VISIBLE) > 0;
if (wasActive != newActive) {
status.emit(wasActive ? Status.PAUSE : Status.RESUME);
wasActive = newActive;
}
if (newActive) {
processFrame();
}
glfwPollEvents();
sync.sync(setting.fps);
glfwSwapBuffers(window);
}
((Lwjgl3InputMake) input).shutdown();
((Lwjgl3Graphics) graphics).shutdown();
errorCallback.close();
glfwDestroyWindow(window);
glfwTerminate();
}
}