parent
a694f45af0
commit
e56148559c
@ -0,0 +1,7 @@ |
|||||||
|
<?xml version="1.0" encoding="UTF-8"?> |
||||||
|
<classpath> |
||||||
|
<classpathentry kind="src" path="src"/> |
||||||
|
<classpathentry kind="con" path="org.eclipse.jdt.launching.JRE_CONTAINER/org.eclipse.jdt.internal.debug.ui.launcher.StandardVMType/JavaSE-1.7"/> |
||||||
|
<classpathentry combineaccessrules="false" kind="src" path="/MightyUtils"/> |
||||||
|
<classpathentry kind="output" path="bin"/> |
||||||
|
</classpath> |
@ -1,12 +1,6 @@ |
|||||||
*.class |
/bin/ |
||||||
|
/target/ |
||||||
# Mobile Tools for Java (J2ME) |
/~local/ |
||||||
.mtj.tmp/ |
*.log |
||||||
|
.attach_pid* |
||||||
# Package Files # |
*~ |
||||||
*.jar |
|
||||||
*.war |
|
||||||
*.ear |
|
||||||
|
|
||||||
# virtual machine crash logs, see http://www.java.com/en/download/help/error_hotspot.xml |
|
||||||
hs_err_pid* |
|
||||||
|
@ -0,0 +1,17 @@ |
|||||||
|
<?xml version="1.0" encoding="UTF-8"?> |
||||||
|
<projectDescription> |
||||||
|
<name>GameCore</name> |
||||||
|
<comment></comment> |
||||||
|
<projects> |
||||||
|
</projects> |
||||||
|
<buildSpec> |
||||||
|
<buildCommand> |
||||||
|
<name>org.eclipse.jdt.core.javabuilder</name> |
||||||
|
<arguments> |
||||||
|
</arguments> |
||||||
|
</buildCommand> |
||||||
|
</buildSpec> |
||||||
|
<natures> |
||||||
|
<nature>org.eclipse.jdt.core.javanature</nature> |
||||||
|
</natures> |
||||||
|
</projectDescription> |
@ -0,0 +1,11 @@ |
|||||||
|
eclipse.preferences.version=1 |
||||||
|
org.eclipse.jdt.core.compiler.codegen.inlineJsrBytecode=enabled |
||||||
|
org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.7 |
||||||
|
org.eclipse.jdt.core.compiler.codegen.unusedLocal=preserve |
||||||
|
org.eclipse.jdt.core.compiler.compliance=1.7 |
||||||
|
org.eclipse.jdt.core.compiler.debug.lineNumber=generate |
||||||
|
org.eclipse.jdt.core.compiler.debug.localVariable=generate |
||||||
|
org.eclipse.jdt.core.compiler.debug.sourceFile=generate |
||||||
|
org.eclipse.jdt.core.compiler.problem.assertIdentifier=error |
||||||
|
org.eclipse.jdt.core.compiler.problem.enumIdentifier=error |
||||||
|
org.eclipse.jdt.core.compiler.source=1.7 |
@ -0,0 +1,133 @@ |
|||||||
|
package junk; |
||||||
|
|
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
import java.util.ArrayList; |
||||||
|
import java.util.List; |
||||||
|
import java.util.logging.Level; |
||||||
|
|
||||||
|
import mightypork.gamecore.core.AppBackend; |
||||||
|
import mightypork.gamecore.resources.ResourceSetup; |
||||||
|
import mightypork.gamecore.resources.loading.AsyncResourceLoader; |
||||||
|
import mightypork.gamecore.resources.loading.ResourceLoader; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Init options holder class
|
||||||
|
*/ |
||||||
|
public class AppInitOptions { |
||||||
|
|
||||||
|
String logDir = "log"; |
||||||
|
String logFilePrefix = "runtime"; |
||||||
|
|
||||||
|
String screenshotDir = "screenshots"; |
||||||
|
|
||||||
|
boolean busLogging = false; |
||||||
|
|
||||||
|
String configFile = "settings.cfg"; |
||||||
|
String configComment = "Main config file"; |
||||||
|
|
||||||
|
final List<ResourceSetup> resourceLists = new ArrayList<>(); |
||||||
|
final List<KeySetup> keyLists = new ArrayList<>(); |
||||||
|
final List<ConfigSetup> configLists = new ArrayList<>(); |
||||||
|
|
||||||
|
ResourceLoader resourceLoader = new AsyncResourceLoader(); |
||||||
|
Level logLevel = Level.ALL; |
||||||
|
public boolean sigleInstance = true; |
||||||
|
Level logSoutLevel = Level.ALL; |
||||||
|
|
||||||
|
|
||||||
|
public void setConfigFile(String filename, String comment) |
||||||
|
{ |
||||||
|
configFile = filename; |
||||||
|
configComment = comment; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void addConfig(ConfigSetup cfg) |
||||||
|
{ |
||||||
|
configLists.add(cfg); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void addKeys(KeySetup keys) |
||||||
|
{ |
||||||
|
keyLists.add(keys); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void addResources(ResourceSetup res) |
||||||
|
{ |
||||||
|
resourceLists.add(res); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setBackend(AppBackend backend) |
||||||
|
{ |
||||||
|
this.backend = backend; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set whether to run in single instance mode, or allow multiple instances.<br> |
||||||
|
* Multiple instances running can cause various collisions (eg. when writing |
||||||
|
* config file or logging). |
||||||
|
* |
||||||
|
* @param sigleInstance true to allow only one instance |
||||||
|
*/ |
||||||
|
public void setSigleInstance(boolean sigleInstance) |
||||||
|
{ |
||||||
|
this.sigleInstance = sigleInstance; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set working directory path. If not exists, it will be created. |
||||||
|
* |
||||||
|
* @param workdir work dir path |
||||||
|
*/ |
||||||
|
public void setWorkdir(File workdir) |
||||||
|
{ |
||||||
|
this.workdir = workdir; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setBusLogging(boolean yes) |
||||||
|
{ |
||||||
|
busLogging = yes; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setLogOptions(String logDir, String filePrefix, int archivedCount, Level logLevel) |
||||||
|
{ |
||||||
|
this.logDir = logDir; |
||||||
|
this.logFilePrefix = filePrefix; |
||||||
|
this.logArchiveCount = archivedCount; |
||||||
|
this.logLevel = logLevel; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setResourceLoader(ResourceLoader resLoader) |
||||||
|
{ |
||||||
|
resourceLoader = resLoader; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setScreenshotDir(String path) |
||||||
|
{ |
||||||
|
this.screenshotDir = path; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setLockFile(String lockFile) |
||||||
|
{ |
||||||
|
this.lockFile = lockFile; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setLogLevel(Level logLevel, Level soutLevel) |
||||||
|
{ |
||||||
|
this.logLevel = logLevel; |
||||||
|
this.logSoutLevel = soutLevel; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,184 @@ |
|||||||
|
package junk; |
||||||
|
|
||||||
|
|
||||||
|
import java.lang.Thread.UncaughtExceptionHandler; |
||||||
|
|
||||||
|
import mightypork.gamecore.backends.lwjgl.LwjglInputModule; |
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.core.AppBackend; |
||||||
|
import mightypork.gamecore.core.MainLoop; |
||||||
|
import mightypork.gamecore.core.WorkDir; |
||||||
|
import mightypork.gamecore.core.config.Config; |
||||||
|
import mightypork.gamecore.gui.screens.ScreenRegistry; |
||||||
|
import mightypork.gamecore.gui.screens.impl.CrossfadeOverlay; |
||||||
|
import mightypork.gamecore.resources.Res; |
||||||
|
import mightypork.gamecore.resources.ResourceSetup; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Basic screen-based game with subsystems.<br> |
||||||
|
* This class takes care of the initialization sequence. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class BaseApp extends App implements UncaughtExceptionHandler { |
||||||
|
|
||||||
|
// modules
|
||||||
|
private MainLoop gameLoop; |
||||||
|
private ScreenRegistry screenRegistry; |
||||||
|
|
||||||
|
private boolean started = false; |
||||||
|
private boolean lockObtained = false; |
||||||
|
|
||||||
|
// init opt holder
|
||||||
|
private final AppInitOptions opt = new AppInitOptions(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get init options |
||||||
|
* |
||||||
|
* @return opt holder |
||||||
|
*/ |
||||||
|
public AppInitOptions getInitOptions() |
||||||
|
{ |
||||||
|
if (started) { |
||||||
|
throw new IllegalStateException("Cannot alter init options after starting the App."); |
||||||
|
} |
||||||
|
|
||||||
|
return opt; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public BaseApp(AppBackend backend) { |
||||||
|
super(backend); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Start the application |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public final void start() |
||||||
|
{ |
||||||
|
initialize(); |
||||||
|
|
||||||
|
Log.i("Starting main loop..."); |
||||||
|
|
||||||
|
// open first screen !!!
|
||||||
|
started = true; |
||||||
|
gameLoop.start(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Init the app |
||||||
|
*/ |
||||||
|
protected void initialize() |
||||||
|
{ |
||||||
|
WorkDir.init(opt.workdir); |
||||||
|
|
||||||
|
if (opt.sigleInstance) initLock(); |
||||||
|
lockObtained = true; |
||||||
|
|
||||||
|
for (final RouteSetup rs : opt.routeLists) { |
||||||
|
WorkDir.registerRoutes(rs); |
||||||
|
} |
||||||
|
WorkDir.addPath("_screenshot_dir", opt.screenshotDir); |
||||||
|
|
||||||
|
// apply configurations
|
||||||
|
Config.init(WorkDir.getFile(opt.configFile), opt.configComment); |
||||||
|
|
||||||
|
// add keys to config
|
||||||
|
for (final KeySetup l : opt.keyLists) { |
||||||
|
Config.registerKeys(l); |
||||||
|
} |
||||||
|
|
||||||
|
// add options to config
|
||||||
|
for (final ConfigSetup c : opt.configLists) { |
||||||
|
Config.registerOptions(c); |
||||||
|
} |
||||||
|
Config.load(); |
||||||
|
|
||||||
|
/* |
||||||
|
* Display |
||||||
|
*/ |
||||||
|
Log.f2("Initializing Display System..."); |
||||||
|
initDisplay(gfx()); |
||||||
|
|
||||||
|
/* |
||||||
|
* Audio |
||||||
|
*/ |
||||||
|
Log.f2("Initializing Sound System..."); |
||||||
|
soundSystem = new SoundSystem(this); |
||||||
|
initSoundSystem(soundSystem); |
||||||
|
|
||||||
|
/* |
||||||
|
* Input |
||||||
|
*/ |
||||||
|
Log.f2("Initializing Input System..."); |
||||||
|
inputSystem = new LwjglInputModule(this); |
||||||
|
initInputSystem(inputSystem); |
||||||
|
|
||||||
|
/* |
||||||
|
* Prepare main loop |
||||||
|
*/ |
||||||
|
Log.f1("Creating Screen Registry and Game Loop..."); |
||||||
|
screenRegistry = new ScreenRegistry(this); |
||||||
|
gameLoop = createMainLoop(); |
||||||
|
gameLoop.setRootRenderable(screenRegistry); |
||||||
|
|
||||||
|
/* |
||||||
|
* Load resources |
||||||
|
* |
||||||
|
* Resources should be registered to registries, and AsyncResourceLoader will load them. |
||||||
|
*/ |
||||||
|
Log.f1("Loading resources..."); |
||||||
|
if (opt.resourceLoader != null) { |
||||||
|
opt.resourceLoader.init(this); |
||||||
|
} |
||||||
|
|
||||||
|
Res.init(this); |
||||||
|
|
||||||
|
for (final ResourceSetup rl : opt.resourceLists) { |
||||||
|
Res.load(rl); |
||||||
|
} |
||||||
|
|
||||||
|
/* |
||||||
|
* Screen registry |
||||||
|
* |
||||||
|
* Must be after resources, because screens can request them during instantiation. |
||||||
|
*/ |
||||||
|
Log.f2("Registering screens..."); |
||||||
|
initScreens(screenRegistry); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Register game screens to the registry. |
||||||
|
* |
||||||
|
* @param screens |
||||||
|
*/ |
||||||
|
protected void initScreens(ScreenRegistry screens) |
||||||
|
{ |
||||||
|
screens.addOverlay(new CrossfadeOverlay(this)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Create game loop instance |
||||||
|
* |
||||||
|
* @return the game loop. |
||||||
|
*/ |
||||||
|
protected MainLoop createMainLoop() |
||||||
|
{ |
||||||
|
return new MainLoop(this); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
protected void beforeShutdown() |
||||||
|
{ |
||||||
|
// ???
|
||||||
|
if (lockObtained) Config.save(); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,299 @@ |
|||||||
|
//package junk;
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//import static org.lwjgl.opengl.GL11.*;
|
||||||
|
//
|
||||||
|
//import java.nio.ByteBuffer;
|
||||||
|
//
|
||||||
|
//import mightypork.gamecore.backend.lwjgl.AwtScreenshot;
|
||||||
|
//import mightypork.gamecore.core.modules.AppAccess;
|
||||||
|
//import mightypork.gamecore.core.modules.AppModule;
|
||||||
|
//import mightypork.gamecore.render.events.DisplayReadyEvent;
|
||||||
|
//import mightypork.gamecore.render.events.ViewportChangeEvent;
|
||||||
|
//import mightypork.utils.logging.Log;
|
||||||
|
//import mightypork.utils.math.constraints.rect.Rect;
|
||||||
|
//import mightypork.utils.math.constraints.rect.RectBound;
|
||||||
|
//import mightypork.utils.math.constraints.vect.Vect;
|
||||||
|
//import mightypork.utils.math.timing.FpsMeter;
|
||||||
|
//
|
||||||
|
//import org.lwjgl.BufferUtils;
|
||||||
|
//import org.lwjgl.LWJGLException;
|
||||||
|
//import org.lwjgl.opengl.Display;
|
||||||
|
//import org.lwjgl.opengl.DisplayMode;
|
||||||
|
//
|
||||||
|
//
|
||||||
|
///**
|
||||||
|
// * Display system
|
||||||
|
// *
|
||||||
|
// * @author Ondřej Hruška (MightyPork)
|
||||||
|
// */
|
||||||
|
//@Deprecated
|
||||||
|
//public class DisplaySystem extends AppModule implements RectBound {
|
||||||
|
//
|
||||||
|
// private DisplayMode windowDisplayMode;
|
||||||
|
// private int targetFps;
|
||||||
|
// private FpsMeter fpsMeter;
|
||||||
|
// private boolean fullscreenSwitchRequested;
|
||||||
|
//
|
||||||
|
// /** Current screen size */
|
||||||
|
// private static final Vect screenSize = new Vect() {
|
||||||
|
//
|
||||||
|
// @Override
|
||||||
|
// public double y()
|
||||||
|
// {
|
||||||
|
// return Display.getHeight();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// @Override
|
||||||
|
// public double x()
|
||||||
|
// {
|
||||||
|
// return Display.getWidth();
|
||||||
|
// }
|
||||||
|
// };
|
||||||
|
//
|
||||||
|
// private static final Rect rect = Rect.make(screenSize);
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * @param app app access
|
||||||
|
// */
|
||||||
|
// public DisplaySystem(AppAccess app) {
|
||||||
|
// super(app);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// @Override
|
||||||
|
// protected void deinit()
|
||||||
|
// {
|
||||||
|
// Display.destroy();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Set target fps (for syncing in endFrame() call).<br>
|
||||||
|
// * With vsync enabled, the target fps may not be met.
|
||||||
|
// *
|
||||||
|
// * @param fps requested fps
|
||||||
|
// */
|
||||||
|
// public void setTargetFps(int fps)
|
||||||
|
// {
|
||||||
|
// this.targetFps = fps;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Create a main window
|
||||||
|
// *
|
||||||
|
// * @param width requested width
|
||||||
|
// * @param height requested height
|
||||||
|
// * @param resizable is resizable by the user
|
||||||
|
// * @param fullscreen is in fullscreen
|
||||||
|
// * @param title window title
|
||||||
|
// */
|
||||||
|
// public void createMainWindow(int width, int height, boolean resizable, boolean fullscreen, String title)
|
||||||
|
// {
|
||||||
|
// try {
|
||||||
|
// Display.setDisplayMode(windowDisplayMode = new DisplayMode(width, height));
|
||||||
|
// Display.setResizable(resizable);
|
||||||
|
// Display.setVSyncEnabled(true);
|
||||||
|
// Display.setTitle(title);
|
||||||
|
// Display.create();
|
||||||
|
//
|
||||||
|
// fpsMeter = new FpsMeter();
|
||||||
|
//
|
||||||
|
// if (fullscreen) {
|
||||||
|
// switchFullscreen();
|
||||||
|
// Display.update();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// getEventBus().send(new DisplayReadyEvent());
|
||||||
|
//
|
||||||
|
// } catch (final LWJGLException e) {
|
||||||
|
// throw new RuntimeException("Could not initialize screen", e);
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Toggle FS if possible
|
||||||
|
// */
|
||||||
|
// public void switchFullscreen()
|
||||||
|
// {
|
||||||
|
// fullscreenSwitchRequested = true;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// private void doSwitchFullscreen()
|
||||||
|
// {
|
||||||
|
// try {
|
||||||
|
//
|
||||||
|
// if (!Display.isFullscreen()) {
|
||||||
|
// Log.f3("Entering fullscreen.");
|
||||||
|
// // save window resize
|
||||||
|
// windowDisplayMode = new DisplayMode(Display.getWidth(), Display.getHeight());
|
||||||
|
//
|
||||||
|
// Display.setDisplayMode(Display.getDesktopDisplayMode());
|
||||||
|
// Display.setFullscreen(true);
|
||||||
|
// Display.update();
|
||||||
|
// } else {
|
||||||
|
// Log.f3("Leaving fullscreen.");
|
||||||
|
// Display.setDisplayMode(windowDisplayMode);
|
||||||
|
// Display.update();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// getEventBus().send(new ViewportChangeEvent(getSize()));
|
||||||
|
//
|
||||||
|
// } catch (final Throwable t) {
|
||||||
|
// Log.e("Failed to toggle fullscreen mode.", t);
|
||||||
|
// try {
|
||||||
|
// Display.setDisplayMode(windowDisplayMode);
|
||||||
|
// Display.update();
|
||||||
|
// } catch (final Throwable t1) {
|
||||||
|
// throw new RuntimeException("Failed to revert failed fullscreen toggle.", t1);
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Take screenshot (expensive processing is done on-demand when screenshot
|
||||||
|
// * is processed).
|
||||||
|
// *
|
||||||
|
// * @return screenshot object
|
||||||
|
// */
|
||||||
|
// public static AwtScreenshot prepareScreenshot()
|
||||||
|
// {
|
||||||
|
// glReadBuffer(GL_FRONT);
|
||||||
|
// final int width = Display.getWidth();
|
||||||
|
// final int height = Display.getHeight();
|
||||||
|
// final int bpp = 4;
|
||||||
|
// final ByteBuffer buffer = BufferUtils.createByteBuffer(width * height * bpp);
|
||||||
|
// glReadPixels(0, 0, width, height, GL_RGBA, GL_UNSIGNED_BYTE, buffer);
|
||||||
|
//
|
||||||
|
// final AwtScreenshot sc = new AwtScreenshot(width, height, bpp, buffer);
|
||||||
|
//
|
||||||
|
// return sc;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * @return true if close was requested (i.e. click on cross)
|
||||||
|
// */
|
||||||
|
// public static boolean isCloseRequested()
|
||||||
|
// {
|
||||||
|
// return Display.isCloseRequested();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Get fullscreen state
|
||||||
|
// *
|
||||||
|
// * @return is fullscreen
|
||||||
|
// */
|
||||||
|
// public static boolean isFullscreen()
|
||||||
|
// {
|
||||||
|
// return Display.isFullscreen();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Get screen size. This Vect is final and views at it can safely be made.
|
||||||
|
// *
|
||||||
|
// * @return size
|
||||||
|
// */
|
||||||
|
// public static Vect getSize()
|
||||||
|
// {
|
||||||
|
// return screenSize;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Get screen rect. Static version of getRect().
|
||||||
|
// *
|
||||||
|
// * @return size
|
||||||
|
// */
|
||||||
|
// public static Rect getBounds()
|
||||||
|
// {
|
||||||
|
// return rect;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * @return screen width
|
||||||
|
// */
|
||||||
|
// public static int getWidth()
|
||||||
|
// {
|
||||||
|
// return screenSize.xi();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * @return screen height
|
||||||
|
// */
|
||||||
|
// public static int getHeight()
|
||||||
|
// {
|
||||||
|
// return screenSize.yi();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Start a OpenGL frame
|
||||||
|
// */
|
||||||
|
// public void beginFrame()
|
||||||
|
// {
|
||||||
|
// // handle resize
|
||||||
|
// if (Display.wasResized()) {
|
||||||
|
// getEventBus().send(new ViewportChangeEvent(getSize()));
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// if (fullscreenSwitchRequested) {
|
||||||
|
// fullscreenSwitchRequested = false;
|
||||||
|
// doSwitchFullscreen();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// glLoadIdentity();
|
||||||
|
// glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
|
||||||
|
// fpsMeter.frame();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * End an OpenGL frame, flip buffers, sync to fps.
|
||||||
|
// */
|
||||||
|
// public void endFrame()
|
||||||
|
// {
|
||||||
|
// Display.update(false); // don't poll input devices
|
||||||
|
// Display.sync(targetFps);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Get screen rect. This Rect is final and views at it can safely be made.
|
||||||
|
// */
|
||||||
|
// @Override
|
||||||
|
// public Rect getRect()
|
||||||
|
// {
|
||||||
|
// return getBounds();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * @return current FPS
|
||||||
|
// */
|
||||||
|
// public long getFps()
|
||||||
|
// {
|
||||||
|
// return fpsMeter.getFPS();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Get screen center. This vect is final and views at it can safely be made.
|
||||||
|
// *
|
||||||
|
// * @return screen center.
|
||||||
|
// */
|
||||||
|
// public static Vect getCenter()
|
||||||
|
// {
|
||||||
|
// return rect.center();
|
||||||
|
// }
|
||||||
|
//}
|
@ -0,0 +1,542 @@ |
|||||||
|
//package junk;
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//import static org.lwjgl.opengl.GL11.*;
|
||||||
|
//
|
||||||
|
//import java.io.IOException;
|
||||||
|
//
|
||||||
|
//import mightypork.gamecore.resources.textures.FilterMode;
|
||||||
|
//import mightypork.gamecore.resources.textures.ITexture;
|
||||||
|
//import mightypork.gamecore.resources.textures.TxQuad;
|
||||||
|
//import mightypork.utils.files.FileUtils;
|
||||||
|
//import mightypork.utils.logging.Log;
|
||||||
|
//import mightypork.utils.math.color.Color;
|
||||||
|
//import mightypork.utils.math.color.pal.RGB;
|
||||||
|
//import mightypork.utils.math.constraints.rect.Rect;
|
||||||
|
//import mightypork.utils.math.constraints.rect.caching.RectDigest;
|
||||||
|
//import mightypork.utils.math.constraints.vect.Vect;
|
||||||
|
//import mightypork.utils.math.constraints.vect.VectConst;
|
||||||
|
//
|
||||||
|
//import org.lwjgl.opengl.GL11;
|
||||||
|
//import org.newdawn.slick.opengl.Texture;
|
||||||
|
//import org.newdawn.slick.opengl.TextureLoader;
|
||||||
|
//
|
||||||
|
//
|
||||||
|
///**
|
||||||
|
// * Render utilities
|
||||||
|
// *
|
||||||
|
// * @author Ondřej Hruška (MightyPork)
|
||||||
|
// */
|
||||||
|
//@Deprecated
|
||||||
|
//public class Render {
|
||||||
|
//
|
||||||
|
// public static final VectConst AXIS_X = Vect.make(1, 0, 0);
|
||||||
|
// public static final VectConst AXIS_Y = Vect.make(0, 1, 0);
|
||||||
|
// public static final VectConst AXIS_Z = Vect.make(0, 0, 1);
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Bind GL color
|
||||||
|
// *
|
||||||
|
// * @param color Color color
|
||||||
|
// */
|
||||||
|
// public static void setColor(Color color)
|
||||||
|
// {
|
||||||
|
// if (color != null) glColor4d(color.r(), color.g(), color.b(), color.a());
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Bind GL color
|
||||||
|
// *
|
||||||
|
// * @param color Color color
|
||||||
|
// * @param alpha alpha multiplier
|
||||||
|
// */
|
||||||
|
// public static void setColor(Color color, double alpha)
|
||||||
|
// {
|
||||||
|
// if (color != null) glColor4d(color.r(), color.g(), color.b(), color.a() * alpha);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Translate
|
||||||
|
// *
|
||||||
|
// * @param x
|
||||||
|
// * @param y
|
||||||
|
// */
|
||||||
|
// public static void translate(double x, double y)
|
||||||
|
// {
|
||||||
|
// glTranslated(x, y, 0);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Translate
|
||||||
|
// *
|
||||||
|
// * @param x
|
||||||
|
// * @param y
|
||||||
|
// * @param z
|
||||||
|
// */
|
||||||
|
// public static void translate(double x, double y, double z)
|
||||||
|
// {
|
||||||
|
// glTranslated(x, y, z);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Translate with coord
|
||||||
|
// *
|
||||||
|
// * @param coord coord
|
||||||
|
// */
|
||||||
|
// public static void translate(Vect coord)
|
||||||
|
// {
|
||||||
|
// glTranslated(coord.x(), coord.y(), coord.z());
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Translate with coord, discard Z
|
||||||
|
// *
|
||||||
|
// * @param coord coord
|
||||||
|
// */
|
||||||
|
// public static void translateXY(Vect coord)
|
||||||
|
// {
|
||||||
|
// glTranslated(coord.x(), coord.y(), 0);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Scale
|
||||||
|
// *
|
||||||
|
// * @param x
|
||||||
|
// * @param y
|
||||||
|
// */
|
||||||
|
// public static void scale(double x, double y)
|
||||||
|
// {
|
||||||
|
// glScaled(x, y, 0);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Scale
|
||||||
|
// *
|
||||||
|
// * @param x
|
||||||
|
// * @param y
|
||||||
|
// * @param z
|
||||||
|
// */
|
||||||
|
// public static void scale(double x, double y, double z)
|
||||||
|
// {
|
||||||
|
// glScaled(x, y, z);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Scale
|
||||||
|
// *
|
||||||
|
// * @param factor vector of scaling factors
|
||||||
|
// */
|
||||||
|
// public static void scale(Vect factor)
|
||||||
|
// {
|
||||||
|
// glScaled(factor.x(), factor.y(), factor.z());
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Scale by X factor
|
||||||
|
// *
|
||||||
|
// * @param factor scaling factor
|
||||||
|
// */
|
||||||
|
// public static void scaleXY(double factor)
|
||||||
|
// {
|
||||||
|
// glScaled(factor, factor, 1);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Scale by X factor
|
||||||
|
// *
|
||||||
|
// * @param factor scaling factor
|
||||||
|
// */
|
||||||
|
// public static void scaleX(double factor)
|
||||||
|
// {
|
||||||
|
// glScaled(factor, 1, 1);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Scale by Y factor
|
||||||
|
// *
|
||||||
|
// * @param factor scaling factor
|
||||||
|
// */
|
||||||
|
// public static void scaleY(double factor)
|
||||||
|
// {
|
||||||
|
// glScaled(1, factor, 1);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Scale by Z factor
|
||||||
|
// *
|
||||||
|
// * @param factor scaling factor
|
||||||
|
// */
|
||||||
|
// public static void scaleZ(double factor)
|
||||||
|
// {
|
||||||
|
// glScaled(1, 1, factor);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Rotate around X axis
|
||||||
|
// *
|
||||||
|
// * @param angle deg
|
||||||
|
// */
|
||||||
|
// public static void rotateX(double angle)
|
||||||
|
// {
|
||||||
|
// rotate(angle, AXIS_X);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Rotate around Y axis
|
||||||
|
// *
|
||||||
|
// * @param angle deg
|
||||||
|
// */
|
||||||
|
// public static void rotateY(double angle)
|
||||||
|
// {
|
||||||
|
// rotate(angle, AXIS_Y);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Rotate around Z axis
|
||||||
|
// *
|
||||||
|
// * @param angle deg
|
||||||
|
// */
|
||||||
|
// public static void rotateZ(double angle)
|
||||||
|
// {
|
||||||
|
// rotate(angle, AXIS_Z);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Rotate
|
||||||
|
// *
|
||||||
|
// * @param angle rotate angle
|
||||||
|
// * @param axis rotation axis
|
||||||
|
// */
|
||||||
|
// public static void rotate(double angle, Vect axis)
|
||||||
|
// {
|
||||||
|
// final Vect vec = axis.norm(1);
|
||||||
|
// glRotated(angle, vec.x(), vec.y(), vec.z());
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// private static int pushed = 0;
|
||||||
|
// /** Can be used to avoid texture binding and glBegin/glEnd in textured quads */
|
||||||
|
// public static boolean batchTexturedQuadMode;
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Store GL state
|
||||||
|
// */
|
||||||
|
// public static void pushState()
|
||||||
|
// {
|
||||||
|
// pushed++;
|
||||||
|
//
|
||||||
|
// if (pushed >= 100) {
|
||||||
|
// Log.w("Suspicious number of state pushes: " + pushed);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// GL11.glPushAttrib(GL11.GL_ALL_ATTRIB_BITS);
|
||||||
|
// GL11.glPushClientAttrib(GL11.GL_ALL_CLIENT_ATTRIB_BITS);
|
||||||
|
// GL11.glMatrixMode(GL11.GL_MODELVIEW);
|
||||||
|
// GL11.glPushMatrix();
|
||||||
|
// GL11.glMatrixMode(GL11.GL_PROJECTION);
|
||||||
|
// GL11.glPushMatrix();
|
||||||
|
// GL11.glMatrixMode(GL11.GL_MODELVIEW);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Restore Gl state
|
||||||
|
// */
|
||||||
|
// public static void popState()
|
||||||
|
// {
|
||||||
|
// if (pushed == 0) {
|
||||||
|
// Log.w("Pop without push.");
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// pushed--;
|
||||||
|
//
|
||||||
|
// GL11.glMatrixMode(GL11.GL_PROJECTION);
|
||||||
|
// GL11.glPopMatrix();
|
||||||
|
// GL11.glMatrixMode(GL11.GL_MODELVIEW);
|
||||||
|
// GL11.glPopMatrix();
|
||||||
|
// GL11.glPopClientAttrib();
|
||||||
|
// GL11.glPopAttrib();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Store matrix
|
||||||
|
// */
|
||||||
|
// public static void pushMatrix()
|
||||||
|
// {
|
||||||
|
// GL11.glPushMatrix();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Restore Gl state
|
||||||
|
// */
|
||||||
|
// public static void popMatrix()
|
||||||
|
// {
|
||||||
|
// GL11.glPopMatrix();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Load texture
|
||||||
|
// *
|
||||||
|
// * @param resourcePath
|
||||||
|
// * @param filtering filtering mode to use while loading.
|
||||||
|
// * @return the loaded texture
|
||||||
|
// */
|
||||||
|
// public synchronized static Texture loadSlickTexture(String resourcePath, FilterMode filtering)
|
||||||
|
// {
|
||||||
|
//
|
||||||
|
// try {
|
||||||
|
//
|
||||||
|
// final String ext = FileUtils.getExtension(resourcePath).toUpperCase();
|
||||||
|
//
|
||||||
|
// final Texture texture = TextureLoader.getTexture(ext, FileUtils.getResource(resourcePath), false, filtering.num);
|
||||||
|
//
|
||||||
|
// if (texture == null) {
|
||||||
|
// Log.w("Texture " + resourcePath + " could not be loaded.");
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// return texture;
|
||||||
|
//
|
||||||
|
// } catch (final IOException e) {
|
||||||
|
// Log.e("Loading of texture " + resourcePath + " failed.", e);
|
||||||
|
// throw new RuntimeException("Could not load texture " + resourcePath + ".", e);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Render quad 2D
|
||||||
|
// *
|
||||||
|
// * @param rect rectangle
|
||||||
|
// * @param color draw color
|
||||||
|
// */
|
||||||
|
// public static void quad(Rect rect, Color color)
|
||||||
|
// {
|
||||||
|
// setColor(color);
|
||||||
|
// quad(rect);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Render quad
|
||||||
|
// *
|
||||||
|
// * @param quad the quad to draw (px)
|
||||||
|
// */
|
||||||
|
// public static void quad(Rect quad)
|
||||||
|
// {
|
||||||
|
// final RectDigest q = quad.digest();
|
||||||
|
//
|
||||||
|
// // draw with color
|
||||||
|
//
|
||||||
|
// glDisable(GL_TEXTURE_2D);
|
||||||
|
//
|
||||||
|
// // quad
|
||||||
|
// glBegin(GL_QUADS);
|
||||||
|
// glVertex2d(q.left, q.bottom);
|
||||||
|
// glVertex2d(q.right, q.bottom);
|
||||||
|
// glVertex2d(q.right, q.top);
|
||||||
|
// glVertex2d(q.left, q.top);
|
||||||
|
// glEnd();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Draw quad with horizontal gradient
|
||||||
|
// *
|
||||||
|
// * @param quad drawn quad bounds
|
||||||
|
// * @param color1 left color
|
||||||
|
// * @param color2 right color
|
||||||
|
// */
|
||||||
|
// public static void quadGradH(Rect quad, Color color1, Color color2)
|
||||||
|
// {
|
||||||
|
// quadColor(quad, color1, color2, color2, color1);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// public static void quadColor(Rect quad, Color color)
|
||||||
|
// {
|
||||||
|
// quadColor(quad, color, color, color, color);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Draw quad with coloured vertices.
|
||||||
|
// *
|
||||||
|
// * @param quad drawn quad bounds
|
||||||
|
// * @param colorHMinVMin
|
||||||
|
// * @param colorHMaxVMin
|
||||||
|
// * @param colorHMaxVMax
|
||||||
|
// * @param colorHMinVMax
|
||||||
|
// */
|
||||||
|
// public static void quadColor(Rect quad, Color colorHMinVMin, Color colorHMaxVMin, Color colorHMaxVMax, Color colorHMinVMax)
|
||||||
|
// {
|
||||||
|
// final RectDigest r = quad.digest();
|
||||||
|
//
|
||||||
|
// // draw with color
|
||||||
|
//
|
||||||
|
// glDisable(GL_TEXTURE_2D);
|
||||||
|
//
|
||||||
|
// glBegin(GL_QUADS);
|
||||||
|
// setColor(colorHMinVMax);
|
||||||
|
// glVertex2d(r.left, r.bottom);
|
||||||
|
//
|
||||||
|
// setColor(colorHMaxVMax);
|
||||||
|
// glVertex2d(r.right, r.bottom);
|
||||||
|
//
|
||||||
|
// setColor(colorHMaxVMin);
|
||||||
|
// glVertex2d(r.right, r.top);
|
||||||
|
//
|
||||||
|
// setColor(colorHMinVMin);
|
||||||
|
// glVertex2d(r.left, r.top);
|
||||||
|
// glEnd();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Draw quad with vertical gradient
|
||||||
|
// *
|
||||||
|
// * @param quad drawn quad bounds
|
||||||
|
// * @param color1 top color
|
||||||
|
// * @param color2 bottom color
|
||||||
|
// */
|
||||||
|
// public static void quadGradV(Rect quad, Color color1, Color color2)
|
||||||
|
// {
|
||||||
|
// quadColor(quad, color1, color1, color2, color2);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Render textured rect
|
||||||
|
// *
|
||||||
|
// * @param quad rectangle (px)
|
||||||
|
// * @param txquad texture quad
|
||||||
|
// */
|
||||||
|
// public static void quadTextured(Rect quad, TxQuad txquad)
|
||||||
|
// {
|
||||||
|
// quadTextured(quad, txquad, RGB.WHITE);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Render textured rect
|
||||||
|
// *
|
||||||
|
// * @param quad rectangle (px)
|
||||||
|
// * @param txquad texture instance
|
||||||
|
// * @param tint color tint
|
||||||
|
// */
|
||||||
|
// public static void quadTextured(Rect quad, TxQuad txquad, Color tint)
|
||||||
|
// {
|
||||||
|
// if (!batchTexturedQuadMode) {
|
||||||
|
// glEnable(GL_TEXTURE_2D);
|
||||||
|
// txquad.tx.bind();
|
||||||
|
// glBegin(GL_QUADS);
|
||||||
|
// setColor(tint);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// final RectDigest q = quad.digest();
|
||||||
|
// final RectDigest u = txquad.uvs.digest();
|
||||||
|
//
|
||||||
|
// final double offs = 0.0001;// hack to avoid white stitching
|
||||||
|
//
|
||||||
|
// double tL = u.left + offs, tR = u.right - offs, tT = u.top + offs, tB = u.bottom - offs;
|
||||||
|
//
|
||||||
|
// // handle flip
|
||||||
|
// if (txquad.isFlippedY()) {
|
||||||
|
// final double swap = tT;
|
||||||
|
// tT = tB;
|
||||||
|
// tB = swap;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// if (txquad.isFlippedX()) {
|
||||||
|
// final double swap = tL;
|
||||||
|
// tL = tR;
|
||||||
|
// tR = swap;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// final double w = txquad.tx.getWidth01();
|
||||||
|
// final double h = txquad.tx.getHeight01();
|
||||||
|
//
|
||||||
|
// // quad with texture
|
||||||
|
// glTexCoord2d(tL * w, tB * h);
|
||||||
|
// glVertex2d(q.left, q.bottom);
|
||||||
|
//
|
||||||
|
// glTexCoord2d(tR * w, tB * h);
|
||||||
|
// glVertex2d(q.right, q.bottom);
|
||||||
|
//
|
||||||
|
// glTexCoord2d(tR * w, tT * h);
|
||||||
|
// glVertex2d(q.right, q.top);
|
||||||
|
//
|
||||||
|
// glTexCoord2d(tL * w, tT * h);
|
||||||
|
// glVertex2d(q.left, q.top);
|
||||||
|
//
|
||||||
|
// if (!batchTexturedQuadMode) glEnd();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Setup Ortho projection for 2D graphics
|
||||||
|
// *
|
||||||
|
// * @param size viewport size (screen size)
|
||||||
|
// */
|
||||||
|
// public static void setupOrtho(Vect size)
|
||||||
|
// {
|
||||||
|
// // fix projection for changed size
|
||||||
|
// glMatrixMode(GL_PROJECTION);
|
||||||
|
// glLoadIdentity();
|
||||||
|
// glViewport(0, 0, size.xi(), size.yi());
|
||||||
|
// glOrtho(0, size.xi(), size.yi(), 0, -1000, 1000);
|
||||||
|
//
|
||||||
|
// // back to modelview
|
||||||
|
// glMatrixMode(GL_MODELVIEW);
|
||||||
|
//
|
||||||
|
// glLoadIdentity();
|
||||||
|
//
|
||||||
|
// glDisable(GL_LIGHTING);
|
||||||
|
//
|
||||||
|
// glClearDepth(1f);
|
||||||
|
// glEnable(GL_DEPTH_TEST);
|
||||||
|
// glDepthFunc(GL_LEQUAL);
|
||||||
|
//
|
||||||
|
// glEnable(GL_NORMALIZE);
|
||||||
|
//
|
||||||
|
// glShadeModel(GL_SMOOTH);
|
||||||
|
//
|
||||||
|
// glEnable(GL_BLEND);
|
||||||
|
// glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// public static void enterBatchTexturedQuadMode(ITexture texture)
|
||||||
|
// {
|
||||||
|
// texture.bind();
|
||||||
|
// glBegin(GL11.GL_QUADS);
|
||||||
|
// batchTexturedQuadMode = true;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// public static void leaveBatchTexturedQuadMode()
|
||||||
|
// {
|
||||||
|
// glEnd();
|
||||||
|
// batchTexturedQuadMode = false;
|
||||||
|
// }
|
||||||
|
//}
|
@ -0,0 +1,261 @@ |
|||||||
|
package junk; |
||||||
|
|
||||||
|
|
||||||
|
//package mightypork.gamecore.resources.audio;
|
||||||
|
//
|
||||||
|
//
|
||||||
|
//import java.nio.FloatBuffer;
|
||||||
|
//import java.util.ArrayList;
|
||||||
|
//import java.util.List;
|
||||||
|
//
|
||||||
|
//import mightypork.gamecore.backend.lwjgl.SlickAudio;
|
||||||
|
//import mightypork.gamecore.core.modules.App;
|
||||||
|
//import mightypork.gamecore.resources.ResourceLoadRequest;
|
||||||
|
//import mightypork.gamecore.resources.audio.players.EffectPlayer;
|
||||||
|
//import mightypork.gamecore.resources.audio.players.LoopPlayer;
|
||||||
|
//import mightypork.gamecore.util.BufferHelper;
|
||||||
|
//import mightypork.utils.eventbus.clients.BusNode;
|
||||||
|
//import mightypork.utils.interfaces.Destroyable;
|
||||||
|
//import mightypork.utils.interfaces.Updateable;
|
||||||
|
//import mightypork.utils.logging.Log;
|
||||||
|
//import mightypork.utils.math.constraints.vect.Vect;
|
||||||
|
//import mightypork.utils.math.constraints.vect.var.VectVar;
|
||||||
|
//
|
||||||
|
//import org.lwjgl.openal.AL;
|
||||||
|
//import org.lwjgl.openal.AL10;
|
||||||
|
//import org.newdawn.slick.openal.SoundStore;
|
||||||
|
//
|
||||||
|
//
|
||||||
|
///**
|
||||||
|
// * Sound system class (only one instance should be made per application)
|
||||||
|
// *
|
||||||
|
// * @author Ondřej Hruška (MightyPork)
|
||||||
|
// */
|
||||||
|
//@Deprecated
|
||||||
|
//public class SoundSystem extends BusNode implements Updateable, Destroyable {
|
||||||
|
//
|
||||||
|
// private static final Vect INITIAL_LISTENER_POS = Vect.ZERO;
|
||||||
|
// private static final int MAX_SOURCES = 256;
|
||||||
|
//
|
||||||
|
// private static VectVar listener = Vect.makeVar();
|
||||||
|
// private static boolean soundSystemInited = false;
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Set listener pos
|
||||||
|
// *
|
||||||
|
// * @param pos
|
||||||
|
// */
|
||||||
|
// public static void setListener(Vect pos)
|
||||||
|
// {
|
||||||
|
// listener.setTo(pos);
|
||||||
|
// final FloatBuffer buf3 = BufferHelper.alloc(3);
|
||||||
|
// final FloatBuffer buf6 = BufferHelper.alloc(6);
|
||||||
|
// buf3.clear();
|
||||||
|
// BufferHelper.fill(buf3, (float) pos.x(), (float) pos.y(), (float) pos.z());
|
||||||
|
// AL10.alListener(AL10.AL_POSITION, buf3);
|
||||||
|
// buf3.clear();
|
||||||
|
// BufferHelper.fill(buf3, 0, 0, 0);
|
||||||
|
// AL10.alListener(AL10.AL_VELOCITY, buf3);
|
||||||
|
// buf6.clear();
|
||||||
|
// BufferHelper.fill(buf6, 0.0f, 0.0f, -1.0f, 0.0f, 1.0f, 0.0f);
|
||||||
|
// AL10.alListener(AL10.AL_ORIENTATION, buf6);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * @return listener coordinate
|
||||||
|
// */
|
||||||
|
// public static Vect getListener()
|
||||||
|
// {
|
||||||
|
// return listener;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// // -- instance --
|
||||||
|
//
|
||||||
|
// private final Volume masterVolume = new Volume(1D);
|
||||||
|
// private final Volume effectsVolume = new JointVolume(masterVolume);
|
||||||
|
// private final Volume loopsVolume = new JointVolume(masterVolume);
|
||||||
|
//
|
||||||
|
// private final List<LoopPlayer> loopPlayers = new ArrayList<>();
|
||||||
|
// private final List<DeferredAudio> resources = new ArrayList<>();
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * @param busAccess app access
|
||||||
|
// */
|
||||||
|
// public SoundSystem() {
|
||||||
|
//
|
||||||
|
// if (!soundSystemInited) {
|
||||||
|
// soundSystemInited = true;
|
||||||
|
//
|
||||||
|
// try {
|
||||||
|
// SoundStore.get().setMaxSources(MAX_SOURCES);
|
||||||
|
// SoundStore.get().init();
|
||||||
|
// setListener(INITIAL_LISTENER_POS);
|
||||||
|
//
|
||||||
|
// App.bus().send(new AudioReadyEvent());
|
||||||
|
// } catch (final Throwable t) {
|
||||||
|
// Log.e("Error initializing sound system.", t);
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// @Override
|
||||||
|
// public void destroy()
|
||||||
|
// {
|
||||||
|
// for (final DeferredAudio r : resources) {
|
||||||
|
// r.destroy();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// SoundStore.get().clear();
|
||||||
|
// AL.destroy();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// @Override
|
||||||
|
// public void update(double delta)
|
||||||
|
// {
|
||||||
|
// for (final Updateable lp : loopPlayers) {
|
||||||
|
// lp.update(delta);
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Create effect resource
|
||||||
|
// *
|
||||||
|
// * @param resource resource path
|
||||||
|
// * @param pitch default pitch (1 = unchanged)
|
||||||
|
// * @param gain default gain (0-1)
|
||||||
|
// * @return player
|
||||||
|
// */
|
||||||
|
// public EffectPlayer createEffect(String resource, double pitch, double gain)
|
||||||
|
// {
|
||||||
|
// return new EffectPlayer(createResource(resource), pitch, gain, effectsVolume);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Register loop resource (music / effect loop)
|
||||||
|
// *
|
||||||
|
// * @param resource resource path
|
||||||
|
// * @param pitch default pitch (1 = unchanged)
|
||||||
|
// * @param gain default gain (0-1)
|
||||||
|
// * @param fadeIn default time for fadeIn
|
||||||
|
// * @param fadeOut default time for fadeOut
|
||||||
|
// * @return player
|
||||||
|
// */
|
||||||
|
// public LoopPlayer createLoop(String resource, double pitch, double gain, double fadeIn, double fadeOut)
|
||||||
|
// {
|
||||||
|
// final LoopPlayer p = new LoopPlayer(createResource(resource), pitch, gain, loopsVolume);
|
||||||
|
// p.setFadeTimes(fadeIn, fadeOut);
|
||||||
|
// loopPlayers.add(p);
|
||||||
|
// return p;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Create {@link DeferredAudio} for a resource
|
||||||
|
// *
|
||||||
|
// * @param res a resource name
|
||||||
|
// * @return the resource
|
||||||
|
// * @throws IllegalArgumentException if resource is already registered
|
||||||
|
// */
|
||||||
|
// private DeferredAudio createResource(String res)
|
||||||
|
// {
|
||||||
|
// final DeferredAudio a = new SlickAudio(res);
|
||||||
|
// App.bus().send(new ResourceLoadRequest(a));
|
||||||
|
// resources.add(a);
|
||||||
|
// return a;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Fade out all loops (ie. for screen transitions)
|
||||||
|
// */
|
||||||
|
// public void fadeOutAllLoops()
|
||||||
|
// {
|
||||||
|
// for (final LoopPlayer p : loopPlayers) {
|
||||||
|
// p.fadeOut();
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Pause all loops (leave volume unchanged)
|
||||||
|
// */
|
||||||
|
// public void pauseAllLoops()
|
||||||
|
// {
|
||||||
|
// for (final LoopPlayer p : loopPlayers) {
|
||||||
|
// p.pause();
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Set level of master volume
|
||||||
|
// *
|
||||||
|
// * @param d level
|
||||||
|
// */
|
||||||
|
// public void setMasterVolume(double d)
|
||||||
|
// {
|
||||||
|
// masterVolume.set(d);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Set level of effects volume
|
||||||
|
// *
|
||||||
|
// * @param d level
|
||||||
|
// */
|
||||||
|
// public void setEffectsVolume(double d)
|
||||||
|
// {
|
||||||
|
// effectsVolume.set(d);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Set level of music volume
|
||||||
|
// *
|
||||||
|
// * @param d level
|
||||||
|
// */
|
||||||
|
// public void setMusicVolume(double d)
|
||||||
|
// {
|
||||||
|
// loopsVolume.set(d);
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Get level of master volume
|
||||||
|
// *
|
||||||
|
// * @return level
|
||||||
|
// */
|
||||||
|
// public double getMasterVolume()
|
||||||
|
// {
|
||||||
|
// return masterVolume.get();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Get level of effects volume
|
||||||
|
// *
|
||||||
|
// * @return level
|
||||||
|
// */
|
||||||
|
// public double getEffectsVolume()
|
||||||
|
// {
|
||||||
|
// return effectsVolume.get();
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// /**
|
||||||
|
// * Get level of music volume
|
||||||
|
// *
|
||||||
|
// * @return level
|
||||||
|
// */
|
||||||
|
// public double getMusicVolume()
|
||||||
|
// {
|
||||||
|
// return loopsVolume.get();
|
||||||
|
// }
|
||||||
|
//}
|
@ -0,0 +1,220 @@ |
|||||||
|
package mightypork.gamecore.audio; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.ArrayList; |
||||||
|
import java.util.List; |
||||||
|
|
||||||
|
import mightypork.gamecore.audio.players.EffectPlayer; |
||||||
|
import mightypork.gamecore.audio.players.LoopPlayer; |
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.core.BackendModule; |
||||||
|
import mightypork.gamecore.resources.loading.ResourceLoadRequest; |
||||||
|
import mightypork.utils.interfaces.Updateable; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Abstract audio module. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class AudioModule extends BackendModule implements Updateable { |
||||||
|
|
||||||
|
/** |
||||||
|
* Set listener position |
||||||
|
* |
||||||
|
* @param pos listener position |
||||||
|
*/ |
||||||
|
public abstract void setListenerPos(Vect pos); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get current listener position |
||||||
|
* |
||||||
|
* @return listener position |
||||||
|
*/ |
||||||
|
public abstract Vect getListenerPos(); |
||||||
|
|
||||||
|
// -- instance --
|
||||||
|
|
||||||
|
private final Volume masterVolume = new Volume(1D); |
||||||
|
private final Volume effectsVolume = new JointVolume(masterVolume); |
||||||
|
private final Volume loopsVolume = new JointVolume(masterVolume); |
||||||
|
|
||||||
|
private final List<LoopPlayer> loopPlayers = new ArrayList<>(); |
||||||
|
private final List<DeferredAudio> resources = new ArrayList<>(); |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void destroy() |
||||||
|
{ |
||||||
|
for (final DeferredAudio r : resources) { |
||||||
|
r.destroy(); |
||||||
|
} |
||||||
|
|
||||||
|
deinitSoundSystem(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Deinitialize the soud system, release resources etc.<br> |
||||||
|
* Audio resources are already destroyed. |
||||||
|
*/ |
||||||
|
protected abstract void deinitSoundSystem(); |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void update(double delta) |
||||||
|
{ |
||||||
|
for (final Updateable lp : loopPlayers) { |
||||||
|
lp.update(delta); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Create effect resource |
||||||
|
* |
||||||
|
* @param resource resource path |
||||||
|
* @param pitch default pitch (1 = unchanged) |
||||||
|
* @param gain default gain (0-1) |
||||||
|
* @return player |
||||||
|
*/ |
||||||
|
public EffectPlayer createEffect(String resource, double pitch, double gain) |
||||||
|
{ |
||||||
|
return new EffectPlayer(createResource(resource), pitch, gain, effectsVolume); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Register loop resource (music / effect loop) |
||||||
|
* |
||||||
|
* @param resource resource path |
||||||
|
* @param pitch default pitch (1 = unchanged) |
||||||
|
* @param gain default gain (0-1) |
||||||
|
* @param fadeIn default time for fadeIn |
||||||
|
* @param fadeOut default time for fadeOut |
||||||
|
* @return player |
||||||
|
*/ |
||||||
|
public LoopPlayer createLoop(String resource, double pitch, double gain, double fadeIn, double fadeOut) |
||||||
|
{ |
||||||
|
final LoopPlayer p = new LoopPlayer(createResource(resource), pitch, gain, loopsVolume); |
||||||
|
p.setFadeTimes(fadeIn, fadeOut); |
||||||
|
loopPlayers.add(p); |
||||||
|
return p; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Create {@link DeferredAudio} for a resource, request deferred load and |
||||||
|
* add to the resources list. |
||||||
|
* |
||||||
|
* @param res a resource name |
||||||
|
* @return the resource |
||||||
|
* @throws IllegalArgumentException if resource is already registered |
||||||
|
*/ |
||||||
|
protected DeferredAudio createResource(String res) |
||||||
|
{ |
||||||
|
final DeferredAudio a = doCreateResource(res); |
||||||
|
App.bus().send(new ResourceLoadRequest(a)); |
||||||
|
resources.add(a); |
||||||
|
return a; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Create a backend-specific deferred audio resource |
||||||
|
* |
||||||
|
* @param res resource path |
||||||
|
* @return Deferred Audio |
||||||
|
*/ |
||||||
|
protected abstract DeferredAudio doCreateResource(String res); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Fade out all loops (= fade out the currently playing loops) |
||||||
|
*/ |
||||||
|
public void fadeOutAllLoops() |
||||||
|
{ |
||||||
|
for (final LoopPlayer p : loopPlayers) { |
||||||
|
p.fadeOut(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Pause all loops (leave volume unchanged) |
||||||
|
*/ |
||||||
|
public void pauseAllLoops() |
||||||
|
{ |
||||||
|
for (final LoopPlayer p : loopPlayers) { |
||||||
|
p.pause(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set level of master volume (volume multiplier) |
||||||
|
* |
||||||
|
* @param volume level (0..1) |
||||||
|
*/ |
||||||
|
public void setMasterVolume(double volume) |
||||||
|
{ |
||||||
|
masterVolume.set(volume); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set level of effects volume (volume multiplier) |
||||||
|
* |
||||||
|
* @param volume level (0..1) |
||||||
|
*/ |
||||||
|
public void setEffectsVolume(double volume) |
||||||
|
{ |
||||||
|
effectsVolume.set(volume); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set level of loops volume (volume multiplier) |
||||||
|
* |
||||||
|
* @param volume level (0..1) |
||||||
|
*/ |
||||||
|
public void setLoopsVolume(double volume) |
||||||
|
{ |
||||||
|
loopsVolume.set(volume); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get level of master volume (volume multiplier) |
||||||
|
* |
||||||
|
* @return level (0..1) |
||||||
|
*/ |
||||||
|
public double getMasterVolume() |
||||||
|
{ |
||||||
|
return masterVolume.get(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get level of effects volume (volume multiplier) |
||||||
|
* |
||||||
|
* @return level (0..1) |
||||||
|
*/ |
||||||
|
public double getEffectsVolume() |
||||||
|
{ |
||||||
|
return effectsVolume.get(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get level of loops volume (volume multiplier) |
||||||
|
* |
||||||
|
* @return level (0..1) |
||||||
|
*/ |
||||||
|
public double getLoopsVolume() |
||||||
|
{ |
||||||
|
return loopsVolume.get(); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,47 @@ |
|||||||
|
package mightypork.gamecore.audio; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.resources.BaseDeferredResource; |
||||||
|
import mightypork.utils.annotations.Alias; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Abstract deferred audio, to be extended in backend. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
@Alias(name = "Audio") |
||||||
|
public abstract class DeferredAudio extends BaseDeferredResource implements IAudio { |
||||||
|
|
||||||
|
/** |
||||||
|
* Create audio |
||||||
|
* |
||||||
|
* @param resourceName resource to load (when needed) |
||||||
|
*/ |
||||||
|
public DeferredAudio(String resourceName) { |
||||||
|
super(resourceName); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void play(double pitch, double gain, boolean loop) |
||||||
|
{ |
||||||
|
play(pitch, gain, loop, App.audio().getListenerPos()); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void play(double pitch, double gain, boolean loop, double x, double y) |
||||||
|
{ |
||||||
|
play(pitch, gain, loop, x, y, App.audio().getListenerPos().z()); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void play(double pitch, double gain, boolean loop, Vect pos) |
||||||
|
{ |
||||||
|
play(pitch, gain, loop, pos.x(), pos.y(), pos.z()); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,99 @@ |
|||||||
|
package mightypork.gamecore.audio; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.interfaces.Destroyable; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Audio resource interface (backend independent) |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public interface IAudio extends Destroyable { |
||||||
|
|
||||||
|
/** |
||||||
|
* Pause loop (remember position and stop playing) - if was looping |
||||||
|
*/ |
||||||
|
void pauseLoop(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Resume loop (if was paused) |
||||||
|
*/ |
||||||
|
void resumeLoop(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Adjust gain for the currently playing effect (can be used for fading |
||||||
|
* music) |
||||||
|
* |
||||||
|
* @param gain gain to set 0..1 |
||||||
|
*/ |
||||||
|
void adjustGain(double gain); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Stop audio playback |
||||||
|
*/ |
||||||
|
void stop(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return true if the audio is playing |
||||||
|
*/ |
||||||
|
boolean isPlaying(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return trie if the audio is paused |
||||||
|
*/ |
||||||
|
boolean isPaused(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Play as sound effect at listener position |
||||||
|
* |
||||||
|
* @param pitch pitch (1 = default) |
||||||
|
* @param gain gain (0-1) |
||||||
|
* @param loop looping |
||||||
|
*/ |
||||||
|
void play(double pitch, double gain, boolean loop); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Play as sound effect at given X-Y position |
||||||
|
* |
||||||
|
* @param pitch pitch (1 = default) |
||||||
|
* @param gain gain (0-1) |
||||||
|
* @param loop looping |
||||||
|
* @param x |
||||||
|
* @param y |
||||||
|
*/ |
||||||
|
void play(double pitch, double gain, boolean loop, double x, double y); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Play as sound effect at given position |
||||||
|
* |
||||||
|
* @param pitch pitch (1 = default) |
||||||
|
* @param gain gain (0-1) |
||||||
|
* @param loop looping |
||||||
|
* @param x |
||||||
|
* @param y |
||||||
|
* @param z |
||||||
|
*/ |
||||||
|
void play(double pitch, double gain, boolean loop, double x, double y, double z); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Play as sound effect at given position |
||||||
|
* |
||||||
|
* @param pitch pitch (1 = default) |
||||||
|
* @param gain gain (0-1) |
||||||
|
* @param loop looping |
||||||
|
* @param pos coord |
||||||
|
*/ |
||||||
|
void play(double pitch, double gain, boolean loop, Vect pos); |
||||||
|
|
||||||
|
} |
@ -0,0 +1,51 @@ |
|||||||
|
package mightypork.gamecore.audio; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.math.Calc; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Volume combined of multiple volumes, combining them (multiplication). |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class JointVolume extends Volume { |
||||||
|
|
||||||
|
private final Volume[] volumes; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Create joint volume with master gain of 1 |
||||||
|
* |
||||||
|
* @param volumes individual volumes to join |
||||||
|
*/ |
||||||
|
@SafeVarargs |
||||||
|
public JointVolume(Volume... volumes) { |
||||||
|
super(1D); |
||||||
|
this.volumes = volumes; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get combined gain (multiplied) |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public Double get() |
||||||
|
{ |
||||||
|
double d = super.get(); |
||||||
|
for (final Volume v : volumes) |
||||||
|
d *= v.get(); |
||||||
|
|
||||||
|
return Calc.clamp(d, 0, 1); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set master gain |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public void set(Double o) |
||||||
|
{ |
||||||
|
super.set(o); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,83 @@ |
|||||||
|
package mightypork.gamecore.audio; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.HashMap; |
||||||
|
import java.util.Map; |
||||||
|
|
||||||
|
import mightypork.gamecore.audio.players.EffectPlayer; |
||||||
|
import mightypork.gamecore.audio.players.LoopPlayer; |
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Audio resource storage |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class SoundRegistry { |
||||||
|
|
||||||
|
private final Map<String, EffectPlayer> effects = new HashMap<>(); |
||||||
|
private final Map<String, LoopPlayer> loops = new HashMap<>(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Register effect resource |
||||||
|
* |
||||||
|
* @param key sound key |
||||||
|
* @param resource resource path |
||||||
|
* @param pitch default pitch (1 = unchanged) |
||||||
|
* @param gain default gain (0-1) |
||||||
|
*/ |
||||||
|
public void addEffect(String key, String resource, double pitch, double gain) |
||||||
|
{ |
||||||
|
effects.put(key, App.audio().createEffect(resource, pitch, gain)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Register loop resource (music / effect loop) |
||||||
|
* |
||||||
|
* @param key sound key |
||||||
|
* @param resource resource path |
||||||
|
* @param pitch default pitch (1 = unchanged) |
||||||
|
* @param gain default gain (0-1) |
||||||
|
* @param fadeIn default time for fadeIn |
||||||
|
* @param fadeOut default time for fadeOut |
||||||
|
*/ |
||||||
|
public void addLoop(String key, String resource, double pitch, double gain, double fadeIn, double fadeOut) |
||||||
|
{ |
||||||
|
loops.put(key, App.audio().createLoop(resource, pitch, gain, fadeIn, fadeOut)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get a loop player for key |
||||||
|
* |
||||||
|
* @param key sound key |
||||||
|
* @return loop player |
||||||
|
*/ |
||||||
|
public LoopPlayer getLoop(String key) |
||||||
|
{ |
||||||
|
final LoopPlayer p = loops.get(key); |
||||||
|
if (p == null) { |
||||||
|
throw new RuntimeException("Unknown sound loop \"" + key + "\"."); |
||||||
|
} |
||||||
|
return p; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get a effect player for key |
||||||
|
* |
||||||
|
* @param key sound key |
||||||
|
* @return effect player |
||||||
|
*/ |
||||||
|
public EffectPlayer getEffect(String key) |
||||||
|
{ |
||||||
|
final EffectPlayer p = effects.get(key); |
||||||
|
if (p == null) { |
||||||
|
throw new RuntimeException("Unknown sound effect \"" + key + "\"."); |
||||||
|
} |
||||||
|
return p; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,29 @@ |
|||||||
|
package mightypork.gamecore.audio; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.math.Calc; |
||||||
|
import mightypork.utils.struct.Mutable; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Mutable volume 0-1 |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class Volume extends Mutable<Double> { |
||||||
|
|
||||||
|
/** |
||||||
|
* @param d initial value |
||||||
|
*/ |
||||||
|
public Volume(Double d) { |
||||||
|
super(d); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void set(Double d) |
||||||
|
{ |
||||||
|
super.set(Calc.clamp(d, 0, 1)); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,105 @@ |
|||||||
|
package mightypork.gamecore.audio.players; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.audio.DeferredAudio; |
||||||
|
import mightypork.gamecore.audio.Volume; |
||||||
|
import mightypork.utils.interfaces.Destroyable; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Basic abstract player |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class BaseAudioPlayer implements Destroyable { |
||||||
|
|
||||||
|
/** the track */ |
||||||
|
private final DeferredAudio audio; |
||||||
|
|
||||||
|
/** base gain for sfx */ |
||||||
|
private final double baseGain; |
||||||
|
|
||||||
|
/** base pitch for sfx */ |
||||||
|
private final double basePitch; |
||||||
|
|
||||||
|
/** dedicated volume control */ |
||||||
|
private final Volume gainMultiplier; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param track audio resource |
||||||
|
* @param basePitch base pitch (pitch multiplier) |
||||||
|
* @param baseGain base gain (volume multiplier) |
||||||
|
* @param volume colume control |
||||||
|
*/ |
||||||
|
public BaseAudioPlayer(DeferredAudio track, double basePitch, double baseGain, Volume volume) { |
||||||
|
this.audio = track; |
||||||
|
|
||||||
|
this.baseGain = baseGain; |
||||||
|
this.basePitch = basePitch; |
||||||
|
|
||||||
|
if (volume == null) volume = new Volume(1D); |
||||||
|
|
||||||
|
this.gainMultiplier = volume; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void destroy() |
||||||
|
{ |
||||||
|
audio.destroy(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return audio resource |
||||||
|
*/ |
||||||
|
protected DeferredAudio getAudio() |
||||||
|
{ |
||||||
|
return audio; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get play gain, computed based on volume and given multiplier |
||||||
|
* |
||||||
|
* @param multiplier extra volume adjustment |
||||||
|
* @return computed gain |
||||||
|
*/ |
||||||
|
protected double computeGain(double multiplier) |
||||||
|
{ |
||||||
|
return baseGain * gainMultiplier.get() * multiplier; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get pitch |
||||||
|
* |
||||||
|
* @param multiplier pitch adjustment |
||||||
|
* @return computed pitch |
||||||
|
*/ |
||||||
|
protected double computePitch(double multiplier) |
||||||
|
{ |
||||||
|
return basePitch * multiplier; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get if audio is valid |
||||||
|
* |
||||||
|
* @return is valid |
||||||
|
*/ |
||||||
|
protected boolean hasAudio() |
||||||
|
{ |
||||||
|
return (audio != null); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* force load the resource |
||||||
|
*/ |
||||||
|
public void load() |
||||||
|
{ |
||||||
|
if (hasAudio()) audio.load(); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,66 @@ |
|||||||
|
package mightypork.gamecore.audio.players; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.audio.DeferredAudio; |
||||||
|
import mightypork.gamecore.audio.Volume; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Player for one-off effects |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class EffectPlayer extends BaseAudioPlayer { |
||||||
|
|
||||||
|
/** |
||||||
|
* @param track audio resource |
||||||
|
* @param basePitch base pitch (pitch multiplier) |
||||||
|
* @param baseGain base gain (volume multiplier) |
||||||
|
* @param volume volume control |
||||||
|
*/ |
||||||
|
public EffectPlayer(DeferredAudio track, double basePitch, double baseGain, Volume volume) { |
||||||
|
super(track, (float) basePitch, (float) baseGain, volume); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Play at listener |
||||||
|
* |
||||||
|
* @param pitch play pitch |
||||||
|
* @param gain play gain |
||||||
|
*/ |
||||||
|
public void play(double pitch, double gain) |
||||||
|
{ |
||||||
|
if (!hasAudio()) return; |
||||||
|
|
||||||
|
getAudio().play(computePitch(pitch), computeGain(gain), false); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Play at listener |
||||||
|
* |
||||||
|
* @param gain play gain |
||||||
|
*/ |
||||||
|
public void play(double gain) |
||||||
|
{ |
||||||
|
play(1, gain); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Play at given position |
||||||
|
* |
||||||
|
* @param pitch play pitch |
||||||
|
* @param gain play gain |
||||||
|
* @param pos play position |
||||||
|
*/ |
||||||
|
public void play(double pitch, double gain, Vect pos) |
||||||
|
{ |
||||||
|
if (!hasAudio()) return; |
||||||
|
|
||||||
|
getAudio().play(computePitch(pitch), computeGain(gain), false, pos); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,166 @@ |
|||||||
|
package mightypork.gamecore.audio.players; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.audio.DeferredAudio; |
||||||
|
import mightypork.gamecore.audio.Volume; |
||||||
|
import mightypork.utils.interfaces.Pauseable; |
||||||
|
import mightypork.utils.interfaces.Updateable; |
||||||
|
import mightypork.utils.math.animation.NumAnimated; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Audio loop player (with fading, good for music) |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class LoopPlayer extends BaseAudioPlayer implements Updateable, Pauseable { |
||||||
|
|
||||||
|
private int sourceID = -1; |
||||||
|
|
||||||
|
/** animator for fade in and fade out */ |
||||||
|
private final NumAnimated fadeAnim = new NumAnimated(0); |
||||||
|
|
||||||
|
private double lastUpdateGain = 0; |
||||||
|
|
||||||
|
/** flag that track is paused */ |
||||||
|
private boolean paused = true; |
||||||
|
|
||||||
|
/** Default fadeIn time */ |
||||||
|
private double inTime = 1; |
||||||
|
|
||||||
|
/** Default fadeOut time */ |
||||||
|
private double outTime = 1; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param track audio resource |
||||||
|
* @param basePitch base pitch (pitch multiplier) |
||||||
|
* @param baseGain base gain (volume multiplier) |
||||||
|
* @param volume volume control |
||||||
|
*/ |
||||||
|
public LoopPlayer(DeferredAudio track, double basePitch, double baseGain, Volume volume) { |
||||||
|
super(track, (float) basePitch, (float) baseGain, volume); |
||||||
|
|
||||||
|
paused = true; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set fading duration (seconds) |
||||||
|
* |
||||||
|
* @param in duration of fade-in |
||||||
|
* @param out duration of fade-out |
||||||
|
*/ |
||||||
|
public void setFadeTimes(double in, double out) |
||||||
|
{ |
||||||
|
inTime = in; |
||||||
|
outTime = out; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
private void initLoop() |
||||||
|
{ |
||||||
|
if (hasAudio() && sourceID == -1) { |
||||||
|
getAudio().play(computePitch(1), computeGain(1), true); |
||||||
|
getAudio().pauseLoop(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void pause() |
||||||
|
{ |
||||||
|
if (!hasAudio() || paused) return; |
||||||
|
|
||||||
|
initLoop(); |
||||||
|
|
||||||
|
getAudio().pauseLoop(); |
||||||
|
paused = true; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean isPaused() |
||||||
|
{ |
||||||
|
return paused; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void resume() |
||||||
|
{ |
||||||
|
if (!hasAudio() || !paused) return; |
||||||
|
|
||||||
|
initLoop(); |
||||||
|
|
||||||
|
paused = false; |
||||||
|
|
||||||
|
getAudio().adjustGain(computeGain(fadeAnim.value())); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void update(double delta) |
||||||
|
{ |
||||||
|
if (!hasAudio() || paused) return; |
||||||
|
|
||||||
|
initLoop(); |
||||||
|
|
||||||
|
fadeAnim.update(delta); |
||||||
|
|
||||||
|
final double gain = computeGain(fadeAnim.value()); |
||||||
|
if (!paused && gain != lastUpdateGain) { |
||||||
|
getAudio().adjustGain(gain); |
||||||
|
lastUpdateGain = gain; |
||||||
|
} |
||||||
|
|
||||||
|
if (gain == 0 && !paused) pause(); // pause on zero volume
|
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Resume if paused, and fade in (pick up from current volume). |
||||||
|
* |
||||||
|
* @param secs |
||||||
|
*/ |
||||||
|
public void fadeIn(double secs) |
||||||
|
{ |
||||||
|
if (!hasAudio()) return; |
||||||
|
|
||||||
|
if (isPaused()) fadeAnim.setTo(0); |
||||||
|
resume(); |
||||||
|
fadeAnim.fadeIn(secs); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Fade out and pause when reached zero volume |
||||||
|
* |
||||||
|
* @param secs fade duration |
||||||
|
*/ |
||||||
|
public void fadeOut(double secs) |
||||||
|
{ |
||||||
|
if (!hasAudio()) return; |
||||||
|
if (isPaused()) return; |
||||||
|
fadeAnim.fadeOut(secs); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Fade in with default duration |
||||||
|
*/ |
||||||
|
public void fadeIn() |
||||||
|
{ |
||||||
|
fadeIn(inTime); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Fade out with default duration |
||||||
|
*/ |
||||||
|
public void fadeOut() |
||||||
|
{ |
||||||
|
fadeOut(outTime); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,277 @@ |
|||||||
|
package mightypork.gamecore.core; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.ArrayList; |
||||||
|
import java.util.List; |
||||||
|
|
||||||
|
import mightypork.gamecore.audio.AudioModule; |
||||||
|
import mightypork.gamecore.core.config.Config; |
||||||
|
import mightypork.gamecore.core.events.ShutdownEvent; |
||||||
|
import mightypork.gamecore.graphics.GraphicsModule; |
||||||
|
import mightypork.gamecore.input.InputModule; |
||||||
|
import mightypork.utils.annotations.Stub; |
||||||
|
import mightypork.utils.eventbus.EventBus; |
||||||
|
import mightypork.utils.eventbus.clients.BusNode; |
||||||
|
import mightypork.utils.eventbus.clients.DelegatingList; |
||||||
|
import mightypork.utils.eventbus.events.DestroyEvent; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Game base class & static subsystem access |
||||||
|
* |
||||||
|
* @author MightyPork |
||||||
|
*/ |
||||||
|
public class App extends BusNode { |
||||||
|
|
||||||
|
private static App instance; |
||||||
|
|
||||||
|
private final AppBackend backend; |
||||||
|
private final EventBus eventBus = new EventBus(); |
||||||
|
private boolean started = false; |
||||||
|
|
||||||
|
protected final DelegatingList plugins = new DelegatingList(); |
||||||
|
protected final List<InitTask> initializers = new ArrayList<>(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Create an app with given backend. |
||||||
|
* |
||||||
|
* @param backend |
||||||
|
*/ |
||||||
|
public App(AppBackend backend) { |
||||||
|
if (App.instance != null) { |
||||||
|
throw new IllegalStateException("App already initialized"); |
||||||
|
} |
||||||
|
|
||||||
|
// store current instance in static field
|
||||||
|
App.instance = this; |
||||||
|
|
||||||
|
// join the bus
|
||||||
|
this.eventBus.subscribe(this); |
||||||
|
|
||||||
|
// create plugin registry attached to bus
|
||||||
|
this.eventBus.subscribe(this.plugins); |
||||||
|
|
||||||
|
// initialize and use backend
|
||||||
|
this.backend = backend; |
||||||
|
this.eventBus.subscribe(backend); |
||||||
|
this.backend.bind(this); |
||||||
|
this.backend.initialize(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a plugin to the app. Plugins can eg. listen to bus events and react |
||||||
|
* to them. |
||||||
|
* |
||||||
|
* @param plugin the added plugin. |
||||||
|
*/ |
||||||
|
public void addPlugin(AppPlugin plugin) |
||||||
|
{ |
||||||
|
if (started) { |
||||||
|
throw new IllegalStateException("App already started, cannot add plugins."); |
||||||
|
} |
||||||
|
|
||||||
|
// attach to event bus
|
||||||
|
plugins.add(plugin); |
||||||
|
plugin.bind(this); |
||||||
|
plugin.initialize(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add an initializer to the app. |
||||||
|
* |
||||||
|
* @param initializer |
||||||
|
*/ |
||||||
|
public void addInitTask(InitTask initializer) |
||||||
|
{ |
||||||
|
if (started) { |
||||||
|
throw new IllegalStateException("App already started, cannot add initializers."); |
||||||
|
} |
||||||
|
|
||||||
|
initializers.add(initializer); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get current backend |
||||||
|
* |
||||||
|
* @return the backend |
||||||
|
*/ |
||||||
|
public AppBackend getBackend() |
||||||
|
{ |
||||||
|
return backend; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize the App and start operating.<br> |
||||||
|
* This method should be called after adding all required initializers and |
||||||
|
* plugins. |
||||||
|
*/ |
||||||
|
public final void start() |
||||||
|
{ |
||||||
|
if (started) { |
||||||
|
throw new IllegalStateException("Already started."); |
||||||
|
} |
||||||
|
started = true; |
||||||
|
|
||||||
|
// pre-init hook, just in case anyone wanted to have one.
|
||||||
|
Log.f2("Calling pre-init hook..."); |
||||||
|
preInit(); |
||||||
|
|
||||||
|
Log.i("=== Starting initialization sequence ==="); |
||||||
|
|
||||||
|
// sort initializers by order.
|
||||||
|
List<InitTask> orderedInitializers = InitTask.inOrder(initializers); |
||||||
|
|
||||||
|
for (InitTask initializer : orderedInitializers) { |
||||||
|
Log.f1("Running init task \"" + initializer.getName() + "\"..."); |
||||||
|
initializer.bind(this); |
||||||
|
initializer.init(); |
||||||
|
initializer.run(); |
||||||
|
} |
||||||
|
|
||||||
|
Log.i("=== Initialization sequence completed ==="); |
||||||
|
|
||||||
|
// user can now start the main loop etc.
|
||||||
|
Log.f2("Calling post-init hook..."); |
||||||
|
postInit(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Hook called before the initialization sequence starts. |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
protected void preInit() |
||||||
|
{ |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Hook called after the initialization sequence is finished. |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
protected void postInit() |
||||||
|
{ |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Shut down the running instance.<br> |
||||||
|
* Deinitialize backend modules and terminate the JVM. |
||||||
|
*/ |
||||||
|
public static void shutdown() |
||||||
|
{ |
||||||
|
if (instance != null) { |
||||||
|
Log.i("Dispatching Shutdown event..."); |
||||||
|
|
||||||
|
bus().send(new ShutdownEvent(new Runnable() { |
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() |
||||||
|
{ |
||||||
|
try { |
||||||
|
final EventBus bus = bus(); |
||||||
|
if (bus != null) { |
||||||
|
bus.send(new DestroyEvent()); |
||||||
|
bus.destroy(); |
||||||
|
} |
||||||
|
} catch (final Throwable e) { |
||||||
|
Log.e(e); |
||||||
|
} |
||||||
|
|
||||||
|
Log.i("Shutdown completed."); |
||||||
|
System.exit(0); |
||||||
|
} |
||||||
|
})); |
||||||
|
|
||||||
|
} else { |
||||||
|
Log.w("App is not running."); |
||||||
|
System.exit(0); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get the currently running App instance. |
||||||
|
* |
||||||
|
* @return app instance |
||||||
|
*/ |
||||||
|
public static App instance() |
||||||
|
{ |
||||||
|
return instance; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get graphics module from the running app's backend |
||||||
|
* |
||||||
|
* @return graphics module |
||||||
|
*/ |
||||||
|
public static GraphicsModule gfx() |
||||||
|
{ |
||||||
|
return instance.backend.getGraphics(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get audio module from the running app's backend |
||||||
|
* |
||||||
|
* @return audio module |
||||||
|
*/ |
||||||
|
public static AudioModule audio() |
||||||
|
{ |
||||||
|
return instance.backend.getAudio(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get input module from the running app's backend |
||||||
|
* |
||||||
|
* @return input module |
||||||
|
*/ |
||||||
|
public static InputModule input() |
||||||
|
{ |
||||||
|
return instance.backend.getInput(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get event bus instance. |
||||||
|
* |
||||||
|
* @return event bus |
||||||
|
*/ |
||||||
|
public static EventBus bus() |
||||||
|
{ |
||||||
|
return instance.eventBus; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get the main config, if initialized. |
||||||
|
* |
||||||
|
* @return main config |
||||||
|
* @throws IllegalArgumentException if there is no such config. |
||||||
|
*/ |
||||||
|
public static Config cfg() |
||||||
|
{ |
||||||
|
return cfg("main"); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get a config by alias. |
||||||
|
* |
||||||
|
* @param alias config alias |
||||||
|
* @return the config |
||||||
|
* @throws IllegalArgumentException if there is no such config. |
||||||
|
*/ |
||||||
|
public static Config cfg(String alias) |
||||||
|
{ |
||||||
|
return Config.forAlias(alias); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,65 @@ |
|||||||
|
package mightypork.gamecore.core; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.audio.AudioModule; |
||||||
|
import mightypork.gamecore.graphics.GraphicsModule; |
||||||
|
import mightypork.gamecore.input.InputModule; |
||||||
|
import mightypork.utils.eventbus.clients.BusNode; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Application backend interface (set of core modules).<br> |
||||||
|
* The goal of this abstraction is to allow easy migration to different |
||||||
|
* environment with different libraries etc. It should be as simple as using |
||||||
|
* different backend. |
||||||
|
* |
||||||
|
* @author MightyPork |
||||||
|
*/ |
||||||
|
public abstract class AppBackend extends BusNode { |
||||||
|
|
||||||
|
protected App app; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Assign an app instance. |
||||||
|
* |
||||||
|
* @param app app |
||||||
|
*/ |
||||||
|
public void bind(App app) |
||||||
|
{ |
||||||
|
if (this.app != null) { |
||||||
|
throw new IllegalStateException("App already set."); |
||||||
|
} |
||||||
|
this.app = app; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize backend modules, add them to event bus. |
||||||
|
*/ |
||||||
|
public abstract void initialize(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get graphics module (screen manager, texture and font loader, renderer) |
||||||
|
* |
||||||
|
* @return graphics module |
||||||
|
*/ |
||||||
|
public abstract GraphicsModule getGraphics(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get audio module ( |
||||||
|
* |
||||||
|
* @return audio module |
||||||
|
*/ |
||||||
|
public abstract AudioModule getAudio(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get input module |
||||||
|
* |
||||||
|
* @return input module |
||||||
|
*/ |
||||||
|
public abstract InputModule getInput(); |
||||||
|
} |
@ -0,0 +1,34 @@ |
|||||||
|
package mightypork.gamecore.core; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.annotations.Stub; |
||||||
|
import mightypork.utils.eventbus.clients.BusNode; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* App plugin. Plugins are an easy way to extend app functionality.<br> |
||||||
|
* Typically, a plugin waits for trigger event(s) and performs some action upon |
||||||
|
* receiving them. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class AppPlugin extends BusNode { |
||||||
|
|
||||||
|
protected App app; |
||||||
|
|
||||||
|
|
||||||
|
void bind(App app) |
||||||
|
{ |
||||||
|
this.app = app; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize the plugin for the given App.<br> |
||||||
|
* The plugin is already attached to the event bus. |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
public void initialize() |
||||||
|
{ |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,24 @@ |
|||||||
|
package mightypork.gamecore.core; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.annotations.Stub; |
||||||
|
import mightypork.utils.eventbus.clients.BusNode; |
||||||
|
import mightypork.utils.interfaces.Destroyable; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Abstract application backend module. |
||||||
|
* |
||||||
|
* @author MightyPork |
||||||
|
*/ |
||||||
|
public abstract class BackendModule extends BusNode implements Destroyable { |
||||||
|
|
||||||
|
public abstract void init(); |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
@Stub |
||||||
|
public void destroy() |
||||||
|
{ |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,150 @@ |
|||||||
|
package mightypork.gamecore.core; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.ArrayList; |
||||||
|
import java.util.HashSet; |
||||||
|
import java.util.Iterator; |
||||||
|
import java.util.List; |
||||||
|
import java.util.Set; |
||||||
|
|
||||||
|
import mightypork.utils.Reflect; |
||||||
|
import mightypork.utils.annotations.Stub; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* App initializer. A sequence of initializers is executed once the start() |
||||||
|
* method on App is called. Adding initializers is one way to customize the App |
||||||
|
* behavior and features. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class InitTask { |
||||||
|
|
||||||
|
protected App app; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Assign the initialized app instance to a protected "app" field. |
||||||
|
* |
||||||
|
* @param app app |
||||||
|
*/ |
||||||
|
void bind(App app) |
||||||
|
{ |
||||||
|
this.app = app; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* An intialization method that is called before the run() method.<br> |
||||||
|
* This method should be left unimplemented in the task, and can be used to |
||||||
|
* configure the init task when using it as anonymous inner type. |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
public void init() |
||||||
|
{ |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Run the initalizer on app. |
||||||
|
*/ |
||||||
|
public abstract void run(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get name of this initializer (for dependency resolver).<br> |
||||||
|
* The name should be short, snake_case and precise. |
||||||
|
* |
||||||
|
* @return name |
||||||
|
*/ |
||||||
|
public abstract String getName(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get what other initializers must be already loaded before this can load.<br> |
||||||
|
* Depending on itself or creating a circular dependency will cause error.<br> |
||||||
|
* If the dependencies cannot be satisfied, the initialization sequence will |
||||||
|
* be aborted. |
||||||
|
* |
||||||
|
* @return array of names of required initializers. |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
public String[] getDependencies() |
||||||
|
{ |
||||||
|
return new String[] {}; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Order init tasks so that all dependencies are loaded before thye are |
||||||
|
* needed by the tasks. |
||||||
|
* |
||||||
|
* @param tasks task list |
||||||
|
* @return task list ordered |
||||||
|
*/ |
||||||
|
public static List<InitTask> inOrder(List<InitTask> tasks) |
||||||
|
{ |
||||||
|
List<InitTask> remaining = new ArrayList<>(tasks); |
||||||
|
|
||||||
|
List<InitTask> ordered = new ArrayList<>(); |
||||||
|
Set<String> loaded = new HashSet<>(); |
||||||
|
|
||||||
|
// resolve task order
|
||||||
|
int addedThisIteration = 0; |
||||||
|
do { |
||||||
|
for (Iterator<InitTask> i = remaining.iterator(); i.hasNext();) { |
||||||
|
InitTask task = i.next(); |
||||||
|
|
||||||
|
String[] deps = task.getDependencies(); |
||||||
|
if (deps == null) deps = new String[] {}; |
||||||
|
|
||||||
|
int unmetDepsCount = deps.length; |
||||||
|
|
||||||
|
for (String d : deps) { |
||||||
|
if (loaded.contains(d)) unmetDepsCount--; |
||||||
|
} |
||||||
|
|
||||||
|
if (unmetDepsCount == 0) { |
||||||
|
ordered.add(task); |
||||||
|
loaded.add(task.getName()); |
||||||
|
i.remove(); |
||||||
|
addedThisIteration++; |
||||||
|
} |
||||||
|
} |
||||||
|
} while (addedThisIteration > 0); |
||||||
|
|
||||||
|
// check if any tasks are left out
|
||||||
|
if (remaining.size() > 0) { |
||||||
|
|
||||||
|
// build error message for each bad task
|
||||||
|
int badInitializers = 0; |
||||||
|
for (InitTask task : remaining) { |
||||||
|
if (Reflect.hasAnnotation(task.getClass(), OptionalInitTask.class)) { |
||||||
|
continue; |
||||||
|
} |
||||||
|
|
||||||
|
badInitializers++; |
||||||
|
|
||||||
|
String notSatisfied = ""; |
||||||
|
|
||||||
|
for (String d : task.getDependencies()) { |
||||||
|
if (!loaded.contains(d)) { |
||||||
|
|
||||||
|
if (!notSatisfied.isEmpty()) { |
||||||
|
notSatisfied += ", "; |
||||||
|
} |
||||||
|
|
||||||
|
notSatisfied += d; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
Log.w("InitTask \"" + task.getName() + "\" - missing dependencies: " + notSatisfied); |
||||||
|
} |
||||||
|
|
||||||
|
if (badInitializers > 0) throw new RuntimeException("Some InitTask dependencies could not be satisfied."); |
||||||
|
} |
||||||
|
|
||||||
|
return ordered; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,130 @@ |
|||||||
|
package mightypork.gamecore.core; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.Deque; |
||||||
|
import java.util.concurrent.ConcurrentLinkedDeque; |
||||||
|
|
||||||
|
import mightypork.gamecore.graphics.Renderable; |
||||||
|
import mightypork.gamecore.gui.screens.ScreenRegistry; |
||||||
|
import mightypork.utils.annotations.Stub; |
||||||
|
import mightypork.utils.eventbus.clients.BusNode; |
||||||
|
import mightypork.utils.eventbus.events.UpdateEvent; |
||||||
|
import mightypork.utils.interfaces.Destroyable; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
import mightypork.utils.math.timing.Profiler; |
||||||
|
import mightypork.utils.math.timing.TimerDelta; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Delta-timed game loop with task queue etc. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class MainLoop extends BusNode implements Destroyable { |
||||||
|
|
||||||
|
private static final double MAX_TIME_TASKS = 1 / 30D; // (avoid queue from hogging timing)
|
||||||
|
private static final double MAX_DELTA = 1 / 20D; // (skip huge gaps caused by loading resources etc)
|
||||||
|
|
||||||
|
private final Deque<Runnable> tasks = new ConcurrentLinkedDeque<>(); |
||||||
|
private TimerDelta timer; |
||||||
|
private Renderable rootRenderable; |
||||||
|
private volatile boolean running = true; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set primary renderable |
||||||
|
* |
||||||
|
* @param rootRenderable main {@link Renderable}, typically a |
||||||
|
* {@link ScreenRegistry} |
||||||
|
*/ |
||||||
|
public void setRootRenderable(Renderable rootRenderable) |
||||||
|
{ |
||||||
|
this.rootRenderable = rootRenderable; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Start the loop |
||||||
|
*/ |
||||||
|
public void start() |
||||||
|
{ |
||||||
|
timer = new TimerDelta(); |
||||||
|
|
||||||
|
while (running) { |
||||||
|
App.gfx().beginFrame(); |
||||||
|
|
||||||
|
double delta = timer.getDelta(); |
||||||
|
if (delta > MAX_DELTA) { |
||||||
|
Log.f3("(timing) Cropping delta: was " + delta + " , limit " + MAX_DELTA); |
||||||
|
delta = MAX_DELTA; |
||||||
|
} |
||||||
|
|
||||||
|
App.bus().sendDirect(new UpdateEvent(delta)); |
||||||
|
|
||||||
|
Runnable r; |
||||||
|
final long t = Profiler.begin(); |
||||||
|
while ((r = tasks.poll()) != null) { |
||||||
|
Log.f3(" * Main loop task."); |
||||||
|
r.run(); |
||||||
|
if (Profiler.end(t) > MAX_TIME_TASKS) { |
||||||
|
Log.f3("! Postponing main loop tasks to next cycle."); |
||||||
|
break; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
beforeRender(); |
||||||
|
|
||||||
|
if (rootRenderable != null) { |
||||||
|
rootRenderable.render(); |
||||||
|
} |
||||||
|
|
||||||
|
afterRender(); |
||||||
|
|
||||||
|
App.gfx().endFrame(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Called before render |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
protected void beforeRender() |
||||||
|
{ |
||||||
|
//
|
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Called after render |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
protected void afterRender() |
||||||
|
{ |
||||||
|
//
|
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void destroy() |
||||||
|
{ |
||||||
|
running = false; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a task to queue to be executed in the main loop (OpenGL thread) |
||||||
|
* |
||||||
|
* @param request task |
||||||
|
* @param priority if true, skip other tasks |
||||||
|
*/ |
||||||
|
public synchronized void queueTask(Runnable request, boolean priority) |
||||||
|
{ |
||||||
|
if (priority) { |
||||||
|
tasks.addFirst(request); |
||||||
|
} else { |
||||||
|
tasks.addLast(request); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,19 @@ |
|||||||
|
package mightypork.gamecore.core; |
||||||
|
|
||||||
|
|
||||||
|
import java.lang.annotation.*; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Indicates that an {@link InitTask} can safely be ignored if it's dependencies |
||||||
|
* are not satisfied. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
@Retention(RetentionPolicy.RUNTIME) |
||||||
|
@Target(ElementType.TYPE) |
||||||
|
@Documented |
||||||
|
@Inherited |
||||||
|
public @interface OptionalInitTask { |
||||||
|
|
||||||
|
} |
@ -0,0 +1,91 @@ |
|||||||
|
package mightypork.gamecore.core; |
||||||
|
|
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
import java.util.HashMap; |
||||||
|
import java.util.Map; |
||||||
|
|
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Static application workdir accessor. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class WorkDir { |
||||||
|
|
||||||
|
private static File workdir; |
||||||
|
private static Map<String, String> namedPaths = new HashMap<>(); |
||||||
|
|
||||||
|
|
||||||
|
public static void init(File workdir) |
||||||
|
{ |
||||||
|
WorkDir.workdir = workdir; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a path alias (dir or file), relative to the workdir. |
||||||
|
* |
||||||
|
* @param alias path alias |
||||||
|
* @param path path relative to workdir |
||||||
|
*/ |
||||||
|
public static void addPath(String alias, String path) |
||||||
|
{ |
||||||
|
namedPaths.put(alias, path); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get workdir folder, create if not exists. |
||||||
|
* |
||||||
|
* @param path dir path relative to workdir |
||||||
|
* @return dir file |
||||||
|
*/ |
||||||
|
public static File getDir(String path) |
||||||
|
{ |
||||||
|
if (namedPaths.containsKey(path)) path = namedPaths.get(path); |
||||||
|
|
||||||
|
final File f = new File(workdir, path); |
||||||
|
if (!f.exists()) { |
||||||
|
if (!f.mkdirs()) { |
||||||
|
Log.w("Could not create a directory: " + f + " (path: " + path + ")"); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
return f; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get workdir file, create parent if not exists. |
||||||
|
* |
||||||
|
* @param path dir path relative to workdir |
||||||
|
* @return dir file |
||||||
|
*/ |
||||||
|
public static File getFile(String path) |
||||||
|
{ |
||||||
|
if (namedPaths.containsKey(path)) path = namedPaths.get(path); |
||||||
|
|
||||||
|
final File f = new File(workdir, path); |
||||||
|
|
||||||
|
// create the parent dir
|
||||||
|
if (!f.getParent().equals(workdir)) { |
||||||
|
f.getParentFile().mkdirs(); |
||||||
|
} |
||||||
|
|
||||||
|
return f; |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return the workdir File |
||||||
|
*/ |
||||||
|
public static File getWorkDir() |
||||||
|
{ |
||||||
|
return workdir; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,269 @@ |
|||||||
|
package mightypork.gamecore.core.config; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.HashMap; |
||||||
|
import java.util.Map; |
||||||
|
|
||||||
|
import mightypork.gamecore.core.WorkDir; |
||||||
|
import mightypork.gamecore.input.Key; |
||||||
|
import mightypork.gamecore.input.KeyStroke; |
||||||
|
import mightypork.utils.config.propmgr.Property; |
||||||
|
import mightypork.utils.config.propmgr.PropertyManager; |
||||||
|
import mightypork.utils.config.propmgr.PropertyStore; |
||||||
|
import mightypork.utils.config.propmgr.store.PropertyFile; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Settings repository. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class Config { |
||||||
|
|
||||||
|
protected static Map<String, Config> configs = new HashMap<>(); |
||||||
|
|
||||||
|
private Map<String, KeyStrokeProperty> strokes = new HashMap<>(); |
||||||
|
|
||||||
|
private PropertyManager propertyManager; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get a config from the static map, by given alias |
||||||
|
* |
||||||
|
* @param alias alias |
||||||
|
* @return the config |
||||||
|
*/ |
||||||
|
public static Config forAlias(String alias) |
||||||
|
{ |
||||||
|
Config c = configs.get(alias); |
||||||
|
|
||||||
|
if (c == null) { |
||||||
|
throw new IllegalArgumentException("There is no config with alias \"" + alias + "\""); |
||||||
|
} |
||||||
|
|
||||||
|
return c; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Register a config by alias. |
||||||
|
* |
||||||
|
* @param alias config alias |
||||||
|
* @param config the config |
||||||
|
*/ |
||||||
|
public static void register(String alias, Config config) |
||||||
|
{ |
||||||
|
if (configs.get(alias) != null) { |
||||||
|
throw new IllegalArgumentException("The alias \"" + alias + "\" is already used."); |
||||||
|
} |
||||||
|
|
||||||
|
configs.put(alias, config); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize property manager for a file |
||||||
|
* |
||||||
|
* @param file config file, relative to workdir |
||||||
|
* @param headComment file comment |
||||||
|
*/ |
||||||
|
public Config(String file, String headComment) { |
||||||
|
this(new PropertyFile(WorkDir.getFile(file), headComment)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize property manager for a given store |
||||||
|
* |
||||||
|
* @param store property store backing the property manager |
||||||
|
*/ |
||||||
|
public Config(PropertyStore store) { |
||||||
|
if (propertyManager != null) { |
||||||
|
throw new IllegalStateException("Config already initialized."); |
||||||
|
} |
||||||
|
|
||||||
|
propertyManager = new PropertyManager(store); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a keystroke property |
||||||
|
* |
||||||
|
* @param key key in config file |
||||||
|
* @param defval default value (keystroke datastring) |
||||||
|
* @param comment optional comment, can be null |
||||||
|
*/ |
||||||
|
public void addKeyStroke(String key, String defval, String comment) |
||||||
|
{ |
||||||
|
final KeyStrokeProperty kprop = new KeyStrokeProperty(prefixKeyStroke(key), KeyStroke.createFromString(defval), comment); |
||||||
|
strokes.put(prefixKeyStroke(key), kprop); |
||||||
|
propertyManager.addProperty(kprop); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a boolean property (flag) |
||||||
|
* |
||||||
|
* @param key key in config file |
||||||
|
* @param defval default value |
||||||
|
* @param comment optional comment, can be null |
||||||
|
*/ |
||||||
|
public void addBoolean(String key, boolean defval, String comment) |
||||||
|
{ |
||||||
|
propertyManager.addBoolean(key, defval, comment); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add an integer property |
||||||
|
* |
||||||
|
* @param key key in config file |
||||||
|
* @param defval default value |
||||||
|
* @param comment optional comment, can be null |
||||||
|
*/ |
||||||
|
public void addInteger(String key, int defval, String comment) |
||||||
|
{ |
||||||
|
propertyManager.addInteger(key, defval, comment); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a double property |
||||||
|
* |
||||||
|
* @param key key in config file |
||||||
|
* @param defval default value |
||||||
|
* @param comment optional comment, can be null |
||||||
|
*/ |
||||||
|
public void addDouble(String key, double defval, String comment) |
||||||
|
{ |
||||||
|
propertyManager.addDouble(key, defval, comment); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a string property |
||||||
|
* |
||||||
|
* @param key key in config file |
||||||
|
* @param defval default value |
||||||
|
* @param comment optional comment, can be null |
||||||
|
*/ |
||||||
|
public void addString(String key, String defval, String comment) |
||||||
|
{ |
||||||
|
propertyManager.addString(key, defval, comment); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add an arbitrary property (can be custom type) |
||||||
|
* |
||||||
|
* @param prop the property to add |
||||||
|
*/ |
||||||
|
public <T> void addProperty(Property<T> prop) |
||||||
|
{ |
||||||
|
propertyManager.addProperty(prop); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Load config from file |
||||||
|
*/ |
||||||
|
public void load() |
||||||
|
{ |
||||||
|
propertyManager.load(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Save config to file |
||||||
|
*/ |
||||||
|
public void save() |
||||||
|
{ |
||||||
|
Log.f3("Saving config."); |
||||||
|
propertyManager.save(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get an option for key |
||||||
|
* |
||||||
|
* @param key |
||||||
|
* @return option value |
||||||
|
*/ |
||||||
|
public <T> T getValue(String key) |
||||||
|
{ |
||||||
|
try { |
||||||
|
if (propertyManager.getProperty(key) == null) { |
||||||
|
throw new IllegalArgumentException("No such property: " + key); |
||||||
|
} |
||||||
|
|
||||||
|
return propertyManager.getValue(key); |
||||||
|
} catch (final ClassCastException cce) { |
||||||
|
throw new RuntimeException("Property of incompatible type: " + key); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set option to a value. Call the save() method to make the change |
||||||
|
* permanent. |
||||||
|
* |
||||||
|
* @param key option key |
||||||
|
* @param value value to set |
||||||
|
*/ |
||||||
|
public <T> void setValue(String key, T value) |
||||||
|
{ |
||||||
|
if (propertyManager.getProperty(key) == null) { |
||||||
|
throw new IllegalArgumentException("No such property: " + key); |
||||||
|
} |
||||||
|
|
||||||
|
propertyManager.setValue(key, value); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add "key." before the given config file key |
||||||
|
* |
||||||
|
* @param cfgKey config key |
||||||
|
* @return key. + cfgKey |
||||||
|
*/ |
||||||
|
private String prefixKeyStroke(String cfgKey) |
||||||
|
{ |
||||||
|
return "key." + cfgKey; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get keystroke for name |
||||||
|
* |
||||||
|
* @param cfgKey stroke identifier in config file |
||||||
|
* @return the stroke |
||||||
|
*/ |
||||||
|
public KeyStroke getKeyStroke(String cfgKey) |
||||||
|
{ |
||||||
|
final KeyStrokeProperty kp = strokes.get(prefixKeyStroke(cfgKey)); |
||||||
|
if (kp == null) { |
||||||
|
throw new IllegalArgumentException("No such stroke: " + cfgKey); |
||||||
|
} |
||||||
|
|
||||||
|
return kp.getValue(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set a keystroke for name |
||||||
|
* |
||||||
|
* @param cfgKey stroke identifier in config file |
||||||
|
* @param key stroke key |
||||||
|
* @param mod stroke modifiers |
||||||
|
*/ |
||||||
|
public void setKeyStroke(String cfgKey, Key key, int mod) |
||||||
|
{ |
||||||
|
final KeyStrokeProperty kp = strokes.get(prefixKeyStroke(cfgKey)); |
||||||
|
if (kp == null) { |
||||||
|
throw new IllegalArgumentException("No such stroke: " + cfgKey); |
||||||
|
} |
||||||
|
|
||||||
|
kp.getValue().setTo(key, mod); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,75 @@ |
|||||||
|
package mightypork.gamecore.core.config; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.InitTask; |
||||||
|
import mightypork.utils.annotations.Stub; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize config. To apply this initializer, you must extend it. That |
||||||
|
* ensures that the workdir initializer has already finished when the code is |
||||||
|
* executed (such as resolving a file path for the config file). |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class InitTaskConfig extends InitTask { |
||||||
|
|
||||||
|
/** |
||||||
|
* Add a config with given alias |
||||||
|
* |
||||||
|
* @param alias config alias |
||||||
|
* @param config config to add |
||||||
|
*/ |
||||||
|
protected void addConfig(String alias, Config config) |
||||||
|
{ |
||||||
|
Config.register(alias, config); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize the main config. |
||||||
|
* |
||||||
|
* @return the main config. |
||||||
|
*/ |
||||||
|
protected abstract Config buildConfig(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize extra configs.<br> |
||||||
|
* the addConfig() method can be used to register configs. |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
protected void buildExtraConfigs() |
||||||
|
{ |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
// locked to encourage the use of the build* methods.
|
||||||
|
@Override |
||||||
|
public final void init() |
||||||
|
{ |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final void run() |
||||||
|
{ |
||||||
|
addConfig("main", buildConfig()); |
||||||
|
buildExtraConfigs(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String getName() |
||||||
|
{ |
||||||
|
return "config"; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String[] getDependencies() |
||||||
|
{ |
||||||
|
return new String[] { "workdir" }; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,55 @@ |
|||||||
|
package mightypork.gamecore.core.config; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.input.Key; |
||||||
|
import mightypork.gamecore.input.KeyStroke; |
||||||
|
import mightypork.gamecore.input.Keys; |
||||||
|
import mightypork.utils.config.propmgr.Property; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Key property.<br> |
||||||
|
* The stored value must stay the same instance ({@link KeyStroke} is mutable).<br> |
||||||
|
* That ensures that bindings based on this keystroke are automatically updated |
||||||
|
* when the settings change. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class KeyStrokeProperty extends Property<KeyStroke> { |
||||||
|
|
||||||
|
public KeyStrokeProperty(String key, KeyStroke defaultValue, String comment) { |
||||||
|
super(key, defaultValue, comment); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void fromString(String string) |
||||||
|
{ |
||||||
|
if (string != null) { |
||||||
|
// keep the same instance
|
||||||
|
|
||||||
|
final Key backup_key = value.getKey(); |
||||||
|
final int backup_mod = value.getMod(); |
||||||
|
|
||||||
|
value.loadFromString(string); |
||||||
|
if (value.getKey() == Keys.NONE) { |
||||||
|
value.setTo(backup_key, backup_mod); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String toString() |
||||||
|
{ |
||||||
|
return value.saveToString(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setValue(Object value) |
||||||
|
{ |
||||||
|
// keep the same instance
|
||||||
|
this.value.setTo(((KeyStroke) value).getKey(), ((KeyStroke) value).getMod()); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,44 @@ |
|||||||
|
package mightypork.gamecore.core.events; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.MainLoop; |
||||||
|
import mightypork.utils.eventbus.BusEvent; |
||||||
|
import mightypork.utils.eventbus.events.flags.SingleReceiverEvent; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Request to execute given {@link Runnable} in main loop. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
@SingleReceiverEvent |
||||||
|
public class MainLoopRequest extends BusEvent<MainLoop> { |
||||||
|
|
||||||
|
private final Runnable task; |
||||||
|
private final boolean priority; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param task task to run on main thread in rendering context |
||||||
|
*/ |
||||||
|
public MainLoopRequest(Runnable task) { |
||||||
|
this(task, false); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param task task to run on main thread in rendering context |
||||||
|
* @param priority if true, skip other tasks in queue |
||||||
|
*/ |
||||||
|
public MainLoopRequest(Runnable task, boolean priority) { |
||||||
|
this.task = task; |
||||||
|
this.priority = priority; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void handleBy(MainLoop handler) |
||||||
|
{ |
||||||
|
handler.queueTask(task, priority); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,47 @@ |
|||||||
|
package mightypork.gamecore.core.events; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.eventbus.BusEvent; |
||||||
|
import mightypork.utils.eventbus.EventBus; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Shutdown event.<br> |
||||||
|
* This event is dispatched when the <code>App.shutdown()</code> method is |
||||||
|
* called. If no client consumes it, the shutdown will immediately follow.<br> |
||||||
|
* This is a way to allow clients to abort the shutdown (ie. ask user to save |
||||||
|
* game). After the game is saved, the <code>App.shutdown()</code> method can be |
||||||
|
* called again. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class ShutdownEvent extends BusEvent<ShutdownListener> { |
||||||
|
|
||||||
|
private Runnable shutdownTask; |
||||||
|
|
||||||
|
|
||||||
|
public ShutdownEvent(Runnable doShutdown) { |
||||||
|
this.shutdownTask = doShutdown; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
protected void handleBy(ShutdownListener handler) |
||||||
|
{ |
||||||
|
handler.onShutdown(this); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void onDispatchComplete(EventBus bus) |
||||||
|
{ |
||||||
|
if (!isConsumed()) { |
||||||
|
Log.i("Shutting down..."); |
||||||
|
shutdownTask.run(); |
||||||
|
} else { |
||||||
|
Log.i("Shutdown aborted."); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,18 @@ |
|||||||
|
package mightypork.gamecore.core.events; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Quit request listener; implementing client can abort shutdown. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public interface ShutdownListener { |
||||||
|
|
||||||
|
/** |
||||||
|
* Intercept quit request.<br> |
||||||
|
* Consume the event to abort shutdown (ie. ask user to save) |
||||||
|
* |
||||||
|
* @param event quit request event. |
||||||
|
*/ |
||||||
|
void onShutdown(ShutdownEvent event); |
||||||
|
} |
@ -0,0 +1,42 @@ |
|||||||
|
package mightypork.gamecore.core.init; |
||||||
|
|
||||||
|
|
||||||
|
import java.lang.Thread.UncaughtExceptionHandler; |
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.core.InitTask; |
||||||
|
import mightypork.utils.annotations.Stub; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a crash handler to the app.<br> |
||||||
|
* For customized crash message / crash dialog etc, override the |
||||||
|
* uncaughtException() method. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class InitTaskCrashHandler extends InitTask implements UncaughtExceptionHandler { |
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() |
||||||
|
{ |
||||||
|
Thread.setDefaultUncaughtExceptionHandler(this); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
@Stub |
||||||
|
public void uncaughtException(Thread thread, Throwable throwable) |
||||||
|
{ |
||||||
|
Log.e("The game has crashed.", throwable); |
||||||
|
App.shutdown(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String getName() |
||||||
|
{ |
||||||
|
return "crash_handler"; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,97 @@ |
|||||||
|
package mightypork.gamecore.core.init; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.InitTask; |
||||||
|
import mightypork.gamecore.graphics.GraphicsModule; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Setup main window. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class InitTaskDisplay extends InitTask { |
||||||
|
|
||||||
|
private int width = 800, height = 600, fps = 60; |
||||||
|
private boolean resizable, fullscreen; |
||||||
|
private String title = "Game"; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set initial window size |
||||||
|
* |
||||||
|
* @param width width (px) |
||||||
|
* @param height height (px) |
||||||
|
*/ |
||||||
|
public void setSize(int width, int height) |
||||||
|
{ |
||||||
|
this.width = width; |
||||||
|
this.height = height; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set whether the window should be resizable |
||||||
|
* |
||||||
|
* @param resizable true for resizable |
||||||
|
*/ |
||||||
|
public void setResizable(boolean resizable) |
||||||
|
{ |
||||||
|
this.resizable = resizable; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set window title |
||||||
|
* |
||||||
|
* @param title title text |
||||||
|
*/ |
||||||
|
public void setTitle(String title) |
||||||
|
{ |
||||||
|
this.title = title; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set desired framerate. |
||||||
|
* |
||||||
|
* @param fps FPS |
||||||
|
*/ |
||||||
|
public void setTargetFps(int fps) |
||||||
|
{ |
||||||
|
this.fps = fps; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set whether the window should start in fullscreen |
||||||
|
* |
||||||
|
* @param fullscreen true for fullscreen |
||||||
|
*/ |
||||||
|
public void setFullscreen(boolean fullscreen) |
||||||
|
{ |
||||||
|
this.fullscreen = fullscreen; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() |
||||||
|
{ |
||||||
|
GraphicsModule gfx = app.getBackend().getGraphics(); |
||||||
|
|
||||||
|
gfx.setSize(width, height); |
||||||
|
gfx.setResizable(resizable); |
||||||
|
gfx.setTitle(title); |
||||||
|
gfx.setTargetFps(fps); |
||||||
|
|
||||||
|
if (fullscreen) gfx.setFullscreen(true); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String getName() |
||||||
|
{ |
||||||
|
return "display"; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,74 @@ |
|||||||
|
package mightypork.gamecore.core.init; |
||||||
|
|
||||||
|
|
||||||
|
import java.io.IOException; |
||||||
|
|
||||||
|
import mightypork.gamecore.core.InitTask; |
||||||
|
import mightypork.utils.ion.Ion; |
||||||
|
import mightypork.utils.ion.IonInput; |
||||||
|
import mightypork.utils.ion.IonOutput; |
||||||
|
import mightypork.utils.ion.IonizerBinary; |
||||||
|
import mightypork.utils.math.algo.Coord; |
||||||
|
import mightypork.utils.math.algo.Move; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Register extra ionizables added by the game library (non-native ION types).<br> |
||||||
|
* This initializer can be called anywhere in the initialization sequence. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class InitTaskIonizables extends InitTask { |
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() |
||||||
|
{ |
||||||
|
Ion.registerIndirect(255, new IonizerBinary<Coord>() { |
||||||
|
|
||||||
|
@Override |
||||||
|
public void save(Coord object, IonOutput out) throws IOException |
||||||
|
{ |
||||||
|
out.writeInt(object.x); |
||||||
|
out.writeInt(object.y); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public Coord load(IonInput in) throws IOException |
||||||
|
{ |
||||||
|
final int x = in.readInt(); |
||||||
|
final int y = in.readInt(); |
||||||
|
return new Coord(x, y); |
||||||
|
} |
||||||
|
|
||||||
|
}); |
||||||
|
|
||||||
|
Ion.registerIndirect(254, new IonizerBinary<Move>() { |
||||||
|
|
||||||
|
@Override |
||||||
|
public void save(Move object, IonOutput out) throws IOException |
||||||
|
{ |
||||||
|
out.writeInt(object.x()); |
||||||
|
out.writeInt(object.y()); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public Move load(IonInput in) throws IOException |
||||||
|
{ |
||||||
|
final int x = in.readInt(); |
||||||
|
final int y = in.readInt(); |
||||||
|
return new Move(x, y); |
||||||
|
} |
||||||
|
|
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String getName() |
||||||
|
{ |
||||||
|
return "ion"; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,108 @@ |
|||||||
|
package mightypork.gamecore.core.init; |
||||||
|
|
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
import java.util.logging.Level; |
||||||
|
|
||||||
|
import mightypork.gamecore.core.InitTask; |
||||||
|
import mightypork.gamecore.core.WorkDir; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
import mightypork.utils.logging.writers.LogWriter; |
||||||
|
import mightypork.utils.string.StringUtil; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Init main logger and console log printing.<br> |
||||||
|
* Must be called after workdir is initialized. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class InitTaskLog extends InitTask { |
||||||
|
|
||||||
|
private String logDir = "log"; |
||||||
|
private String logName = "runtime"; |
||||||
|
private int archiveCount = 5; |
||||||
|
|
||||||
|
private Level levelWrite = Level.ALL; |
||||||
|
private Level levelPrint = Level.ALL; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set log directory (relative to workdir).<br> |
||||||
|
* Defaults to "log". |
||||||
|
* |
||||||
|
* @param logDir log directory. |
||||||
|
*/ |
||||||
|
public void setLogDir(String logDir) |
||||||
|
{ |
||||||
|
this.logDir = logDir; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set log name. This name is used as a prefix for archived log files.<br> |
||||||
|
* Should contain only valid filename characters.<br> |
||||||
|
* Defaults to "runtime". |
||||||
|
* |
||||||
|
* @param logName log name |
||||||
|
*/ |
||||||
|
public void setLogName(String logName) |
||||||
|
{ |
||||||
|
if (!StringUtil.isValidFilenameString(logName)) { |
||||||
|
throw new IllegalArgumentException("Invalid log name."); |
||||||
|
} |
||||||
|
|
||||||
|
this.logName = logName; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set number of logs to keep in the logs directory.<br> |
||||||
|
* Set to 0 to keep just the last log, -1 to keep unlimited number of logs.<br> |
||||||
|
* Defaults to 5. |
||||||
|
* |
||||||
|
* @param archiveCount logs to keep |
||||||
|
*/ |
||||||
|
public void setArchiveCount(int archiveCount) |
||||||
|
{ |
||||||
|
this.archiveCount = archiveCount; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set logging levels (minimal level of message to be accepted)<br> |
||||||
|
* Defaults to ALL, ALL. |
||||||
|
* |
||||||
|
* @param levelWrite level for writing to file |
||||||
|
* @param levelPrint level for writing to stdout / stderr |
||||||
|
*/ |
||||||
|
public void setLevels(Level levelWrite, Level levelPrint) |
||||||
|
{ |
||||||
|
this.levelWrite = levelWrite; |
||||||
|
this.levelPrint = levelPrint; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() |
||||||
|
{ |
||||||
|
final LogWriter log = Log.create(logName, new File(WorkDir.getDir(logDir), logName + ".log"), archiveCount); |
||||||
|
Log.setMainLogger(log); |
||||||
|
Log.setLevel(levelWrite); |
||||||
|
Log.setSysoutLevel(levelPrint); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String getName() |
||||||
|
{ |
||||||
|
return "log"; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String[] getDependencies() |
||||||
|
{ |
||||||
|
return new String[] { "workdir" }; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,54 @@ |
|||||||
|
package mightypork.gamecore.core.init; |
||||||
|
|
||||||
|
|
||||||
|
import java.io.IOException; |
||||||
|
|
||||||
|
import mightypork.gamecore.core.InitTask; |
||||||
|
import mightypork.gamecore.core.OptionalInitTask; |
||||||
|
import mightypork.gamecore.core.WorkDir; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* initializer task that writes a system info header to the log file.<br> |
||||||
|
* Must be called after log is initialized. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
@OptionalInitTask |
||||||
|
public class InitTaskLogHeader extends InitTask { |
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() |
||||||
|
{ |
||||||
|
String txt = ""; |
||||||
|
|
||||||
|
txt += "\n### SYSTEM INFO ###\n\n"; |
||||||
|
txt += " Platform ...... " + System.getProperty("os.name") + "\n"; |
||||||
|
txt += " Runtime ....... " + System.getProperty("java.runtime.name") + "\n"; |
||||||
|
txt += " Java .......... " + System.getProperty("java.version") + "\n"; |
||||||
|
txt += " Launch path ... " + System.getProperty("user.dir") + "\n"; |
||||||
|
|
||||||
|
try { |
||||||
|
txt += " Workdir ....... " + WorkDir.getWorkDir().getCanonicalPath() + "\n"; |
||||||
|
} catch (final IOException e) { |
||||||
|
Log.e(e); |
||||||
|
} |
||||||
|
|
||||||
|
Log.i(txt); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String getName() |
||||||
|
{ |
||||||
|
return "log_header"; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String[] getDependencies() |
||||||
|
{ |
||||||
|
return new String[] { "log", "workdir" }; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,136 @@ |
|||||||
|
package mightypork.gamecore.core.init; |
||||||
|
|
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
import java.util.HashMap; |
||||||
|
import java.util.Map; |
||||||
|
import java.util.Map.Entry; |
||||||
|
|
||||||
|
import javax.swing.JOptionPane; |
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.core.InitTask; |
||||||
|
import mightypork.gamecore.core.WorkDir; |
||||||
|
import mightypork.utils.annotations.Stub; |
||||||
|
import mightypork.utils.files.InstanceLock; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Initializer that takes care of setting up the proper workdir. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class InitTaskWorkdir extends InitTask { |
||||||
|
|
||||||
|
private File workdirPath; |
||||||
|
private boolean doLock; |
||||||
|
private String lockFile = ".lock"; |
||||||
|
private Map<String, String> namedPaths = new HashMap<>(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param workdir path to the working directory |
||||||
|
* @param lock whether to lock the directory (single instance mode) |
||||||
|
*/ |
||||||
|
public InitTaskWorkdir(File workdir, boolean lock) { |
||||||
|
this.workdirPath = workdir; |
||||||
|
this.doLock = lock; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set workdir root path |
||||||
|
* |
||||||
|
* @param path workdir path |
||||||
|
*/ |
||||||
|
public void setWorkdirPath(File path) |
||||||
|
{ |
||||||
|
this.workdirPath = path; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set whether the workdir should be locked when the app is running, to |
||||||
|
* prevent other instances from running simultaneously. |
||||||
|
* |
||||||
|
* @param lock |
||||||
|
*/ |
||||||
|
public void setInstanceLock(boolean lock) |
||||||
|
{ |
||||||
|
this.doLock = lock; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set name of the lock file. |
||||||
|
* |
||||||
|
* @param lockFile |
||||||
|
*/ |
||||||
|
public void setLockFileName(String lockFile) |
||||||
|
{ |
||||||
|
this.lockFile = lockFile; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a named path |
||||||
|
* |
||||||
|
* @param alias path alias (snake_case) |
||||||
|
* @param path path (relative to the workdir) |
||||||
|
*/ |
||||||
|
public void addPath(String alias, String path) |
||||||
|
{ |
||||||
|
namedPaths.put(alias, path); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() |
||||||
|
{ |
||||||
|
WorkDir.init(workdirPath); |
||||||
|
|
||||||
|
// lock working directory
|
||||||
|
if (doLock) { |
||||||
|
final File lock = WorkDir.getFile(lockFile); |
||||||
|
if (!InstanceLock.onFile(lock)) { |
||||||
|
onLockError(); |
||||||
|
return; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
for (Entry<String, String> e : namedPaths.entrySet()) { |
||||||
|
WorkDir.addPath(e.getKey(), e.getValue()); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Called when the lock file could not be obtained (cannot write or already |
||||||
|
* exists).<br> |
||||||
|
* Feel free to override this method to define custom behavior. |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
protected void onLockError() |
||||||
|
{ |
||||||
|
Log.e("Could not obtain lock file.\nOnly one instance can run at a time."); |
||||||
|
|
||||||
|
//@formatter:off
|
||||||
|
JOptionPane.showMessageDialog( |
||||||
|
null, |
||||||
|
"Another instance is already running.\n(Delete the "+lockFile +" file in the working directory to override)", |
||||||
|
"Lock Error", |
||||||
|
JOptionPane.ERROR_MESSAGE |
||||||
|
); |
||||||
|
//@formatter:on
|
||||||
|
|
||||||
|
App.shutdown(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String getName() |
||||||
|
{ |
||||||
|
return "workdir"; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,68 @@ |
|||||||
|
package mightypork.gamecore.core.plugins.screenshot; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.InitTask; |
||||||
|
import mightypork.gamecore.core.WorkDir; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Register screenshot plugin to the App. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class InitTaskPluginScreenshot extends InitTask { |
||||||
|
|
||||||
|
private String screenshotDir; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize to use the "screenshots" directory |
||||||
|
*/ |
||||||
|
public InitTaskPluginScreenshot() { |
||||||
|
this("screenshots"); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Initialize to use the given directory for saving. |
||||||
|
* |
||||||
|
* @param dir screenshot dir (relative to workdir) |
||||||
|
*/ |
||||||
|
public InitTaskPluginScreenshot(String dir) { |
||||||
|
this.screenshotDir = dir; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set screenshot directory |
||||||
|
* |
||||||
|
* @param dir screenshot dir (relative to workdir) |
||||||
|
*/ |
||||||
|
public void setScreenshotDir(String dir) |
||||||
|
{ |
||||||
|
this.screenshotDir = dir; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() |
||||||
|
{ |
||||||
|
WorkDir.addPath("_screenshot_dir", screenshotDir); |
||||||
|
app.addPlugin(new ScreenshotPlugin()); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String getName() |
||||||
|
{ |
||||||
|
return "plugin_screenshot"; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public String[] getDependencies() |
||||||
|
{ |
||||||
|
return new String[] { "workdir" }; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,34 @@ |
|||||||
|
package mightypork.gamecore.core.plugins.screenshot; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.core.AppPlugin; |
||||||
|
import mightypork.gamecore.core.events.MainLoopRequest; |
||||||
|
import mightypork.utils.Support; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* This plugin waits for a {@link ScreenshotRequest} event.<br> |
||||||
|
* Upon receiving it, a screenshot is captured and written to file |
||||||
|
* asynchronously. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class ScreenshotPlugin extends AppPlugin { |
||||||
|
|
||||||
|
/** |
||||||
|
* Take screenshot. Called by the trigger event. |
||||||
|
*/ |
||||||
|
void takeScreenshot() |
||||||
|
{ |
||||||
|
App.bus().send(new MainLoopRequest(new Runnable() { |
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() |
||||||
|
{ |
||||||
|
Runnable tts = new TaskTakeScreenshot(); |
||||||
|
Support.runAsThread(tts); |
||||||
|
} |
||||||
|
})); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,22 @@ |
|||||||
|
package mightypork.gamecore.core.plugins.screenshot; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.eventbus.BusEvent; |
||||||
|
import mightypork.utils.eventbus.events.flags.SingleReceiverEvent; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Event used to request screenshot capture. |
||||||
|
* |
||||||
|
* @author MightyPork |
||||||
|
*/ |
||||||
|
@SingleReceiverEvent |
||||||
|
public class ScreenshotRequest extends BusEvent<ScreenshotPlugin> { |
||||||
|
|
||||||
|
@Override |
||||||
|
protected void handleBy(ScreenshotPlugin handler) |
||||||
|
{ |
||||||
|
handler.takeScreenshot(); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,100 @@ |
|||||||
|
package mightypork.gamecore.core.plugins.screenshot; |
||||||
|
|
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
import java.io.IOException; |
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.core.WorkDir; |
||||||
|
import mightypork.gamecore.graphics.Screenshot; |
||||||
|
import mightypork.utils.Support; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Task that takes screenshot and asynchronously saves it to a file.<br> |
||||||
|
* Can be run in a separate thread, but must be instantiated in the render |
||||||
|
* thread. |
||||||
|
* |
||||||
|
* @author MightyPork |
||||||
|
*/ |
||||||
|
public class TaskTakeScreenshot implements Runnable { |
||||||
|
|
||||||
|
private final Screenshot scr; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Take screenshot. Must be called in render thread. |
||||||
|
*/ |
||||||
|
public TaskTakeScreenshot() { |
||||||
|
scr = App.gfx().takeScreenshot(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() |
||||||
|
{ |
||||||
|
// generate unique filename
|
||||||
|
final File file = getScreenshotFile(); |
||||||
|
|
||||||
|
Log.f3("Saving screenshot to file: " + file); |
||||||
|
|
||||||
|
// save to disk
|
||||||
|
try { |
||||||
|
scr.save(file); |
||||||
|
} catch (final IOException e) { |
||||||
|
Log.e("Failed to save screenshot.", e); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return File to save the screenshot to. |
||||||
|
*/ |
||||||
|
protected File getScreenshotFile() |
||||||
|
{ |
||||||
|
final String fname = getBaseFilename(); |
||||||
|
return findFreeFile(fname); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return directory for screenshots |
||||||
|
*/ |
||||||
|
protected File getScreenshotDirectory() |
||||||
|
{ |
||||||
|
return WorkDir.getDir("_screenshot_dir"); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get base filename for the screenshot, without extension. |
||||||
|
* |
||||||
|
* @return filename |
||||||
|
*/ |
||||||
|
protected String getBaseFilename() |
||||||
|
{ |
||||||
|
return Support.getTime("yyyy-MM-dd_HH-mm-ss"); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Find first free filename for the screenshot, by adding -NUMBER after the |
||||||
|
* base filename and before extension. |
||||||
|
* |
||||||
|
* @param base_name base filename |
||||||
|
* @return full path to screenshot file |
||||||
|
*/ |
||||||
|
protected File findFreeFile(String base_name) |
||||||
|
{ |
||||||
|
File file; |
||||||
|
int index = 0; |
||||||
|
while (true) { |
||||||
|
file = new File(getScreenshotDirectory(), base_name + (index > 0 ? "-" + index : "") + ".png"); |
||||||
|
if (!file.exists()) break; |
||||||
|
index++; |
||||||
|
} |
||||||
|
return file; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,16 @@ |
|||||||
|
package mightypork.gamecore.graphics; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.eventbus.BusEvent; |
||||||
|
import mightypork.utils.eventbus.events.flags.SingleReceiverEvent; |
||||||
|
|
||||||
|
|
||||||
|
@SingleReceiverEvent |
||||||
|
public class FullscreenToggleRequest extends BusEvent<GraphicsModule> { |
||||||
|
|
||||||
|
@Override |
||||||
|
protected void handleBy(GraphicsModule handler) |
||||||
|
{ |
||||||
|
handler.switchFullscreen(); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,406 @@ |
|||||||
|
package mightypork.gamecore.graphics; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.BackendModule; |
||||||
|
import mightypork.gamecore.graphics.textures.DeferredTexture; |
||||||
|
import mightypork.gamecore.graphics.textures.TxQuad; |
||||||
|
import mightypork.gamecore.gui.events.ViewportChangeEvent; |
||||||
|
import mightypork.utils.math.color.Color; |
||||||
|
import mightypork.utils.math.color.Grad; |
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
import mightypork.utils.math.constraints.vect.VectConst; |
||||||
|
import mightypork.utils.math.timing.FpsMeter; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Render and display backend module.<br> |
||||||
|
* This module takes care of setting and getting screen size and parameters, |
||||||
|
* drawing on screen and timing render frames. |
||||||
|
* |
||||||
|
* @author MightyPork |
||||||
|
*/ |
||||||
|
public abstract class GraphicsModule extends BackendModule { |
||||||
|
|
||||||
|
protected static final VectConst AXIS_X = Vect.make(1, 0, 0); |
||||||
|
protected static final VectConst AXIS_Y = Vect.make(0, 1, 0); |
||||||
|
protected static final VectConst AXIS_Z = Vect.make(0, 0, 1); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set drawing color |
||||||
|
* |
||||||
|
* @param color color |
||||||
|
*/ |
||||||
|
public abstract void setColor(Color color); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set drawing color, adjust alpha |
||||||
|
* |
||||||
|
* @param color color |
||||||
|
* @param alpha alpha multiplier |
||||||
|
*/ |
||||||
|
public abstract void setColor(Color color, double alpha); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Translate by x, y |
||||||
|
* |
||||||
|
* @param x x offset |
||||||
|
* @param y y offset |
||||||
|
*/ |
||||||
|
public abstract void translate(double x, double y); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Translate by x, y, z |
||||||
|
* |
||||||
|
* @param x x offset |
||||||
|
* @param y y offset |
||||||
|
* @param z z offset |
||||||
|
*/ |
||||||
|
public abstract void translate(double x, double y, double z); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Translate by offset vector |
||||||
|
* |
||||||
|
* @param offset offset coordinate |
||||||
|
*/ |
||||||
|
public abstract void translate(Vect offset); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Translate by offset vector, ignore Z |
||||||
|
* |
||||||
|
* @param offset offset coordinate |
||||||
|
*/ |
||||||
|
public abstract void translateXY(Vect offset); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set scale for translations and coordinates |
||||||
|
* |
||||||
|
* @param x x scale |
||||||
|
* @param y y scale |
||||||
|
*/ |
||||||
|
public abstract void scale(double x, double y); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set scale for translations and coordinates |
||||||
|
* |
||||||
|
* @param x x scale |
||||||
|
* @param y y scale |
||||||
|
* @param z z scale |
||||||
|
*/ |
||||||
|
public abstract void scale(double x, double y, double z); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set scale for translations and coordinates |
||||||
|
* |
||||||
|
* @param scale vector |
||||||
|
*/ |
||||||
|
public abstract void scale(Vect scale); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set scale for translations and coordinates (same value for X and Y scale) |
||||||
|
* |
||||||
|
* @param scale scaling factor |
||||||
|
*/ |
||||||
|
public abstract void scaleXY(double scale); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set X scale for translations and coordinates |
||||||
|
* |
||||||
|
* @param scale scaling factor |
||||||
|
*/ |
||||||
|
public abstract void scaleX(double scale); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set Y scale for translations and coordinates |
||||||
|
* |
||||||
|
* @param scale scaling factor |
||||||
|
*/ |
||||||
|
public abstract void scaleY(double scale); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set Z scale for translations and coordinates |
||||||
|
* |
||||||
|
* @param scale scaling factor |
||||||
|
*/ |
||||||
|
public abstract void scaleZ(double scale); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Rotate coordinate system around X axis |
||||||
|
* |
||||||
|
* @param angle rotation (in degrees) |
||||||
|
*/ |
||||||
|
public abstract void rotateX(double angle); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Rotate coordinate system around Y axis |
||||||
|
* |
||||||
|
* @param angle rotation (in degrees) |
||||||
|
*/ |
||||||
|
public abstract void rotateY(double angle); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Rotate coordinate system around Z axis |
||||||
|
* |
||||||
|
* @param angle rotation (in degrees) |
||||||
|
*/ |
||||||
|
public abstract void rotateZ(double angle); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Rotate coordinate system around given axis |
||||||
|
* |
||||||
|
* @param angle rotation angle |
||||||
|
* @param axis rotation axis (unit vector) |
||||||
|
*/ |
||||||
|
public abstract void rotate(double angle, Vect axis); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Store render state on stack<br> |
||||||
|
* This includes pushGeometry and pushColor. |
||||||
|
*/ |
||||||
|
public abstract void pushState(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Restore state from stack (must be pushed first)<br> |
||||||
|
* This includes popColor and popGeometry. |
||||||
|
*/ |
||||||
|
public abstract void popState(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Store current rotation and translation on stack |
||||||
|
*/ |
||||||
|
public abstract void pushGeometry(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Restore rotation and translation from stack |
||||||
|
*/ |
||||||
|
public abstract void popGeometry(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Store color on stack (so it can later be restored) |
||||||
|
*/ |
||||||
|
public abstract void pushColor(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Restore color from stack (must be pushed first) |
||||||
|
*/ |
||||||
|
public abstract void popColor(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Render 2D quad with currently set color |
||||||
|
* |
||||||
|
* @param rect drawn rect |
||||||
|
*/ |
||||||
|
public abstract void quad(Rect rect); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Render 2D quad with given color.<br> |
||||||
|
* This may change current drawing color. |
||||||
|
* |
||||||
|
* @param rect rectangle |
||||||
|
* @param color draw color |
||||||
|
*/ |
||||||
|
public abstract void quad(Rect rect, Color color); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Render 2D quad with gradient.<br> |
||||||
|
* This may change current drawing color. |
||||||
|
* |
||||||
|
* @param rect rectangle |
||||||
|
* @param grad gradient |
||||||
|
*/ |
||||||
|
public abstract void quad(Rect rect, Grad grad); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Render textured quad with current color |
||||||
|
* |
||||||
|
* @param rect rectangle to draw |
||||||
|
* @param txquad texture quad |
||||||
|
*/ |
||||||
|
public abstract void quad(Rect rect, TxQuad txquad); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Render textured quad with given color |
||||||
|
* |
||||||
|
* @param rect rectangle to draw |
||||||
|
* @param txquad texture instance |
||||||
|
* @param color color tint |
||||||
|
*/ |
||||||
|
public abstract void quad(Rect rect, TxQuad txquad, Color color); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Setup projection for 2D graphics, using current scren size |
||||||
|
*/ |
||||||
|
public abstract void setupProjection(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get backend-flavoured lazy texture |
||||||
|
* |
||||||
|
* @param path path to texture |
||||||
|
* @return the lazy texture |
||||||
|
*/ |
||||||
|
public abstract DeferredTexture getLazyTexture(String path); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set target fps (for syncing in endFrame() call).<br> |
||||||
|
* With vsync enabled, the target fps may not be met. |
||||||
|
* |
||||||
|
* @param fps requested fps |
||||||
|
*/ |
||||||
|
public abstract void setTargetFps(int fps); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set fullscreen. The fullscreen state will be changed when possible (eg. |
||||||
|
* at the end of the current frame) and a {@link ViewportChangeEvent} will |
||||||
|
* be fired. |
||||||
|
* |
||||||
|
* @param fs true for fullscreen |
||||||
|
*/ |
||||||
|
public abstract void setFullscreen(boolean fs); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Request fullscreen toggle. See setFullscreen() for more info) |
||||||
|
*/ |
||||||
|
public abstract void switchFullscreen(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get fullscreen state (note that methods changing fullscreen may not have |
||||||
|
* immediate effect, so this method may report the old state if the |
||||||
|
* fullscreen state has not yet been changed). |
||||||
|
* |
||||||
|
* @return is fullscreen |
||||||
|
*/ |
||||||
|
public abstract boolean isFullscreen(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Take screenshot (expensive processing should be done in separate thread |
||||||
|
* when screenshot is saved).<br> |
||||||
|
* This method is utilized by the Screenshot plugin. |
||||||
|
* |
||||||
|
* @return screenshot object |
||||||
|
*/ |
||||||
|
public abstract Screenshot takeScreenshot(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Start a render frame - clear buffers, prepare rendering context etc. |
||||||
|
*/ |
||||||
|
public abstract void beginFrame(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* End a render frame: flip buffers, sync to fps... |
||||||
|
*/ |
||||||
|
public abstract void endFrame(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set display dimensions |
||||||
|
* |
||||||
|
* @param width display width (pixels) |
||||||
|
* @param height display height (pixels) |
||||||
|
*/ |
||||||
|
public abstract void setSize(int width, int height); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set window titlebar text |
||||||
|
* |
||||||
|
* @param title titlebar text |
||||||
|
*/ |
||||||
|
public abstract void setTitle(String title); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Enable or disable VSync |
||||||
|
* |
||||||
|
* @param vsync true for vsync enabled |
||||||
|
*/ |
||||||
|
public abstract void setVSync(boolean vsync); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set window resizable / fixed |
||||||
|
* |
||||||
|
* @param resizable true for resizable |
||||||
|
*/ |
||||||
|
public abstract void setResizable(boolean resizable); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get screen rect. Should always return the same Rect instance. |
||||||
|
* |
||||||
|
* @return the rect |
||||||
|
*/ |
||||||
|
public abstract Rect getRect(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get current FPS (eg. measured by a {@link FpsMeter}) |
||||||
|
* |
||||||
|
* @return current FPS |
||||||
|
*/ |
||||||
|
public abstract long getFps(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get screen center. Should always return the same Vect instance. |
||||||
|
* |
||||||
|
* @return screen center. |
||||||
|
*/ |
||||||
|
public abstract Vect getCenter(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get screen size. Should always return the same Vect instance. |
||||||
|
* |
||||||
|
* @return size |
||||||
|
*/ |
||||||
|
public abstract Vect getSize(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return screen width |
||||||
|
*/ |
||||||
|
public abstract int getWidth(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return screen height |
||||||
|
*/ |
||||||
|
public abstract int getHeight(); |
||||||
|
} |
@ -0,0 +1,16 @@ |
|||||||
|
package mightypork.gamecore.graphics; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Can be rendered |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public interface Renderable { |
||||||
|
|
||||||
|
/** |
||||||
|
* Render on screen. |
||||||
|
*/ |
||||||
|
void render(); |
||||||
|
|
||||||
|
} |
@ -0,0 +1,34 @@ |
|||||||
|
package mightypork.gamecore.graphics; |
||||||
|
|
||||||
|
|
||||||
|
import java.io.File; |
||||||
|
import java.io.IOException; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* <p> |
||||||
|
* Screenshot object used to save screenshot to a file. The Screenshot object is |
||||||
|
* created by the Graphics module. |
||||||
|
* </p> |
||||||
|
* <p> |
||||||
|
* Screenshot typically takes a byte buffer and converts it to image before |
||||||
|
* saving to file. This image can be cached to speed up repeated saving. |
||||||
|
* </p> |
||||||
|
* <p> |
||||||
|
* Once created (passing byte buffer in constructor), the Screenshot should be |
||||||
|
* safe to process (call the save() method) in separate thread. |
||||||
|
* </p> |
||||||
|
* |
||||||
|
* @author MightyPork |
||||||
|
*/ |
||||||
|
public interface Screenshot { |
||||||
|
|
||||||
|
/** |
||||||
|
* Process byte buffer and write image to a file.<br> |
||||||
|
* Image can be cached for future save. |
||||||
|
* |
||||||
|
* @param file target file |
||||||
|
* @throws IOException on error writing to file |
||||||
|
*/ |
||||||
|
void save(File file) throws IOException; |
||||||
|
} |
@ -0,0 +1,98 @@ |
|||||||
|
package mightypork.gamecore.graphics.fonts; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.graphics.textures.FilterMode; |
||||||
|
import mightypork.gamecore.resources.BaseDeferredResource; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Abstract deferred font stub. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class DeferredFont extends BaseDeferredResource implements IFont { |
||||||
|
|
||||||
|
public static enum FontStyle |
||||||
|
{ |
||||||
|
PLAIN(0), BOLD(1), ITALIC(2), BOLD_ITALIC(3); |
||||||
|
|
||||||
|
public int numval; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Font style |
||||||
|
* |
||||||
|
* @param style style index as in awt Font. Not using constants to be |
||||||
|
* independent on awt. |
||||||
|
*/ |
||||||
|
private FontStyle(int style) { |
||||||
|
this.numval = style; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
protected double size = 12; |
||||||
|
protected FontStyle style = FontStyle.PLAIN; |
||||||
|
protected String chars = Glyphs.basic; |
||||||
|
protected FilterMode filter = FilterMode.NEAREST; |
||||||
|
protected boolean antialias = false; |
||||||
|
protected double discardTop = 0; |
||||||
|
protected double discardBottom = 0; |
||||||
|
|
||||||
|
|
||||||
|
public DeferredFont(String resource) { |
||||||
|
super(resource); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setSize(double size) |
||||||
|
{ |
||||||
|
this.size = size; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setStyle(FontStyle style) |
||||||
|
{ |
||||||
|
this.style = style; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setChars(String chars) |
||||||
|
{ |
||||||
|
this.chars = chars; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setFilter(FilterMode filter) |
||||||
|
{ |
||||||
|
this.filter = filter; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setAntialias(boolean antialias) |
||||||
|
{ |
||||||
|
this.antialias = antialias; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setDiscardRatio(double top, double bottom) |
||||||
|
{ |
||||||
|
discardTop = top; |
||||||
|
discardBottom = bottom; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public double getTopDiscardRatio() |
||||||
|
{ |
||||||
|
return discardTop; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public double getBottomDiscardRatio() |
||||||
|
{ |
||||||
|
return discardBottom; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,79 @@ |
|||||||
|
package mightypork.gamecore.graphics.fonts; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.HashMap; |
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.resources.loading.ResourceLoadRequest; |
||||||
|
import mightypork.utils.eventbus.clients.BusNode; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Font loader and registry |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class FontRegistry extends BusNode { |
||||||
|
|
||||||
|
private final HashMap<String, IFont> fonts = new HashMap<>(); |
||||||
|
private final HashMap<String, String> aliases = new HashMap<>(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Load a {@link DeferredLwjglFont} |
||||||
|
* |
||||||
|
* @param key font key |
||||||
|
* @param font font instance |
||||||
|
*/ |
||||||
|
public void addFont(String key, DeferredFont font) |
||||||
|
{ |
||||||
|
App.bus().send(new ResourceLoadRequest(font)); |
||||||
|
|
||||||
|
fonts.put(key, font); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a {@link IFont} to the bank. |
||||||
|
* |
||||||
|
* @param key font key |
||||||
|
* @param font font instance |
||||||
|
*/ |
||||||
|
public void addFont(String key, IFont font) |
||||||
|
{ |
||||||
|
fonts.put(key, font); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a font alias. |
||||||
|
* |
||||||
|
* @param alias_key alias key |
||||||
|
* @param font_key font key |
||||||
|
*/ |
||||||
|
public void addAlias(String alias_key, String font_key) |
||||||
|
{ |
||||||
|
aliases.put(alias_key, font_key); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get a loaded {@link Texture} |
||||||
|
* |
||||||
|
* @param key texture key |
||||||
|
* @return the texture |
||||||
|
*/ |
||||||
|
public IFont getFont(String key) |
||||||
|
{ |
||||||
|
IFont f = fonts.get(key); |
||||||
|
|
||||||
|
if (f == null) f = fonts.get(aliases.get(key)); |
||||||
|
|
||||||
|
if (f == null) { |
||||||
|
throw new RuntimeException("There's no font called " + key + "!"); |
||||||
|
} |
||||||
|
|
||||||
|
return f; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,213 @@ |
|||||||
|
package mightypork.gamecore.graphics.fonts; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.utils.math.AlignX; |
||||||
|
import mightypork.utils.math.color.Color; |
||||||
|
import mightypork.utils.math.color.pal.RGB; |
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Font renderer |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class FontRenderer { |
||||||
|
|
||||||
|
private IFont font; |
||||||
|
|
||||||
|
private Color color; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param font used font |
||||||
|
*/ |
||||||
|
public FontRenderer(IFont font) { |
||||||
|
this(font, RGB.WHITE); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param font used font |
||||||
|
* @param color drawing color |
||||||
|
*/ |
||||||
|
public FontRenderer(IFont font, Color color) { |
||||||
|
this.font = font; |
||||||
|
this.color = color; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get region needed to draw text at size |
||||||
|
* |
||||||
|
* @param text text to draw |
||||||
|
* @param height drawing height |
||||||
|
* @return taken space (width, height) |
||||||
|
*/ |
||||||
|
public Vect getNeededSpace(String text, double height) |
||||||
|
{ |
||||||
|
return font.getNeededSpace(text).mul(getScale(height)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get width needed to draw text at size |
||||||
|
* |
||||||
|
* @param text text to draw |
||||||
|
* @param height drawing height |
||||||
|
* @return needed width |
||||||
|
*/ |
||||||
|
public double getWidth(String text, double height) |
||||||
|
{ |
||||||
|
return getNeededSpace(text, height).x(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
private double getScale(double height) |
||||||
|
{ |
||||||
|
return height / font.getLineHeight(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Change drawing font |
||||||
|
* |
||||||
|
* @param font font to use for drawing |
||||||
|
*/ |
||||||
|
public void setFont(IFont font) |
||||||
|
{ |
||||||
|
this.font = font; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set drawing color |
||||||
|
* |
||||||
|
* @param color color |
||||||
|
*/ |
||||||
|
public void setColor(Color color) |
||||||
|
{ |
||||||
|
this.color = color; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Draw on screen |
||||||
|
* |
||||||
|
* @param text text to draw |
||||||
|
* @param pos origin (min coord) |
||||||
|
* @param height drawing height |
||||||
|
* @param color drawing color |
||||||
|
*/ |
||||||
|
public void draw(String text, Vect pos, double height, Color color) |
||||||
|
{ |
||||||
|
App.gfx().pushGeometry(); |
||||||
|
|
||||||
|
final double sc = getScale(height); |
||||||
|
|
||||||
|
App.gfx().translate(pos.x(), pos.y()); |
||||||
|
App.gfx().scaleXY(sc); |
||||||
|
|
||||||
|
font.draw(text, color); |
||||||
|
|
||||||
|
App.gfx().popGeometry(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Draw on screen |
||||||
|
* |
||||||
|
* @param text text to draw |
||||||
|
* @param bounds drawing bounds (height for font height, horizontal bounds |
||||||
|
* for align) |
||||||
|
* @param align horizontal alignment (with respect to bounds) |
||||||
|
*/ |
||||||
|
public void draw(String text, Rect bounds, AlignX align) |
||||||
|
{ |
||||||
|
this.draw(text, bounds, align, this.color); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Draw on screen |
||||||
|
* |
||||||
|
* @param text text to draw |
||||||
|
* @param bounds drawing bounds (height for font height, horizontal bounds |
||||||
|
* for align) |
||||||
|
* @param align horizontal alignment (with respect to bounds) |
||||||
|
* @param color drawing color |
||||||
|
*/ |
||||||
|
public void draw(String text, Rect bounds, AlignX align, Color color) |
||||||
|
{ |
||||||
|
Vect start; |
||||||
|
|
||||||
|
switch (align) { |
||||||
|
case LEFT: |
||||||
|
start = bounds.topLeft(); |
||||||
|
break; |
||||||
|
|
||||||
|
case CENTER: |
||||||
|
start = bounds.topCenter(); |
||||||
|
break; |
||||||
|
|
||||||
|
case RIGHT: |
||||||
|
default: |
||||||
|
start = bounds.topRight(); |
||||||
|
break; |
||||||
|
} |
||||||
|
|
||||||
|
draw(text, start, bounds.height().value(), align, color); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Draw on screen |
||||||
|
* |
||||||
|
* @param text text to draw |
||||||
|
* @param pos origin (min coord) |
||||||
|
* @param height drawing height |
||||||
|
* @param align horizontal alignment |
||||||
|
*/ |
||||||
|
public void draw(String text, Vect pos, double height, AlignX align) |
||||||
|
{ |
||||||
|
draw(text, pos, height, align, this.color); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Draw on screen |
||||||
|
* |
||||||
|
* @param text text to draw |
||||||
|
* @param pos origin (min coord) |
||||||
|
* @param height drawing height |
||||||
|
* @param align horizontal alignment |
||||||
|
* @param color drawing color |
||||||
|
*/ |
||||||
|
public void draw(String text, Vect pos, double height, AlignX align, Color color) |
||||||
|
{ |
||||||
|
|
||||||
|
final double w = getWidth(text, height); |
||||||
|
|
||||||
|
Vect start; |
||||||
|
|
||||||
|
switch (align) { |
||||||
|
case LEFT: |
||||||
|
start = pos; |
||||||
|
break; |
||||||
|
|
||||||
|
case CENTER: |
||||||
|
start = pos.sub(w / 2D, 0); |
||||||
|
break; |
||||||
|
|
||||||
|
case RIGHT: |
||||||
|
default: |
||||||
|
start = pos.sub(w, 0); |
||||||
|
break; |
||||||
|
} |
||||||
|
|
||||||
|
draw(text, start, height, color); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,76 @@ |
|||||||
|
package mightypork.gamecore.graphics.fonts; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.math.color.Color; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Interface bor drawable font. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public interface IFont { |
||||||
|
|
||||||
|
/** |
||||||
|
* Draw without scaling at (0, 0) in given color. |
||||||
|
* |
||||||
|
* @param text text to draw |
||||||
|
* @param color draw color |
||||||
|
*/ |
||||||
|
void draw(String text, Color color); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get suize needed to render give string |
||||||
|
* |
||||||
|
* @param text string to check |
||||||
|
* @return coord (width, height) |
||||||
|
*/ |
||||||
|
Vect getNeededSpace(String text); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return font height |
||||||
|
*/ |
||||||
|
int getLineHeight(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param text texted text |
||||||
|
* @return space needed |
||||||
|
*/ |
||||||
|
int getWidth(String text); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return specified font size |
||||||
|
*/ |
||||||
|
int getFontSize(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set what vertical ratio of the font size is blank and should be cut off |
||||||
|
* when rendering |
||||||
|
* |
||||||
|
* @param top top ratio (0-1) |
||||||
|
* @param bottom bottom ratio (0-1) |
||||||
|
*/ |
||||||
|
void setDiscardRatio(double top, double bottom); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get top discard ratio (blank unused space) |
||||||
|
* |
||||||
|
* @return ratio |
||||||
|
*/ |
||||||
|
double getTopDiscardRatio(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get bottom discard ratio (blank unused space) |
||||||
|
* |
||||||
|
* @return ratio |
||||||
|
*/ |
||||||
|
double getBottomDiscardRatio(); |
||||||
|
} |
@ -0,0 +1,57 @@ |
|||||||
|
package mightypork.gamecore.graphics.textures; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.resources.BaseDeferredResource; |
||||||
|
import mightypork.gamecore.resources.loading.MustLoadInRenderingContext; |
||||||
|
import mightypork.utils.annotations.Alias; |
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Deferred texture (to be extended by backend texture) |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
@Alias(name = "Texture") |
||||||
|
@MustLoadInRenderingContext |
||||||
|
public abstract class DeferredTexture extends BaseDeferredResource implements ITexture { |
||||||
|
|
||||||
|
protected FilterMode filter = FilterMode.NEAREST; |
||||||
|
protected WrapMode wrap = WrapMode.CLAMP; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param resourcePath resource path |
||||||
|
*/ |
||||||
|
public DeferredTexture(String resourcePath) { |
||||||
|
super(resourcePath); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public TxQuad makeQuad(Rect uvs) |
||||||
|
{ |
||||||
|
return new TxQuad(this, uvs); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setFilter(FilterMode filterMin) |
||||||
|
{ |
||||||
|
this.filter = filterMin; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setWrap(WrapMode wrapping) |
||||||
|
{ |
||||||
|
this.wrap = wrapping; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public QuadGrid grid(int x, int y) |
||||||
|
{ |
||||||
|
return new QuadGrid(this, x, y); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,12 @@ |
|||||||
|
package mightypork.gamecore.graphics.textures; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Texture filtering mode |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public enum FilterMode |
||||||
|
{ |
||||||
|
LINEAR, NEAREST; |
||||||
|
} |
@ -0,0 +1,64 @@ |
|||||||
|
package mightypork.gamecore.graphics.textures; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.interfaces.Destroyable; |
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Texture interface, backend independent |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public interface ITexture extends Destroyable { |
||||||
|
|
||||||
|
/** |
||||||
|
* Set filter for scaling |
||||||
|
* |
||||||
|
* @param filter filter |
||||||
|
*/ |
||||||
|
void setFilter(FilterMode filter); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param wrapping wrap mode |
||||||
|
*/ |
||||||
|
void setWrap(WrapMode wrapping); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get a quad from this texture of given position/size |
||||||
|
* |
||||||
|
* @param uvs quad rect |
||||||
|
* @return the quad |
||||||
|
*/ |
||||||
|
TxQuad makeQuad(Rect uvs); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get a grid for given number of tiles |
||||||
|
* |
||||||
|
* @param x horizontal tile count |
||||||
|
* @param y vertical tile count |
||||||
|
* @return grid |
||||||
|
*/ |
||||||
|
QuadGrid grid(int x, int y); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return source image width (corresponding to width01) |
||||||
|
*/ |
||||||
|
int getImageWidth(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return source image height (corresponding to height01) |
||||||
|
*/ |
||||||
|
int getImageHeight(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return true if the image is RGBA |
||||||
|
*/ |
||||||
|
boolean hasAlpha(); |
||||||
|
} |
@ -0,0 +1,92 @@ |
|||||||
|
package mightypork.gamecore.graphics.textures; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* {@link TxQuad} and {@link TxSheet} building utility |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class QuadGrid { |
||||||
|
|
||||||
|
private final ITexture tx; |
||||||
|
private final int txHeight; |
||||||
|
private final int txWidth; |
||||||
|
private final double tileW; |
||||||
|
private final double tileH; |
||||||
|
|
||||||
|
|
||||||
|
public QuadGrid(ITexture tx, int tilesX, int tilesY) { |
||||||
|
this.tx = tx; |
||||||
|
this.txWidth = tilesX; |
||||||
|
this.txHeight = tilesY; |
||||||
|
this.tileW = 1D / tilesX; |
||||||
|
this.tileH = 1D / tilesY; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Make square quad at given coords (one grid cell) |
||||||
|
* |
||||||
|
* @param x x coordinate (cells) |
||||||
|
* @param y y coordinate (cells) |
||||||
|
* @return the quad |
||||||
|
*/ |
||||||
|
public TxQuad makeQuad(double x, double y) |
||||||
|
{ |
||||||
|
if (x < 0 || x >= txWidth || y < 0 || y >= txHeight) { |
||||||
|
throw new IndexOutOfBoundsException("Requested invalid txquad coordinates."); |
||||||
|
} |
||||||
|
|
||||||
|
return makeQuad(x, y, 1, 1); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Make square quad at given coords, with arbitrary size. Coordinates are |
||||||
|
* multiples of cell size. |
||||||
|
* |
||||||
|
* @param x x coordinate (cells) |
||||||
|
* @param y y coordinate (cells) |
||||||
|
* @param width width (cells) |
||||||
|
* @param height height (cells) |
||||||
|
* @return the quad |
||||||
|
*/ |
||||||
|
public TxQuad makeQuad(double x, double y, double width, double height) |
||||||
|
{ |
||||||
|
if (x < 0 || x >= txWidth || y < 0 || y >= txHeight) { |
||||||
|
throw new IndexOutOfBoundsException("Requested invalid txquad coordinates."); |
||||||
|
} |
||||||
|
|
||||||
|
if (x + width > txWidth || y + height > txHeight) { |
||||||
|
throw new IndexOutOfBoundsException("Requested invalid txquad size (would go beyond texture size)."); |
||||||
|
} |
||||||
|
|
||||||
|
return tx.makeQuad(Rect.make(tileW * x, tileH * y, tileW * width, tileH * height)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Make a sheet. |
||||||
|
* |
||||||
|
* @param x x origin coordinate (cells) |
||||||
|
* @param y y origin coordinate (cells) |
||||||
|
* @param width width (cells) |
||||||
|
* @param height height (cells) |
||||||
|
* @return the sheet |
||||||
|
*/ |
||||||
|
public TxSheet makeSheet(double x, double y, double width, double height) |
||||||
|
{ |
||||||
|
if (x < 0 || x >= txWidth || y < 0 || y >= txHeight) { |
||||||
|
throw new IndexOutOfBoundsException("Requested invalid txquad coordinates."); |
||||||
|
} |
||||||
|
|
||||||
|
if (x + width > txWidth || y + height > txHeight) { |
||||||
|
throw new IndexOutOfBoundsException("Requested invalid txsheet size (would go beyond texture size)."); |
||||||
|
} |
||||||
|
|
||||||
|
return makeQuad(x, y).makeSheet(width, height); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,143 @@ |
|||||||
|
package mightypork.gamecore.graphics.textures; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.HashMap; |
||||||
|
import java.util.Map; |
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.resources.loading.ResourceLoadRequest; |
||||||
|
import mightypork.utils.exceptions.KeyAlreadyExistsException; |
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Texture storage and quad/sheet registry. Quads and Sheets are interchangeable |
||||||
|
* once registered. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class TextureRegistry { |
||||||
|
|
||||||
|
private final Map<String, ITexture> textures = new HashMap<>(); |
||||||
|
private final Map<String, TxSheet> sheets = new HashMap<>(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Load a texture from resource, without a key. This texture will not be |
||||||
|
* added to the bank. |
||||||
|
* |
||||||
|
* @param resourcePath resource path of the texture |
||||||
|
* @param filter |
||||||
|
* @param wrap |
||||||
|
* @return texture reference |
||||||
|
*/ |
||||||
|
public ITexture addTexture(String resourcePath, FilterMode filter, WrapMode wrap) |
||||||
|
{ |
||||||
|
return addTexture(resourcePath, resourcePath, filter, wrap); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Load a texture from resource; if key is not null, the texture will be |
||||||
|
* added to the bank. |
||||||
|
* |
||||||
|
* @param key texture key, can be null. |
||||||
|
* @param resourcePath resource path of the texture |
||||||
|
* @param filter |
||||||
|
* @param wrap |
||||||
|
* @return texture reference |
||||||
|
*/ |
||||||
|
public ITexture addTexture(String key, String resourcePath, FilterMode filter, WrapMode wrap) |
||||||
|
{ |
||||||
|
if (key != null) if (textures.containsKey(key)) throw new KeyAlreadyExistsException(); |
||||||
|
|
||||||
|
final DeferredTexture texture = App.gfx().getLazyTexture(resourcePath); |
||||||
|
texture.setFilter(filter); |
||||||
|
texture.setWrap(wrap); |
||||||
|
|
||||||
|
App.bus().send(new ResourceLoadRequest(texture)); |
||||||
|
|
||||||
|
if (key != null) { |
||||||
|
textures.put(key, texture); |
||||||
|
add(key, texture.makeQuad(Rect.ONE)); |
||||||
|
} |
||||||
|
|
||||||
|
return texture; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add already created quad to the quad registry |
||||||
|
* |
||||||
|
* @param quadKey key |
||||||
|
* @param quad quad to add |
||||||
|
*/ |
||||||
|
public void add(String quadKey, TxQuad quad) |
||||||
|
{ |
||||||
|
if (sheets.containsKey(quadKey)) throw new KeyAlreadyExistsException(); |
||||||
|
|
||||||
|
sheets.put(quadKey, quad.makeSheet(1, 1)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add an already created sheet |
||||||
|
* |
||||||
|
* @param sheetKey key |
||||||
|
* @param sheet sheet to add |
||||||
|
*/ |
||||||
|
public void add(String sheetKey, TxSheet sheet) |
||||||
|
{ |
||||||
|
if (sheets.containsKey(sheetKey)) throw new KeyAlreadyExistsException(); |
||||||
|
|
||||||
|
sheets.put(sheetKey, sheet); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get a {@link TxQuad} for key; if it was added as sheet, the first quad |
||||||
|
* ofthe sheet is returned. |
||||||
|
* |
||||||
|
* @param key quad key |
||||||
|
* @return the quad |
||||||
|
*/ |
||||||
|
public TxQuad getQuad(String key) |
||||||
|
{ |
||||||
|
return getSheet(key).getQuad(0); // get the first
|
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get a loaded {@link ITexture} |
||||||
|
* |
||||||
|
* @param key texture key |
||||||
|
* @return the texture |
||||||
|
*/ |
||||||
|
public ITexture getTexture(String key) |
||||||
|
{ |
||||||
|
final ITexture tx = textures.get(key); |
||||||
|
|
||||||
|
if (tx == null) throw new RuntimeException("There's no texture called \"" + key + "\"!"); |
||||||
|
|
||||||
|
return tx; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get a {@link TxSheet} for key |
||||||
|
* |
||||||
|
* @param key sheet key |
||||||
|
* @return the sheet |
||||||
|
*/ |
||||||
|
public TxSheet getSheet(String key) |
||||||
|
{ |
||||||
|
final TxSheet sh = sheets.get(key); |
||||||
|
|
||||||
|
if (sh == null) { |
||||||
|
throw new RuntimeException("There's no sheet called \"" + key + "\"!"); |
||||||
|
} |
||||||
|
|
||||||
|
return sh; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,164 @@ |
|||||||
|
package mightypork.gamecore.graphics.textures; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
import mightypork.utils.math.constraints.rect.RectConst; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Texture Quad (describing a part of a texture) |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class TxQuad { |
||||||
|
|
||||||
|
/** The texture */ |
||||||
|
public final ITexture tx; |
||||||
|
/** Coords in texture (0-1) */ |
||||||
|
public final RectConst uvs; |
||||||
|
|
||||||
|
private boolean flipX; |
||||||
|
private boolean flipY; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* TxQuad from origin and size in pixels |
||||||
|
* |
||||||
|
* @param tx texture |
||||||
|
* @param xPx left top X (0-1) |
||||||
|
* @param yPx left top Y (0-1) |
||||||
|
* @param widthPx area width (0-1) |
||||||
|
* @param heightPx area height (0-1) |
||||||
|
* @return new TxQuad |
||||||
|
*/ |
||||||
|
public static TxQuad fromSizePx(ITexture tx, double xPx, double yPx, double widthPx, double heightPx) |
||||||
|
{ |
||||||
|
final double w = tx.getImageWidth(); |
||||||
|
final double h = tx.getImageHeight(); |
||||||
|
|
||||||
|
return fromSize(tx, xPx / w, yPx / h, widthPx / w, heightPx / h); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* TxQuad from origin and size 0-1 |
||||||
|
* |
||||||
|
* @param tx texture |
||||||
|
* @param x1 left top X (0-1) |
||||||
|
* @param y1 left top Y (0-1) |
||||||
|
* @param width area width (0-1) |
||||||
|
* @param height area height (0-1) |
||||||
|
* @return new TxQuad |
||||||
|
*/ |
||||||
|
public static TxQuad fromSize(ITexture tx, double x1, double y1, double width, double height) |
||||||
|
{ |
||||||
|
return new TxQuad(tx, x1, y1, x1 + width, y1 + height); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Make of coords |
||||||
|
* |
||||||
|
* @param tx texture |
||||||
|
* @param x1 left top X (0-1) |
||||||
|
* @param y1 left top Y (0-1) |
||||||
|
* @param x2 right bottom X (0-1) |
||||||
|
* @param y2 right bottom Y (0-1) |
||||||
|
*/ |
||||||
|
public TxQuad(ITexture tx, double x1, double y1, double x2, double y2) { |
||||||
|
this(tx, Rect.make(x1, y1, x2, y2)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param tx Texture |
||||||
|
* @param uvs Rect of texture UVs (0-1); will be frozen. |
||||||
|
*/ |
||||||
|
public TxQuad(ITexture tx, Rect uvs) { |
||||||
|
this.tx = tx; |
||||||
|
this.uvs = uvs.freeze(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Clone another |
||||||
|
* |
||||||
|
* @param txQuad a copied quad |
||||||
|
*/ |
||||||
|
public TxQuad(TxQuad txQuad) { |
||||||
|
this.tx = txQuad.tx; |
||||||
|
this.uvs = txQuad.uvs; |
||||||
|
this.flipX = txQuad.flipX; |
||||||
|
this.flipY = txQuad.flipY; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get copy |
||||||
|
* |
||||||
|
* @return copy of this |
||||||
|
*/ |
||||||
|
public TxQuad copy() |
||||||
|
{ |
||||||
|
return new TxQuad(this); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Make a sheet starting with this quad, spannign to right and down. |
||||||
|
* |
||||||
|
* @param width sheet width |
||||||
|
* @param height sheet height |
||||||
|
* @return sheet |
||||||
|
*/ |
||||||
|
public TxSheet makeSheet(double width, double height) |
||||||
|
{ |
||||||
|
return new TxSheet(this, (int) Math.round(width), (int) Math.round(height)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return copy flipped X |
||||||
|
*/ |
||||||
|
public TxQuad flipX() |
||||||
|
{ |
||||||
|
final TxQuad copy = new TxQuad(this); |
||||||
|
copy.flipX ^= true; |
||||||
|
return copy; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return copy flipped Y |
||||||
|
*/ |
||||||
|
public TxQuad flipY() |
||||||
|
{ |
||||||
|
final TxQuad copy = new TxQuad(this); |
||||||
|
copy.flipY ^= true; |
||||||
|
return copy; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public boolean isFlippedY() |
||||||
|
{ |
||||||
|
return flipY; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public boolean isFlippedX() |
||||||
|
{ |
||||||
|
return flipX; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Use the same flit/other attributes as the original txQuad |
||||||
|
* |
||||||
|
* @param original |
||||||
|
*/ |
||||||
|
public void dupeAttrs(TxQuad original) |
||||||
|
{ |
||||||
|
this.flipX = original.flipX; |
||||||
|
this.flipY = original.flipY; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,121 @@ |
|||||||
|
package mightypork.gamecore.graphics.textures; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.Random; |
||||||
|
|
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Basic sprite sheet |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class TxSheet { |
||||||
|
|
||||||
|
private final TxQuad original; |
||||||
|
private final TxQuad[] sprites; |
||||||
|
private final int width; |
||||||
|
|
||||||
|
private final Random rand = new Random(); |
||||||
|
private final Random randForSeed = new Random(); |
||||||
|
private final int count; |
||||||
|
|
||||||
|
|
||||||
|
public TxSheet(TxQuad tx, int width, int height) { |
||||||
|
this.original = tx; |
||||||
|
this.width = width; |
||||||
|
this.count = width * height; |
||||||
|
|
||||||
|
this.sprites = new TxQuad[count]; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return number of quads |
||||||
|
*/ |
||||||
|
public int getQuadCount() |
||||||
|
{ |
||||||
|
return count; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get a quad based on ratio 0-1 (0: first, 1: last) |
||||||
|
* |
||||||
|
* @param ratio ratio |
||||||
|
* @return quad |
||||||
|
*/ |
||||||
|
public TxQuad getQuad(double ratio) |
||||||
|
{ |
||||||
|
return getQuad((int) Math.round((count - 1) * ratio)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get quad of index |
||||||
|
* |
||||||
|
* @param index index |
||||||
|
* @return the quad |
||||||
|
*/ |
||||||
|
public TxQuad getQuad(int index) |
||||||
|
{ |
||||||
|
if (index < 0 || index >= count) { |
||||||
|
Log.w("Index out of bounds: " + index + ", allowed: 0.." + count); |
||||||
|
index = index % count; |
||||||
|
} |
||||||
|
|
||||||
|
// lazy - init only when needed
|
||||||
|
if (sprites[index] == null) { |
||||||
|
final int x = index % width; |
||||||
|
final int y = index / width; |
||||||
|
|
||||||
|
final double origW = original.uvs.width().value(); |
||||||
|
final double origH = original.uvs.height().value(); |
||||||
|
|
||||||
|
final TxQuad txq = new TxQuad(original.tx, original.uvs.move(x * origW, y * origH)); |
||||||
|
txq.dupeAttrs(original); |
||||||
|
|
||||||
|
sprites[index] = txq; |
||||||
|
} |
||||||
|
|
||||||
|
return sprites[index]; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get entirely random TxQuad from this sheet |
||||||
|
* |
||||||
|
* @return the picked quad |
||||||
|
*/ |
||||||
|
public TxQuad getRandomQuad() |
||||||
|
{ |
||||||
|
return getQuad(rand.nextInt(count)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get random TxQuad from this sheet |
||||||
|
* |
||||||
|
* @param seed random number generator seed |
||||||
|
* @return the picked quad |
||||||
|
*/ |
||||||
|
public TxQuad getRandomQuad(long seed) |
||||||
|
{ |
||||||
|
randForSeed.setSeed(seed); |
||||||
|
return getQuad(randForSeed.nextInt(count)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get random TxQuad from this sheet |
||||||
|
* |
||||||
|
* @param seed random number generator seed (double will be converted to |
||||||
|
* long) |
||||||
|
* @return the picked quad |
||||||
|
*/ |
||||||
|
public TxQuad getRandomQuad(double seed) |
||||||
|
{ |
||||||
|
return getRandomQuad(Double.doubleToLongBits(seed)); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,12 @@ |
|||||||
|
package mightypork.gamecore.graphics.textures; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Texture wrap mode |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public enum WrapMode |
||||||
|
{ |
||||||
|
CLAMP, REPEAT; |
||||||
|
} |
@ -0,0 +1,54 @@ |
|||||||
|
package mightypork.gamecore.gui; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.interfaces.Enableable; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Triggered action |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class Action implements Runnable, Enableable { |
||||||
|
|
||||||
|
private boolean enabled = true; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Enable the action |
||||||
|
* |
||||||
|
* @param enable true to enable |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public final void setEnabled(boolean enable) |
||||||
|
{ |
||||||
|
this.enabled = enable; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return true if this action is enabled. |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public final boolean isEnabled() |
||||||
|
{ |
||||||
|
return enabled; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Run the action, if it's enabled. |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public final void run() |
||||||
|
{ |
||||||
|
if (enabled) execute(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Do the work. |
||||||
|
*/ |
||||||
|
protected abstract void execute(); |
||||||
|
|
||||||
|
} |
@ -0,0 +1,38 @@ |
|||||||
|
package mightypork.gamecore.gui; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.HashSet; |
||||||
|
import java.util.Set; |
||||||
|
|
||||||
|
import mightypork.utils.interfaces.Enableable; |
||||||
|
|
||||||
|
|
||||||
|
public class ActionGroup implements Enableable { |
||||||
|
|
||||||
|
private boolean enabled = true; |
||||||
|
|
||||||
|
private final Set<Enableable> groupMembers = new HashSet<>(); |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setEnabled(boolean yes) |
||||||
|
{ |
||||||
|
enabled = yes; |
||||||
|
for (final Enableable e : groupMembers) |
||||||
|
e.setEnabled(yes); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean isEnabled() |
||||||
|
{ |
||||||
|
return enabled; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void add(Enableable action) |
||||||
|
{ |
||||||
|
groupMembers.add(action); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,17 @@ |
|||||||
|
package mightypork.gamecore.gui; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Element that can be assigned an action (ie. button); |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public interface ActionTrigger { |
||||||
|
|
||||||
|
/** |
||||||
|
* Assign an action |
||||||
|
* |
||||||
|
* @param action action |
||||||
|
*/ |
||||||
|
void setAction(Action action); |
||||||
|
} |
@ -0,0 +1,169 @@ |
|||||||
|
package mightypork.gamecore.gui.components; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.graphics.Renderable; |
||||||
|
import mightypork.gamecore.gui.events.LayoutChangeEvent; |
||||||
|
import mightypork.gamecore.gui.events.LayoutChangeListener; |
||||||
|
import mightypork.utils.Support; |
||||||
|
import mightypork.utils.annotations.Stub; |
||||||
|
import mightypork.utils.interfaces.Enableable; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
import mightypork.utils.math.color.Color; |
||||||
|
import mightypork.utils.math.constraints.num.Num; |
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
import mightypork.utils.math.constraints.rect.RectBound; |
||||||
|
import mightypork.utils.math.constraints.rect.caching.AbstractRectCache; |
||||||
|
import mightypork.utils.math.constraints.rect.proxy.RectProxy; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* {@link Renderable} with pluggable context. When caching is enabled, the |
||||||
|
* layout update can be triggered by firing the {@link LayoutChangeEvent}. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class BaseComponent extends AbstractRectCache implements Component, LayoutChangeListener, Enableable { |
||||||
|
|
||||||
|
private Rect source; |
||||||
|
private boolean visible = true; |
||||||
|
private boolean enabled = true; |
||||||
|
private int indirectDisableLevel = 0; |
||||||
|
|
||||||
|
private Num alphaMul = Num.ONE; |
||||||
|
|
||||||
|
|
||||||
|
public BaseComponent() { |
||||||
|
enableCaching(false); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setRect(RectBound rect) |
||||||
|
{ |
||||||
|
this.source = new RectProxy(rect); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final boolean isVisible() |
||||||
|
{ |
||||||
|
return visible; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final void setVisible(boolean visible) |
||||||
|
{ |
||||||
|
this.visible = visible; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final Rect getCacheSource() |
||||||
|
{ |
||||||
|
return source.round(); // round to avoid visual artifacts in fonts and such
|
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final void render() |
||||||
|
{ |
||||||
|
if (!isVisible()) return; |
||||||
|
|
||||||
|
Color.pushAlpha(alphaMul); |
||||||
|
renderComponent(); |
||||||
|
Color.popAlpha(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final void onLayoutChanged() |
||||||
|
{ |
||||||
|
try { |
||||||
|
poll(); |
||||||
|
} catch (final NullPointerException e) { |
||||||
|
Log.e("Component is missing a bounding rect, at: " + Support.str(getClass())); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final void onConstraintChanged() |
||||||
|
{ |
||||||
|
updateLayout(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final boolean isMouseOver() |
||||||
|
{ |
||||||
|
return App.input().getMousePos().isInside(this); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Draw the component (it's visible) |
||||||
|
*/ |
||||||
|
protected abstract void renderComponent(); |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
@Stub |
||||||
|
public void updateLayout() |
||||||
|
{ |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setEnabled(boolean yes) |
||||||
|
{ |
||||||
|
enabled = yes; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean isEnabled() |
||||||
|
{ |
||||||
|
return enabled && isIndirectlyEnabled(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final void setAlpha(Num alpha) |
||||||
|
{ |
||||||
|
this.alphaMul = alpha; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final void setAlpha(double alpha) |
||||||
|
{ |
||||||
|
this.alphaMul = Num.make(alpha); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setIndirectlyEnabled(boolean yes) |
||||||
|
{ |
||||||
|
if (!yes) { |
||||||
|
indirectDisableLevel++; |
||||||
|
} else { |
||||||
|
if (indirectDisableLevel > 0) indirectDisableLevel--; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean isIndirectlyEnabled() |
||||||
|
{ |
||||||
|
return indirectDisableLevel == 0; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean isDirectlyEnabled() |
||||||
|
{ |
||||||
|
return enabled; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,95 @@ |
|||||||
|
package mightypork.gamecore.gui.components; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.interfaces.Enableable; |
||||||
|
import mightypork.utils.interfaces.Hideable; |
||||||
|
import mightypork.utils.math.constraints.num.Num; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Basic UI component interface
|
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public interface Component extends Enableable, Hideable, PluggableRenderable { |
||||||
|
|
||||||
|
/** |
||||||
|
* Render the component, if it is visible. |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
void render(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* The bounding rect was changed. The component should now update any cached |
||||||
|
* constraints derived from it. |
||||||
|
*/ |
||||||
|
void updateLayout(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return true if mouse is currently over the component |
||||||
|
*/ |
||||||
|
boolean isMouseOver(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set alpha multiplier for this and nested components |
||||||
|
* |
||||||
|
* @param alpha alpha multiplier (dynamic value) |
||||||
|
*/ |
||||||
|
void setAlpha(Num alpha); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set alpha multiplier for this and nested components |
||||||
|
* |
||||||
|
* @param alpha alpha multiplier (constant value) |
||||||
|
*/ |
||||||
|
void setAlpha(double alpha); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Indirectly enable / disable, used for nested hierarchies.<br> |
||||||
|
* When component is twice indirectly disabled, it needs to be twice |
||||||
|
* indirectly enabled to be enabled again. |
||||||
|
* |
||||||
|
* @param yes |
||||||
|
*/ |
||||||
|
void setIndirectlyEnabled(boolean yes); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Check if the compionent is not indirectly disabled. May still be directly |
||||||
|
* disabled. |
||||||
|
* |
||||||
|
* @return indirectly enabled |
||||||
|
*/ |
||||||
|
boolean isIndirectlyEnabled(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Check if the component is directly enabled (set by setEnabled()). May |
||||||
|
* still be indirectly disabled. |
||||||
|
* |
||||||
|
* @return directly enabled |
||||||
|
*/ |
||||||
|
boolean isDirectlyEnabled(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Set directly enabled (must be both directly and indirectly enabled to be |
||||||
|
* enabled completely) |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public void setEnabled(boolean yes); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Check if the component is both directly and indirectly enabled |
||||||
|
* |
||||||
|
* @return enabled |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public boolean isEnabled(); |
||||||
|
} |
@ -0,0 +1,7 @@ |
|||||||
|
package mightypork.gamecore.gui.components; |
||||||
|
|
||||||
|
|
||||||
|
public interface DynamicWidthComponent extends Component { |
||||||
|
|
||||||
|
double computeWidth(double height); |
||||||
|
} |
@ -0,0 +1,15 @@ |
|||||||
|
package mightypork.gamecore.gui.components; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.eventbus.clients.ToggleableClient; |
||||||
|
import mightypork.utils.interfaces.Enableable; |
||||||
|
|
||||||
|
|
||||||
|
public abstract class InputComponent extends BaseComponent implements Enableable, ToggleableClient { |
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean isListening() |
||||||
|
{ |
||||||
|
return isEnabled(); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,122 @@ |
|||||||
|
package mightypork.gamecore.gui.components; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.Collection; |
||||||
|
import java.util.LinkedList; |
||||||
|
|
||||||
|
import mightypork.utils.eventbus.clients.ClientHub; |
||||||
|
import mightypork.utils.eventbus.clients.DelegatingList; |
||||||
|
import mightypork.utils.math.constraints.rect.RectBound; |
||||||
|
|
||||||
|
|
||||||
|
public abstract class LayoutComponent extends BaseComponent implements ClientHub { |
||||||
|
|
||||||
|
private final DelegatingList clientList; |
||||||
|
final LinkedList<Component> components = new LinkedList<>(); |
||||||
|
|
||||||
|
|
||||||
|
public LayoutComponent(RectBound context) { |
||||||
|
this.clientList = new DelegatingList(); |
||||||
|
setRect(context); |
||||||
|
enableCaching(true); // layout is typically updated only when screen resizes.
|
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public LayoutComponent() { |
||||||
|
this(null); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public Collection<Object> getChildClients() |
||||||
|
{ |
||||||
|
return clientList; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean doesDelegate() |
||||||
|
{ |
||||||
|
return clientList.doesDelegate(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean isListening() |
||||||
|
{ |
||||||
|
return clientList.isListening(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void addChildClient(Object client) |
||||||
|
{ |
||||||
|
clientList.add(client); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void removeChildClient(Object client) |
||||||
|
{ |
||||||
|
clientList.remove(client); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setEnabled(boolean yes) |
||||||
|
{ |
||||||
|
if (isDirectlyEnabled() != yes) { |
||||||
|
super.setEnabled(yes); |
||||||
|
|
||||||
|
for (final Component c : components) { |
||||||
|
c.setIndirectlyEnabled(yes); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Connect to bus and add to element list |
||||||
|
* |
||||||
|
* @param component added component, whose context has already been set. |
||||||
|
*/ |
||||||
|
protected final void attach(Component component) |
||||||
|
{ |
||||||
|
if (component == null) return; |
||||||
|
if (component == this) { |
||||||
|
throw new IllegalArgumentException("Uruboros. (infinite recursion evaded)"); |
||||||
|
} |
||||||
|
|
||||||
|
components.add(component); |
||||||
|
addChildClient(component); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void renderComponent() |
||||||
|
{ |
||||||
|
for (final Component cmp : components) { |
||||||
|
cmp.render(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void updateLayout() |
||||||
|
{ |
||||||
|
for (final Component cmp : components) { |
||||||
|
cmp.updateLayout(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setIndirectlyEnabled(boolean yes) |
||||||
|
{ |
||||||
|
super.setIndirectlyEnabled(yes); |
||||||
|
|
||||||
|
for (final Component cmp : components) { |
||||||
|
cmp.setIndirectlyEnabled(yes); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,77 @@ |
|||||||
|
package mightypork.gamecore.gui.components; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.math.constraints.num.Num; |
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
import mightypork.utils.math.constraints.rect.RectBound; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
import mightypork.utils.math.constraints.vect.proxy.VectAdapter; |
||||||
|
|
||||||
|
|
||||||
|
public abstract class LinearComponent extends BaseComponent implements DynamicWidthComponent { |
||||||
|
|
||||||
|
private final Rect rect = new Rect() { |
||||||
|
|
||||||
|
@Override |
||||||
|
public Vect size() |
||||||
|
{ |
||||||
|
return new Vect() { |
||||||
|
|
||||||
|
@Override |
||||||
|
public double x() |
||||||
|
{ |
||||||
|
return computeWidth(y()); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public double y() |
||||||
|
{ |
||||||
|
return height.value(); |
||||||
|
} |
||||||
|
|
||||||
|
}; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public Vect origin() |
||||||
|
{ |
||||||
|
return new VectAdapter() { |
||||||
|
|
||||||
|
@Override |
||||||
|
protected Vect getSource() |
||||||
|
{ |
||||||
|
return origin; |
||||||
|
} |
||||||
|
}; |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
private Vect origin; |
||||||
|
private Num height; |
||||||
|
|
||||||
|
|
||||||
|
public LinearComponent() { |
||||||
|
super.setRect(rect); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setRect(RectBound rect) |
||||||
|
{ |
||||||
|
throw new RuntimeException("Cannot assign a rect to a linear component. Set origin and height instead."); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setHeight(Num height) |
||||||
|
{ |
||||||
|
this.height = height; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setOrigin(Vect origin) |
||||||
|
{ |
||||||
|
this.origin = origin; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,28 @@ |
|||||||
|
package mightypork.gamecore.gui.components; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.graphics.Renderable; |
||||||
|
import mightypork.utils.math.constraints.rect.PluggableRectBound; |
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
import mightypork.utils.math.constraints.rect.RectBound; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Renderable that can be assigned different context |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public interface PluggableRenderable extends Renderable, PluggableRectBound { |
||||||
|
|
||||||
|
@Override |
||||||
|
void render(); |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
Rect getRect(); |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
void setRect(RectBound rect); |
||||||
|
|
||||||
|
} |
@ -0,0 +1,49 @@ |
|||||||
|
package mightypork.gamecore.gui.components.input; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.Action; |
||||||
|
import mightypork.gamecore.gui.ActionTrigger; |
||||||
|
import mightypork.gamecore.gui.components.InputComponent; |
||||||
|
import mightypork.gamecore.input.events.MouseButtonEvent; |
||||||
|
import mightypork.gamecore.input.events.MouseButtonHandler; |
||||||
|
|
||||||
|
|
||||||
|
public abstract class ClickableComponent extends InputComponent implements ActionTrigger, MouseButtonHandler { |
||||||
|
|
||||||
|
protected boolean btnDownOver; |
||||||
|
private Action action; |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setAction(Action action) |
||||||
|
{ |
||||||
|
this.action = action; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
protected void triggerAction() |
||||||
|
{ |
||||||
|
if (action != null && isEnabled()) action.run(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void receive(MouseButtonEvent event) |
||||||
|
{ |
||||||
|
if (!event.isButtonEvent()) return; |
||||||
|
|
||||||
|
if (event.isDown()) { |
||||||
|
btnDownOver = event.isOver(this); |
||||||
|
} |
||||||
|
|
||||||
|
if (event.isUp()) { |
||||||
|
|
||||||
|
if (btnDownOver && event.isOver(this)) { |
||||||
|
triggerAction(); |
||||||
|
event.consume(); |
||||||
|
} |
||||||
|
|
||||||
|
btnDownOver = false; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,63 @@ |
|||||||
|
package mightypork.gamecore.gui.components.input; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.Collection; |
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.Component; |
||||||
|
import mightypork.utils.eventbus.clients.ClientList; |
||||||
|
import mightypork.utils.eventbus.clients.DelegatingClient; |
||||||
|
|
||||||
|
|
||||||
|
public class ClickableWrapper extends ClickableComponent implements DelegatingClient { |
||||||
|
|
||||||
|
private final Component wrapped; |
||||||
|
private final ClientList list; |
||||||
|
|
||||||
|
|
||||||
|
public ClickableWrapper(Component wrapped) { |
||||||
|
this.wrapped = wrapped; |
||||||
|
wrapped.setRect(this); |
||||||
|
|
||||||
|
list = new ClientList(wrapped); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public Collection<?> getChildClients() |
||||||
|
{ |
||||||
|
return list; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean doesDelegate() |
||||||
|
{ |
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
protected void renderComponent() |
||||||
|
{ |
||||||
|
wrapped.render(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setEnabled(boolean yes) |
||||||
|
{ |
||||||
|
if (yes != super.isDirectlyEnabled()) { |
||||||
|
super.setEnabled(yes); |
||||||
|
wrapped.setIndirectlyEnabled(yes); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setIndirectlyEnabled(boolean yes) |
||||||
|
{ |
||||||
|
super.setIndirectlyEnabled(yes); |
||||||
|
wrapped.setIndirectlyEnabled(yes); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,77 @@ |
|||||||
|
package mightypork.gamecore.gui.components.input; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.graphics.fonts.IFont; |
||||||
|
import mightypork.gamecore.gui.components.DynamicWidthComponent; |
||||||
|
import mightypork.gamecore.gui.components.painters.TextPainter; |
||||||
|
import mightypork.utils.math.AlignX; |
||||||
|
import mightypork.utils.math.color.Color; |
||||||
|
import mightypork.utils.math.color.pal.RGB; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
import mightypork.utils.math.constraints.vect.var.VectVar; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Menu-like button with shadow and push state |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class TextButton extends ClickableComponent implements DynamicWidthComponent { |
||||||
|
|
||||||
|
public final TextPainter textPainter; |
||||||
|
|
||||||
|
private final VectVar offset = Vect.makeVar(); |
||||||
|
|
||||||
|
public Vect offsetPassive = height().div(16).toVectXY(); |
||||||
|
public Vect offsetOver = height().div(20).toVectXY(); |
||||||
|
public Vect offsetUnder = height().div(32).toVectXY(); |
||||||
|
|
||||||
|
private final Color color; |
||||||
|
|
||||||
|
private boolean hoverMove = true; |
||||||
|
|
||||||
|
|
||||||
|
public TextButton(IFont font, String text, Color color) { |
||||||
|
this.color = color; |
||||||
|
|
||||||
|
this.textPainter = new TextPainter(font, AlignX.CENTER, this.color, text); |
||||||
|
this.textPainter.setRect(this); |
||||||
|
this.textPainter.setShadow(RGB.BLACK_30, offset); |
||||||
|
textPainter.setVPaddingPercent(5); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
protected void renderComponent() |
||||||
|
{ |
||||||
|
if (isMouseOver()) { |
||||||
|
if (App.input().isMouseButtonDown(0)) { |
||||||
|
offset.setTo(offsetUnder); |
||||||
|
} else { |
||||||
|
offset.setTo(hoverMove ? offsetOver : offsetPassive); |
||||||
|
} |
||||||
|
} else { |
||||||
|
offset.setTo(offsetPassive); |
||||||
|
} |
||||||
|
|
||||||
|
textPainter.render(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Disable offset change on hover |
||||||
|
*/ |
||||||
|
public void disableHoverEffect() |
||||||
|
{ |
||||||
|
hoverMove = false; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public double computeWidth(double height) |
||||||
|
{ |
||||||
|
return textPainter.computeWidth(height); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,42 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.Component; |
||||||
|
import mightypork.utils.math.constraints.rect.RectBound; |
||||||
|
|
||||||
|
|
||||||
|
public class ColumnLayout extends GridLayout { |
||||||
|
|
||||||
|
private int col = 0; |
||||||
|
|
||||||
|
|
||||||
|
public ColumnLayout(int rows) { |
||||||
|
this(null, rows); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public ColumnLayout(RectBound context, int cols) { |
||||||
|
super(context, 1, cols); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void add(final Component elem) |
||||||
|
{ |
||||||
|
add(elem, 1); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void add(final Component elem, int colSpan) |
||||||
|
{ |
||||||
|
if (elem == null) return; |
||||||
|
|
||||||
|
put(elem, 0, col, 1, colSpan); |
||||||
|
col += colSpan; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void skip(int cols) |
||||||
|
{ |
||||||
|
col += cols; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,36 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.Component; |
||||||
|
import mightypork.gamecore.gui.components.LayoutComponent; |
||||||
|
import mightypork.utils.math.constraints.rect.RectBound; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Layout for components with arbitrary constraints. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class ConstraintLayout extends LayoutComponent { |
||||||
|
|
||||||
|
public ConstraintLayout() { |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public ConstraintLayout(RectBound context) { |
||||||
|
super(context); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a component to the layout.<br> |
||||||
|
* The component's rect must be set up manually. |
||||||
|
* |
||||||
|
* @param component |
||||||
|
*/ |
||||||
|
public void add(Component component) |
||||||
|
{ |
||||||
|
attach(component); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,85 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.Component; |
||||||
|
import mightypork.gamecore.gui.components.LayoutComponent; |
||||||
|
import mightypork.utils.math.AlignX; |
||||||
|
import mightypork.utils.math.constraints.num.Num; |
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
import mightypork.utils.math.constraints.rect.RectBound; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Holder with same-sized columns, aligned to left or right |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class FlowColumnLayout extends LayoutComponent { |
||||||
|
|
||||||
|
private int col = 0; |
||||||
|
private Num elementWidth; |
||||||
|
private final AlignX align; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param context context |
||||||
|
* @param elementWidth width of all elements |
||||||
|
* @param align component align. Legal values are LEFT and RIGHT. |
||||||
|
*/ |
||||||
|
public FlowColumnLayout(RectBound context, Num elementWidth, AlignX align) { |
||||||
|
super(context); |
||||||
|
this.elementWidth = elementWidth; |
||||||
|
this.align = align; |
||||||
|
|
||||||
|
if (align != AlignX.LEFT && align != AlignX.RIGHT) { |
||||||
|
throw new IllegalArgumentException("Can align only left or right."); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* make a new holder.<br> |
||||||
|
* Context must be assigned before rendering. |
||||||
|
* |
||||||
|
* @param elementWidth width of all elements |
||||||
|
* @param align component align. Legal values are LEFT and RIGHT. |
||||||
|
*/ |
||||||
|
public FlowColumnLayout(Num elementWidth, AlignX align) { |
||||||
|
this(null, elementWidth, align); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add an item |
||||||
|
* |
||||||
|
* @param elem |
||||||
|
*/ |
||||||
|
public void add(final Component elem) |
||||||
|
{ |
||||||
|
if (elem == null) return; |
||||||
|
|
||||||
|
final Rect r; |
||||||
|
|
||||||
|
switch (align) { |
||||||
|
case LEFT: |
||||||
|
r = leftEdge().growRight(elementWidth).moveX(elementWidth.mul(col++)); |
||||||
|
break; |
||||||
|
case RIGHT: |
||||||
|
r = rightEdge().growLeft(elementWidth).moveX(elementWidth.mul(-(col++))); |
||||||
|
break; |
||||||
|
default: |
||||||
|
throw new IllegalArgumentException("Bad align."); |
||||||
|
} |
||||||
|
|
||||||
|
elem.setRect(r); |
||||||
|
|
||||||
|
attach(elem); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setElementWidth(Num elementWidth) |
||||||
|
{ |
||||||
|
this.elementWidth = elementWidth; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,84 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.Component; |
||||||
|
import mightypork.gamecore.gui.components.LayoutComponent; |
||||||
|
import mightypork.utils.math.AlignY; |
||||||
|
import mightypork.utils.math.constraints.num.Num; |
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
import mightypork.utils.math.constraints.rect.RectBound; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Holder with same-sized rows, aligned to top or bottom |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class FlowRowLayout extends LayoutComponent { |
||||||
|
|
||||||
|
private int row = 0; |
||||||
|
private Num elementHeight; |
||||||
|
private final AlignY align; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param context context |
||||||
|
* @param elementHeight height of all elements |
||||||
|
* @param align component align. Legal values are TOP and BOTTOM. |
||||||
|
*/ |
||||||
|
public FlowRowLayout(RectBound context, Num elementHeight, AlignY align) { |
||||||
|
super(context); |
||||||
|
this.elementHeight = elementHeight; |
||||||
|
this.align = align; |
||||||
|
|
||||||
|
if (align != AlignY.TOP && align != AlignY.BOTTOM) { |
||||||
|
throw new IllegalArgumentException("Can align only to top or bottom."); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* make a new holder.<br> |
||||||
|
* Context must be assigned before rendering. |
||||||
|
* |
||||||
|
* @param elementHeight height of all elements |
||||||
|
* @param align component align. Legal values are TOP and BOTTOM. |
||||||
|
*/ |
||||||
|
public FlowRowLayout(Num elementHeight, AlignY align) { |
||||||
|
this(null, elementHeight, align); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add an item |
||||||
|
* |
||||||
|
* @param elem |
||||||
|
*/ |
||||||
|
public void add(final Component elem) |
||||||
|
{ |
||||||
|
if (elem == null) return; |
||||||
|
|
||||||
|
final Rect r; |
||||||
|
|
||||||
|
switch (align) { |
||||||
|
case TOP: |
||||||
|
r = topEdge().growDown(elementHeight).moveY(elementHeight.mul(row++)); |
||||||
|
break; |
||||||
|
case BOTTOM: |
||||||
|
r = bottomEdge().growUp(elementHeight).moveY(elementHeight.mul(-(row++))); |
||||||
|
break; |
||||||
|
default: |
||||||
|
throw new IllegalArgumentException("Bad align."); |
||||||
|
} |
||||||
|
|
||||||
|
elem.setRect(r); |
||||||
|
|
||||||
|
attach(elem); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setElementHeight(Num elementHeight) |
||||||
|
{ |
||||||
|
this.elementHeight = elementHeight; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,78 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.Component; |
||||||
|
import mightypork.gamecore.gui.components.LayoutComponent; |
||||||
|
import mightypork.utils.math.constraints.rect.RectBound; |
||||||
|
import mightypork.utils.math.constraints.rect.builders.TiledRect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Holder with table cells |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class GridLayout extends LayoutComponent { |
||||||
|
|
||||||
|
private final TiledRect tiler; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param context context |
||||||
|
* @param rows number of rows |
||||||
|
* @param cols number of columns |
||||||
|
*/ |
||||||
|
public GridLayout(RectBound context, int rows, int cols) { |
||||||
|
super(context); |
||||||
|
this.tiler = tiles(cols, rows); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* make a new holder.<br> |
||||||
|
* Context must be assigned before rendering. |
||||||
|
* |
||||||
|
* @param rows number of rows |
||||||
|
* @param cols number of columns |
||||||
|
*/ |
||||||
|
public GridLayout(int rows, int cols) { |
||||||
|
this(null, rows, cols); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a row to the holder. |
||||||
|
* |
||||||
|
* @param row row (one-based) |
||||||
|
* @param column column (one-based) |
||||||
|
* @param elem added component |
||||||
|
*/ |
||||||
|
public void put(Component elem, int row, int column) |
||||||
|
{ |
||||||
|
if (elem == null) return; |
||||||
|
|
||||||
|
elem.setRect(tiler.tile(column, row)); |
||||||
|
|
||||||
|
attach(elem); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Put with span |
||||||
|
* |
||||||
|
* @param elem |
||||||
|
* @param row |
||||||
|
* @param column |
||||||
|
* @param rowspan |
||||||
|
* @param colspan |
||||||
|
*/ |
||||||
|
public void put(Component elem, int row, int column, int rowspan, int colspan) |
||||||
|
{ |
||||||
|
if (elem == null) return; |
||||||
|
|
||||||
|
elem.setRect(tiler.span(column, row, colspan, rowspan)); |
||||||
|
|
||||||
|
attach(elem); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,18 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.BaseComponent; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Invisible component that does nothing at all; Null object pattern |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class NullComponent extends BaseComponent { |
||||||
|
|
||||||
|
@Override |
||||||
|
protected void renderComponent() |
||||||
|
{ |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,42 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.Component; |
||||||
|
import mightypork.utils.math.constraints.rect.RectBound; |
||||||
|
|
||||||
|
|
||||||
|
public class RowLayout extends GridLayout { |
||||||
|
|
||||||
|
private int row = 0; |
||||||
|
|
||||||
|
|
||||||
|
public RowLayout(int rows) { |
||||||
|
this(null, rows); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public RowLayout(RectBound context, int rows) { |
||||||
|
super(context, rows, 1); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void add(final Component elem) |
||||||
|
{ |
||||||
|
add(elem, 1); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void add(final Component elem, int rowSpan) |
||||||
|
{ |
||||||
|
if (elem == null) return; |
||||||
|
|
||||||
|
put(elem, row, 0, rowSpan, 1); |
||||||
|
row += rowSpan; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void skip(int rows) |
||||||
|
{ |
||||||
|
row += rows; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,78 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout.linear; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.Collection; |
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.Component; |
||||||
|
import mightypork.gamecore.gui.components.LinearComponent; |
||||||
|
import mightypork.utils.eventbus.clients.ClientList; |
||||||
|
import mightypork.utils.eventbus.clients.DelegatingClient; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Converts a component into a linear component |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class AbstractLinearWrapper extends LinearComponent implements DelegatingClient { |
||||||
|
|
||||||
|
protected final Component wrapped; |
||||||
|
private final ClientList list; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param wrapped wrapped component. Can be null. |
||||||
|
*/ |
||||||
|
public AbstractLinearWrapper(Component wrapped) { |
||||||
|
this.wrapped = wrapped; |
||||||
|
if (wrapped != null) { |
||||||
|
if (wrapped instanceof LinearComponent) { |
||||||
|
((LinearComponent) wrapped).setHeight(height()); |
||||||
|
((LinearComponent) wrapped).setOrigin(origin()); |
||||||
|
} else { |
||||||
|
wrapped.setRect(this); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
list = new ClientList(wrapped); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
protected void renderComponent() |
||||||
|
{ |
||||||
|
if (wrapped != null) wrapped.render(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public Collection<?> getChildClients() |
||||||
|
{ |
||||||
|
return list; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean doesDelegate() |
||||||
|
{ |
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setEnabled(boolean yes) |
||||||
|
{ |
||||||
|
if (yes != super.isDirectlyEnabled()) { |
||||||
|
super.setEnabled(yes); |
||||||
|
wrapped.setIndirectlyEnabled(yes); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setIndirectlyEnabled(boolean yes) |
||||||
|
{ |
||||||
|
super.setIndirectlyEnabled(yes); |
||||||
|
wrapped.setIndirectlyEnabled(yes); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,24 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout.linear; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.layout.NullComponent; |
||||||
|
import mightypork.utils.math.constraints.num.Num; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Gap in linear layout |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class LinearGap extends LinearRectangle { |
||||||
|
|
||||||
|
public LinearGap(Num width) { |
||||||
|
super(new NullComponent(), width); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public LinearGap(double heightPercent) { |
||||||
|
this(Num.ZERO); |
||||||
|
setWidth(height().perc(heightPercent)); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,93 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout.linear; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.DynamicWidthComponent; |
||||||
|
import mightypork.gamecore.gui.components.LayoutComponent; |
||||||
|
import mightypork.gamecore.gui.components.LinearComponent; |
||||||
|
import mightypork.utils.math.AlignX; |
||||||
|
import mightypork.utils.math.constraints.num.Num; |
||||||
|
import mightypork.utils.math.constraints.num.batch.NumSum; |
||||||
|
import mightypork.utils.math.constraints.rect.RectBound; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
import mightypork.utils.math.constraints.vect.proxy.VectAdapter; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Layout that aligns elements while taking into account their actual |
||||||
|
* dimensions.<br> |
||||||
|
* Useful eg. for buttons that stretch based on text length. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class LinearLayout extends LayoutComponent { |
||||||
|
|
||||||
|
public LinearLayout(AlignX align) { |
||||||
|
this.align = align; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public LinearLayout(RectBound context, AlignX align) { |
||||||
|
super(context); |
||||||
|
this.align = align; |
||||||
|
} |
||||||
|
|
||||||
|
private final NumSum totalWidth = new NumSum(); |
||||||
|
|
||||||
|
private final Vect leftAlignOrigin = LinearLayout.this.origin(); |
||||||
|
private final Vect centerAlignOrigin = LinearLayout.this.topCenter().sub(totalWidth.half(), Num.ZERO); |
||||||
|
private final Vect rightAlignOrigin = LinearLayout.this.topRight().sub(totalWidth, Num.ZERO); |
||||||
|
|
||||||
|
private final Vect leftMostOrigin = new VectAdapter() { |
||||||
|
|
||||||
|
@Override |
||||||
|
protected Vect getSource() |
||||||
|
{ |
||||||
|
switch (align) { |
||||||
|
default: |
||||||
|
case LEFT: |
||||||
|
return leftAlignOrigin; |
||||||
|
case CENTER: |
||||||
|
return centerAlignOrigin; |
||||||
|
case RIGHT: |
||||||
|
return rightAlignOrigin; |
||||||
|
} |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
private Vect nextOrigin = leftMostOrigin; |
||||||
|
|
||||||
|
private AlignX align = AlignX.LEFT; |
||||||
|
|
||||||
|
|
||||||
|
public void add(DynamicWidthComponent dwcomp) |
||||||
|
{ |
||||||
|
add(new LinearWrapper(dwcomp)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void add(LinearComponent lincomp) |
||||||
|
{ |
||||||
|
lincomp.setHeight(height()); |
||||||
|
lincomp.setOrigin(nextOrigin); |
||||||
|
nextOrigin = nextOrigin.add(lincomp.width(), Num.ZERO); |
||||||
|
totalWidth.addSummand(lincomp.width()); |
||||||
|
attach(lincomp); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setAlign(AlignX align) |
||||||
|
{ |
||||||
|
this.align = align; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a gap. |
||||||
|
* |
||||||
|
* @param heightPercent percent of height for gap width |
||||||
|
*/ |
||||||
|
public void gap(double heightPercent) |
||||||
|
{ |
||||||
|
add(new LinearGap(heightPercent)); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,31 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout.linear; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.Component; |
||||||
|
import mightypork.utils.math.constraints.num.Num; |
||||||
|
|
||||||
|
|
||||||
|
public class LinearRectangle extends AbstractLinearWrapper { |
||||||
|
|
||||||
|
private Num width; |
||||||
|
|
||||||
|
|
||||||
|
public LinearRectangle(Component wrapped, Num width) { |
||||||
|
super(wrapped); |
||||||
|
this.width = width; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setWidth(Num width) |
||||||
|
{ |
||||||
|
this.width = width; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public double computeWidth(double height) |
||||||
|
{ |
||||||
|
return this.width.value(); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,20 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout.linear; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.Component; |
||||||
|
|
||||||
|
|
||||||
|
public class LinearSquare extends AbstractLinearWrapper { |
||||||
|
|
||||||
|
public LinearSquare(Component wrapped) { |
||||||
|
super(wrapped); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public double computeWidth(double height) |
||||||
|
{ |
||||||
|
return height; |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,20 @@ |
|||||||
|
package mightypork.gamecore.gui.components.layout.linear; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.gui.components.DynamicWidthComponent; |
||||||
|
|
||||||
|
|
||||||
|
public class LinearWrapper extends AbstractLinearWrapper { |
||||||
|
|
||||||
|
public LinearWrapper(DynamicWidthComponent wrapped) { |
||||||
|
super(wrapped); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public double computeWidth(double height) |
||||||
|
{ |
||||||
|
return ((DynamicWidthComponent) wrapped).computeWidth(height); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,46 @@ |
|||||||
|
package mightypork.gamecore.gui.components.painters; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.graphics.textures.TxQuad; |
||||||
|
import mightypork.gamecore.gui.components.BaseComponent; |
||||||
|
import mightypork.gamecore.gui.components.DynamicWidthComponent; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Draws image in given rect |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class ImagePainter extends BaseComponent implements DynamicWidthComponent { |
||||||
|
|
||||||
|
private TxQuad txQuad; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param txQuad drawn image |
||||||
|
*/ |
||||||
|
public ImagePainter(TxQuad txQuad) { |
||||||
|
this.txQuad = txQuad; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void renderComponent() |
||||||
|
{ |
||||||
|
App.gfx().quad(this, txQuad); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public double computeWidth(double height) |
||||||
|
{ |
||||||
|
return (height / txQuad.uvs.height().value()) * txQuad.uvs.width().value(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setTxQuad(TxQuad txQuad) |
||||||
|
{ |
||||||
|
this.txQuad = txQuad; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,62 @@ |
|||||||
|
package mightypork.gamecore.gui.components.painters; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.gui.components.BaseComponent; |
||||||
|
import mightypork.utils.annotations.FactoryMethod; |
||||||
|
import mightypork.utils.math.color.Color; |
||||||
|
import mightypork.utils.math.color.Grad; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Draws image in given rect |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class QuadPainter extends BaseComponent { |
||||||
|
|
||||||
|
@FactoryMethod |
||||||
|
public static QuadPainter gradH(Color colorLeft, Color colorRight) |
||||||
|
{ |
||||||
|
return new QuadPainter(colorLeft, colorRight, colorRight, colorLeft); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@FactoryMethod |
||||||
|
public static QuadPainter gradV(Color colorTop, Color colorBottom) |
||||||
|
{ |
||||||
|
return new QuadPainter(colorTop, colorTop, colorBottom, colorBottom); |
||||||
|
} |
||||||
|
|
||||||
|
private final Grad grad; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Painter with solid color |
||||||
|
* |
||||||
|
* @param color |
||||||
|
*/ |
||||||
|
public QuadPainter(Color color) { |
||||||
|
this.grad = new Grad(color, color, color, color); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Painter with coloured vertices. |
||||||
|
* |
||||||
|
* @param leftTop |
||||||
|
* @param rightTop |
||||||
|
* @param leftBottom |
||||||
|
* @param rightBottom |
||||||
|
*/ |
||||||
|
public QuadPainter(Color leftTop, Color rightTop, Color leftBottom, Color rightBottom) { |
||||||
|
this.grad = new Grad(leftTop, rightTop, rightBottom, leftBottom); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void renderComponent() |
||||||
|
{ |
||||||
|
App.gfx().quad(getRect(), grad); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,163 @@ |
|||||||
|
package mightypork.gamecore.gui.components.painters; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.graphics.fonts.FontRenderer; |
||||||
|
import mightypork.gamecore.graphics.fonts.IFont; |
||||||
|
import mightypork.gamecore.gui.components.BaseComponent; |
||||||
|
import mightypork.gamecore.gui.components.DynamicWidthComponent; |
||||||
|
import mightypork.utils.math.AlignX; |
||||||
|
import mightypork.utils.math.color.Color; |
||||||
|
import mightypork.utils.math.color.pal.RGB; |
||||||
|
import mightypork.utils.math.constraints.num.Num; |
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
import mightypork.utils.string.StringProvider; |
||||||
|
import mightypork.utils.string.StringWrapper; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Text painting component. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class TextPainter extends BaseComponent implements DynamicWidthComponent { |
||||||
|
|
||||||
|
private static final boolean DEBUG_FONT_RENDER = false; |
||||||
|
private final FontRenderer font; |
||||||
|
private Color color; |
||||||
|
private AlignX align; |
||||||
|
private StringProvider text; |
||||||
|
private boolean shadow; |
||||||
|
|
||||||
|
private double yPaddingPerc = 0; |
||||||
|
|
||||||
|
private Color shadowColor = RGB.BLACK; |
||||||
|
private Vect shadowOffset = Vect.make(2, 2); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param font font to use |
||||||
|
*/ |
||||||
|
public TextPainter(IFont font) { |
||||||
|
this(font, AlignX.LEFT, RGB.WHITE); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public TextPainter(IFont font, Color color, String text) { |
||||||
|
this(font, AlignX.LEFT, color, new StringWrapper(text)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public TextPainter(IFont font, Color color, StringProvider text) { |
||||||
|
this(font, AlignX.LEFT, color, text); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public TextPainter(IFont font, Color color) { |
||||||
|
this(font, AlignX.LEFT, color, (StringProvider) null); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public TextPainter(IFont font, AlignX align, Color color, String text) { |
||||||
|
this(font, align, color, new StringWrapper(text)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public TextPainter(IFont font, AlignX align, Color color, StringProvider text) { |
||||||
|
this.font = new FontRenderer(font); |
||||||
|
this.color = color; |
||||||
|
this.align = align; |
||||||
|
this.text = text; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public TextPainter(IFont font, AlignX align, Color color) { |
||||||
|
this(font, align, color, (StringProvider) null); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void renderComponent() |
||||||
|
{ |
||||||
|
if (text == null) return; |
||||||
|
|
||||||
|
final String str = text.getString(); |
||||||
|
|
||||||
|
final Num shrY = height().perc(yPaddingPerc); |
||||||
|
|
||||||
|
final Rect rect = getRect().shrink(Num.ZERO, shrY); |
||||||
|
|
||||||
|
if (shadow) { |
||||||
|
font.draw(str, rect.round(), align, shadowColor); |
||||||
|
} |
||||||
|
|
||||||
|
final Rect r = (shadow ? rect.move(shadowOffset.neg()) : rect).round(); |
||||||
|
font.draw(str, r, align, color); |
||||||
|
|
||||||
|
if (DEBUG_FONT_RENDER) App.gfx().quad(r, RGB.PINK.withAlpha(0.4)); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setShadow(Color color, Vect offset) |
||||||
|
{ |
||||||
|
setShadow(true); |
||||||
|
setShadowColor(color); |
||||||
|
setShadowOffset(offset); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setShadow(boolean shadow) |
||||||
|
{ |
||||||
|
this.shadow = shadow; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setShadowColor(Color shadowColor) |
||||||
|
{ |
||||||
|
this.shadowColor = shadowColor; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setShadowOffset(Vect shadowOffset) |
||||||
|
{ |
||||||
|
this.shadowOffset = shadowOffset; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setColor(Color color) |
||||||
|
{ |
||||||
|
this.color = color; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setAlign(AlignX align) |
||||||
|
{ |
||||||
|
this.align = align; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setText(String text) |
||||||
|
{ |
||||||
|
this.text = new StringWrapper(text); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setText(StringProvider text) |
||||||
|
{ |
||||||
|
this.text = text; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setVPaddingPercent(double percY) |
||||||
|
{ |
||||||
|
yPaddingPerc = percY; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public double computeWidth(double height) |
||||||
|
{ |
||||||
|
return font.getWidth(this.text.getString(), height * ((100 - yPaddingPerc * 2) / 100D)); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,30 @@ |
|||||||
|
package mightypork.gamecore.gui.events; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.eventbus.BusEvent; |
||||||
|
import mightypork.utils.eventbus.events.flags.DirectEvent; |
||||||
|
import mightypork.utils.eventbus.events.flags.NonConsumableEvent; |
||||||
|
import mightypork.utils.eventbus.events.flags.NonRejectableEvent; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Intended use is to notify UI component sub-clients that they should poll |
||||||
|
* their cached constraints. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
@DirectEvent |
||||||
|
@NonConsumableEvent |
||||||
|
@NonRejectableEvent |
||||||
|
public class LayoutChangeEvent extends BusEvent<LayoutChangeListener> { |
||||||
|
|
||||||
|
public LayoutChangeEvent() { |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void handleBy(LayoutChangeListener handler) |
||||||
|
{ |
||||||
|
handler.onLayoutChanged(); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,7 @@ |
|||||||
|
package mightypork.gamecore.gui.events; |
||||||
|
|
||||||
|
|
||||||
|
public interface LayoutChangeListener { |
||||||
|
|
||||||
|
public void onLayoutChanged(); |
||||||
|
} |
@ -0,0 +1,33 @@ |
|||||||
|
package mightypork.gamecore.gui.events; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.eventbus.BusEvent; |
||||||
|
import mightypork.utils.eventbus.events.flags.SingleReceiverEvent; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Request to change screen |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
@SingleReceiverEvent |
||||||
|
public class ScreenRequest extends BusEvent<ScreenRequestListener> { |
||||||
|
|
||||||
|
private final String scrName; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param screenKey screen name |
||||||
|
*/ |
||||||
|
public ScreenRequest(String screenKey) { |
||||||
|
scrName = screenKey; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void handleBy(ScreenRequestListener handler) |
||||||
|
{ |
||||||
|
handler.showScreen(scrName); |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,15 @@ |
|||||||
|
package mightypork.gamecore.gui.events; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* {@link ScreenRequest} listener |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public interface ScreenRequestListener { |
||||||
|
|
||||||
|
/** |
||||||
|
* @param key screen to show |
||||||
|
*/ |
||||||
|
void showScreen(String key); |
||||||
|
} |
@ -0,0 +1,44 @@ |
|||||||
|
package mightypork.gamecore.gui.events; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.eventbus.BusEvent; |
||||||
|
import mightypork.utils.eventbus.events.flags.NonConsumableEvent; |
||||||
|
import mightypork.utils.eventbus.events.flags.NotLoggedEvent; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Screen resolution or mode was changed |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
@NonConsumableEvent |
||||||
|
@NotLoggedEvent |
||||||
|
public class ViewportChangeEvent extends BusEvent<ViewportChangeListener> { |
||||||
|
|
||||||
|
private final Vect screenSize; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param size new screen size |
||||||
|
*/ |
||||||
|
public ViewportChangeEvent(Vect size) { |
||||||
|
this.screenSize = size; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return new screen size |
||||||
|
*/ |
||||||
|
public Vect getScreenSize() |
||||||
|
{ |
||||||
|
return screenSize; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void handleBy(ViewportChangeListener handler) |
||||||
|
{ |
||||||
|
handler.onViewportChanged(this); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,17 @@ |
|||||||
|
package mightypork.gamecore.gui.events; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* {@link ViewportChangeEvent} listener |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public interface ViewportChangeListener { |
||||||
|
|
||||||
|
/** |
||||||
|
* Handle event |
||||||
|
* |
||||||
|
* @param event |
||||||
|
*/ |
||||||
|
void onViewportChanged(ViewportChangeEvent event); |
||||||
|
} |
@ -0,0 +1,112 @@ |
|||||||
|
package mightypork.gamecore.gui.screens; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.ArrayList; |
||||||
|
import java.util.Collection; |
||||||
|
import java.util.Collections; |
||||||
|
import java.util.Comparator; |
||||||
|
import java.util.List; |
||||||
|
|
||||||
|
import mightypork.utils.eventbus.clients.DelegatingClient; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Screen with multiple instances of {@link ScreenLayer} |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class LayeredScreen extends Screen { |
||||||
|
|
||||||
|
/** |
||||||
|
* Wrapper for delegating client, to use custom client ordering. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
private class LayersClient implements DelegatingClient { |
||||||
|
|
||||||
|
@SuppressWarnings({ "unchecked", "rawtypes" }) |
||||||
|
@Override |
||||||
|
public Collection getChildClients() |
||||||
|
{ |
||||||
|
return layersByEventPriority; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean doesDelegate() |
||||||
|
{ |
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
private final List<ScreenLayer> layersByZIndex = new ArrayList<>(); |
||||||
|
private final List<ScreenLayer> layersByEventPriority = new ArrayList<>(); |
||||||
|
|
||||||
|
private final LayersClient layersClient = new LayersClient(); |
||||||
|
|
||||||
|
|
||||||
|
public LayeredScreen() { |
||||||
|
addChildClient(layersClient); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
protected void renderScreen() |
||||||
|
{ |
||||||
|
for (final ScreenLayer layer : layersByZIndex) { |
||||||
|
if (layer.isVisible()) layer.render(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a layer to the screen. |
||||||
|
* |
||||||
|
* @param layer |
||||||
|
*/ |
||||||
|
protected void addLayer(ScreenLayer layer) |
||||||
|
{ |
||||||
|
this.layersByZIndex.add(layer); |
||||||
|
this.layersByEventPriority.add(layer); |
||||||
|
|
||||||
|
Collections.sort(layersByEventPriority, new Comparator<Overlay>() { |
||||||
|
|
||||||
|
@Override |
||||||
|
public int compare(Overlay o1, Overlay o2) |
||||||
|
{ |
||||||
|
return o2.getEventPriority() - o1.getEventPriority(); |
||||||
|
} |
||||||
|
|
||||||
|
}); |
||||||
|
|
||||||
|
Collections.sort(layersByZIndex, new Comparator<Overlay>() { |
||||||
|
|
||||||
|
@Override |
||||||
|
public int compare(Overlay o1, Overlay o2) |
||||||
|
{ |
||||||
|
return o1.getZIndex() - o2.getZIndex(); |
||||||
|
} |
||||||
|
|
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
protected void onScreenEnter() |
||||||
|
{ |
||||||
|
for (final ScreenLayer layer : layersByEventPriority) { |
||||||
|
layer.onScreenEnter(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
protected void onScreenLeave() |
||||||
|
{ |
||||||
|
for (final ScreenLayer layer : layersByEventPriority) { |
||||||
|
layer.onScreenLeave(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} |
@ -0,0 +1,222 @@ |
|||||||
|
package mightypork.gamecore.gui.screens; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.ArrayList; |
||||||
|
import java.util.Collection; |
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.graphics.Renderable; |
||||||
|
import mightypork.gamecore.gui.components.layout.ConstraintLayout; |
||||||
|
import mightypork.gamecore.gui.events.LayoutChangeListener; |
||||||
|
import mightypork.gamecore.input.KeyBinder; |
||||||
|
import mightypork.gamecore.input.KeyBindingPool; |
||||||
|
import mightypork.gamecore.input.KeyStroke; |
||||||
|
import mightypork.gamecore.input.Trigger; |
||||||
|
import mightypork.utils.annotations.Stub; |
||||||
|
import mightypork.utils.eventbus.clients.BusNode; |
||||||
|
import mightypork.utils.interfaces.Enableable; |
||||||
|
import mightypork.utils.interfaces.Hideable; |
||||||
|
import mightypork.utils.interfaces.Updateable; |
||||||
|
import mightypork.utils.math.color.Color; |
||||||
|
import mightypork.utils.math.constraints.num.Num; |
||||||
|
import mightypork.utils.math.constraints.vect.Vect; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Abstract overlay.<br> |
||||||
|
* Overlay is connected to event bus and is renderable. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class Overlay extends BusNode implements Comparable<Overlay>, Updateable, Renderable, KeyBinder, Hideable, Enableable, LayoutChangeListener { |
||||||
|
|
||||||
|
private boolean visible = true; |
||||||
|
private boolean enabled = true; |
||||||
|
|
||||||
|
private final KeyBindingPool keybindings = new KeyBindingPool(); |
||||||
|
|
||||||
|
/** Root layout, rendered and attached to the event bus. */ |
||||||
|
protected final ConstraintLayout root; |
||||||
|
|
||||||
|
/** Constraint: Mouse position. */ |
||||||
|
protected final Vect mouse; |
||||||
|
|
||||||
|
/** Extra rendered items (outside root) */ |
||||||
|
protected final Collection<Renderable> rendered = new ArrayList<>(); |
||||||
|
|
||||||
|
/** Extra updated items (outside root - those can just implement Updateable) */ |
||||||
|
protected final Collection<Updateable> updated = new ArrayList<>(); |
||||||
|
private Num alphaMul = Num.ONE; |
||||||
|
|
||||||
|
|
||||||
|
public Overlay() { |
||||||
|
|
||||||
|
this.mouse = App.input().getMousePos(); |
||||||
|
|
||||||
|
this.root = new ConstraintLayout(App.gfx().getRect()); |
||||||
|
addChildClient(root); |
||||||
|
addChildClient(keybindings); |
||||||
|
|
||||||
|
rendered.add(root); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final void bindKey(KeyStroke stroke, Trigger edge, Runnable task) |
||||||
|
{ |
||||||
|
keybindings.bindKey(stroke, edge, task); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final void unbindKey(KeyStroke stroke) |
||||||
|
{ |
||||||
|
keybindings.unbindKey(stroke); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final boolean isVisible() |
||||||
|
{ |
||||||
|
return visible; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setVisible(boolean visible) |
||||||
|
{ |
||||||
|
if (visible != this.visible) { |
||||||
|
this.visible = visible; |
||||||
|
root.setVisible(visible); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void setEnabled(boolean yes) |
||||||
|
{ |
||||||
|
if (enabled != yes) { |
||||||
|
this.enabled = yes; |
||||||
|
root.setEnabled(yes); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean isEnabled() |
||||||
|
{ |
||||||
|
return enabled; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get rendering layer |
||||||
|
* |
||||||
|
* @return higher = on top. |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
public abstract int getZIndex(); |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Get event bus listening priority - useful to block incoming events. |
||||||
|
* |
||||||
|
* @return higher = first. |
||||||
|
*/ |
||||||
|
public int getEventPriority() |
||||||
|
{ |
||||||
|
return getZIndex(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Render the overlay. The caller MUST check for visibility himself. |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public void render() |
||||||
|
{ |
||||||
|
if (!isVisible()) return; |
||||||
|
|
||||||
|
Color.pushAlpha(alphaMul); |
||||||
|
for (final Renderable r : rendered) { |
||||||
|
r.render(); |
||||||
|
} |
||||||
|
|
||||||
|
Color.popAlpha(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void update(double delta) |
||||||
|
{ |
||||||
|
if (!isEnabled()) return; |
||||||
|
|
||||||
|
for (final Updateable u : updated) { |
||||||
|
u.update(delta); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public int compareTo(Overlay o) |
||||||
|
{ |
||||||
|
return o.getEventPriority() - getEventPriority(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* <p> |
||||||
|
* Screen size changed. |
||||||
|
* </p> |
||||||
|
* <p> |
||||||
|
* Layouts / components should listen for this event and update their cached |
||||||
|
* constraints; components added to root or directly to this overlay as |
||||||
|
* child clients will receive the event. |
||||||
|
* </p> |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
@Stub |
||||||
|
public void onLayoutChanged() |
||||||
|
{ |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setAlpha(Num alpha) |
||||||
|
{ |
||||||
|
this.alphaMul = alpha; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void setAlpha(double alpha) |
||||||
|
{ |
||||||
|
this.alphaMul = Num.make(alpha); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void show() |
||||||
|
{ |
||||||
|
setVisible(true); |
||||||
|
setEnabled(true); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
public void hide() |
||||||
|
{ |
||||||
|
setVisible(false); |
||||||
|
setEnabled(false); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean isListening() |
||||||
|
{ |
||||||
|
return (isVisible() || isEnabled()); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public boolean doesDelegate() |
||||||
|
{ |
||||||
|
return isListening(); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,153 @@ |
|||||||
|
package mightypork.gamecore.gui.screens; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.graphics.Renderable; |
||||||
|
import mightypork.gamecore.gui.events.LayoutChangeEvent; |
||||||
|
import mightypork.gamecore.gui.events.LayoutChangeListener; |
||||||
|
import mightypork.gamecore.input.KeyBinder; |
||||||
|
import mightypork.gamecore.input.KeyBindingPool; |
||||||
|
import mightypork.gamecore.input.KeyStroke; |
||||||
|
import mightypork.gamecore.input.Trigger; |
||||||
|
import mightypork.utils.annotations.Stub; |
||||||
|
import mightypork.utils.eventbus.clients.BusNode; |
||||||
|
import mightypork.utils.math.constraints.rect.Rect; |
||||||
|
import mightypork.utils.math.constraints.rect.RectBound; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Screen class. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class Screen extends BusNode implements Renderable, RectBound, KeyBinder, LayoutChangeListener { |
||||||
|
|
||||||
|
private final KeyBindingPool keybindings = new KeyBindingPool(); |
||||||
|
|
||||||
|
private volatile boolean active; |
||||||
|
private volatile boolean needSetupViewport = false; |
||||||
|
|
||||||
|
|
||||||
|
public Screen() { |
||||||
|
|
||||||
|
// disable events initially
|
||||||
|
setListening(false); |
||||||
|
|
||||||
|
addChildClient(keybindings); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
private void fireLayoutChangeEvent() |
||||||
|
{ |
||||||
|
App.bus().sendDirectToChildren(this, new LayoutChangeEvent()); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final void bindKey(KeyStroke stroke, Trigger edge, Runnable task) |
||||||
|
{ |
||||||
|
keybindings.bindKey(stroke, edge, task); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final void unbindKey(KeyStroke stroke) |
||||||
|
{ |
||||||
|
keybindings.unbindKey(stroke); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Prepare for being shown |
||||||
|
* |
||||||
|
* @param shown true to show, false to hide |
||||||
|
*/ |
||||||
|
public final void setActive(boolean shown) |
||||||
|
{ |
||||||
|
if (shown) { |
||||||
|
active = true; |
||||||
|
needSetupViewport = true; |
||||||
|
|
||||||
|
fireLayoutChangeEvent(); |
||||||
|
onScreenEnter(); |
||||||
|
|
||||||
|
// enable events
|
||||||
|
setListening(true); |
||||||
|
|
||||||
|
} else { |
||||||
|
onScreenLeave(); |
||||||
|
|
||||||
|
active = false; |
||||||
|
|
||||||
|
// disable events
|
||||||
|
setListening(false); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return true if screen is the current screen |
||||||
|
*/ |
||||||
|
public final boolean isActive() |
||||||
|
{ |
||||||
|
return active; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void onLayoutChanged() |
||||||
|
{ |
||||||
|
if (!isActive()) return; |
||||||
|
|
||||||
|
needSetupViewport = true; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public final Rect getRect() |
||||||
|
{ |
||||||
|
return App.gfx().getRect(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void render() |
||||||
|
{ |
||||||
|
if (!isActive()) return; |
||||||
|
|
||||||
|
if (needSetupViewport) { |
||||||
|
App.gfx().setupProjection(); |
||||||
|
} |
||||||
|
|
||||||
|
App.gfx().pushState(); |
||||||
|
|
||||||
|
renderScreen(); |
||||||
|
|
||||||
|
App.gfx().popState(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Called when the screen becomes active |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
protected void onScreenEnter() |
||||||
|
{ |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Called when the screen is no longer active |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
protected void onScreenLeave() |
||||||
|
{ |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Render screen contents (context is ready for 2D rendering) |
||||||
|
*/ |
||||||
|
protected abstract void renderScreen(); |
||||||
|
|
||||||
|
} |
@ -0,0 +1,50 @@ |
|||||||
|
package mightypork.gamecore.gui.screens; |
||||||
|
|
||||||
|
|
||||||
|
import mightypork.utils.annotations.Stub; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Screen display layer |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public abstract class ScreenLayer extends Overlay { |
||||||
|
|
||||||
|
private final Screen screen; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @param screen parent screen |
||||||
|
*/ |
||||||
|
public ScreenLayer(Screen screen) { |
||||||
|
this.screen = screen; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* @return parent screen instance |
||||||
|
*/ |
||||||
|
protected final Screen getScreen() |
||||||
|
{ |
||||||
|
return screen; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Called when the screen becomes active |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
protected void onScreenEnter() |
||||||
|
{ |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Called when the screen is no longer active |
||||||
|
*/ |
||||||
|
@Stub |
||||||
|
protected void onScreenLeave() |
||||||
|
{ |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,106 @@ |
|||||||
|
package mightypork.gamecore.gui.screens; |
||||||
|
|
||||||
|
|
||||||
|
import java.util.Collection; |
||||||
|
import java.util.HashMap; |
||||||
|
import java.util.Map; |
||||||
|
import java.util.TreeSet; |
||||||
|
|
||||||
|
import mightypork.gamecore.core.App; |
||||||
|
import mightypork.gamecore.graphics.Renderable; |
||||||
|
import mightypork.gamecore.gui.events.LayoutChangeEvent; |
||||||
|
import mightypork.gamecore.gui.events.ScreenRequestListener; |
||||||
|
import mightypork.gamecore.gui.events.ViewportChangeEvent; |
||||||
|
import mightypork.gamecore.gui.events.ViewportChangeListener; |
||||||
|
import mightypork.utils.eventbus.clients.BusNode; |
||||||
|
import mightypork.utils.logging.Log; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Game screens holder; Takes care of rendering and screen requests. |
||||||
|
* |
||||||
|
* @author Ondřej Hruška (MightyPork) |
||||||
|
*/ |
||||||
|
public class ScreenRegistry extends BusNode implements ScreenRequestListener, ViewportChangeListener, Renderable { |
||||||
|
|
||||||
|
private final Map<String, Screen> screens = new HashMap<>(); |
||||||
|
private final Collection<Overlay> overlays = new TreeSet<>(); |
||||||
|
private volatile Screen active = null; |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add a screen |
||||||
|
* |
||||||
|
* @param name screen key for calling |
||||||
|
* @param screen added screen |
||||||
|
*/ |
||||||
|
public void addScreen(String name, Screen screen) |
||||||
|
{ |
||||||
|
screens.put(name, screen); |
||||||
|
addChildClient(screen); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Add an overlay |
||||||
|
* |
||||||
|
* @param overlay added overlay |
||||||
|
*/ |
||||||
|
public void addOverlay(Overlay overlay) |
||||||
|
{ |
||||||
|
overlays.add(overlay); |
||||||
|
addChildClient(overlay); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void showScreen(String key) |
||||||
|
{ |
||||||
|
Log.f3("Request to show screen \"" + key + "\""); |
||||||
|
|
||||||
|
// find screen to show
|
||||||
|
final Screen toShow = screens.get(key); |
||||||
|
if (toShow == null) { |
||||||
|
throw new RuntimeException("Screen " + key + " not defined."); |
||||||
|
} |
||||||
|
|
||||||
|
// deactivate last screen
|
||||||
|
if (active != null) { |
||||||
|
active.setActive(false); |
||||||
|
} |
||||||
|
|
||||||
|
// activate new screen
|
||||||
|
toShow.setActive(true); |
||||||
|
|
||||||
|
active = toShow; |
||||||
|
|
||||||
|
fireLayoutUpdateEvent(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void render() |
||||||
|
{ |
||||||
|
if (active != null) { |
||||||
|
active.render(); |
||||||
|
|
||||||
|
for (final Overlay overlay : overlays) { |
||||||
|
if (overlay.isVisible()) overlay.render(); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
@Override |
||||||
|
public void onViewportChanged(ViewportChangeEvent event) |
||||||
|
{ |
||||||
|
if (active != null) fireLayoutUpdateEvent(); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
private void fireLayoutUpdateEvent() |
||||||
|
{ |
||||||
|
App.bus().sendDirectToChildren(this, new LayoutChangeEvent()); |
||||||
|
} |
||||||
|
|
||||||
|
} |
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in new issue