diff options
Diffstat (limited to 'src/main/java/io/polyfrost/oneconfig/renderer')
-rw-r--r-- | src/main/java/io/polyfrost/oneconfig/renderer/TrueTypeFont.java | 508 |
1 files changed, 508 insertions, 0 deletions
diff --git a/src/main/java/io/polyfrost/oneconfig/renderer/TrueTypeFont.java b/src/main/java/io/polyfrost/oneconfig/renderer/TrueTypeFont.java new file mode 100644 index 0000000..1ce922b --- /dev/null +++ b/src/main/java/io/polyfrost/oneconfig/renderer/TrueTypeFont.java @@ -0,0 +1,508 @@ +package io.polyfrost.oneconfig.renderer; + +import org.lwjgl.BufferUtils; +import org.lwjgl.opengl.GL11; +import org.lwjgl.util.glu.GLU; + +import java.awt.*; +import java.awt.image.BufferedImage; +import java.awt.image.DataBuffer; +import java.awt.image.DataBufferByte; +import java.awt.image.DataBufferInt; +import java.nio.ByteBuffer; +import java.nio.ByteOrder; +import java.nio.IntBuffer; +import java.util.HashMap; +import java.util.Map; + + +/** + * A TrueType font implementation originally for Slick, edited for Bobjob's Engine + * + * @original author James Chambers (Jimmy) + * @original author Jeremy Adams (elias4444) + * @original author Kevin Glass (kevglass) + * @original author Peter Korzuszek (genail) + * @new version edited by David Aaron Muhar (bobjob) + */ +public class TrueTypeFont { + public final static int + ALIGN_LEFT = 0, + ALIGN_RIGHT = 1, + ALIGN_CENTER = 2; + /** + * Array that holds necessary information about the font characters + */ + private IntObject[] charArray = new IntObject[256]; + + /** + * Map of user defined font characters (Character <-> IntObject) + */ + private Map customChars = new HashMap(); + + /** + * Boolean flag on whether AntiAliasing is enabled or not + */ + private boolean antiAlias; + + /** + * Font's size + */ + private int fontSize = 0; + + /** + * Font's height + */ + private int fontHeight = 0; + + /** + * Texture used to cache the font 0-255 characters + */ + private int fontTextureID; + + /** + * Default font texture width + */ + private int textureWidth = 512; + + /** + * Default font texture height + */ + private int textureHeight = 512; + + /** + * A reference to Java's AWT Font that we create our font texture from + */ + private Font font; + + /** + * The font metrics for our Java AWT font + */ + private FontMetrics fontMetrics; + + + private int correctL = 9, correctR = 8; + + private class IntObject { + /** + * Character's width + */ + public int width; + + /** + * Character's height + */ + public int height; + + /** + * Character's stored x position + */ + public int storedX; + + /** + * Character's stored y position + */ + public int storedY; + } + + + public TrueTypeFont(Font font, boolean antiAlias, char[] additionalChars) { + this.font = font; + this.fontSize = font.getSize() + 3; + this.antiAlias = antiAlias; + + createSet(additionalChars); + + fontHeight -= 1; + if (fontHeight <= 0) fontHeight = 1; + } + + public TrueTypeFont(Font font, boolean antiAlias) { + this(font, antiAlias, null); + } + + public void setCorrection(boolean on) { + if (on) { + correctL = 2; + correctR = 1; + } else { + correctL = 0; + correctR = 0; + } + } + + private BufferedImage getFontImage(char ch) { + // Create a temporary image to extract the character's size + BufferedImage tempfontImage = new BufferedImage(1, 1, + BufferedImage.TYPE_INT_ARGB); + Graphics2D g = (Graphics2D) tempfontImage.getGraphics(); + if (antiAlias) { + g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, + RenderingHints.VALUE_ANTIALIAS_ON); + } + g.setFont(font); + fontMetrics = g.getFontMetrics(); + int charwidth = fontMetrics.charWidth(ch) + 8; + + if (charwidth <= 0) { + charwidth = 7; + } + int charheight = fontMetrics.getHeight() + 3; + if (charheight <= 0) { + charheight = fontSize; + } + + // Create another image holding the character we are creating + BufferedImage fontImage; + fontImage = new BufferedImage(charwidth, charheight, + BufferedImage.TYPE_INT_ARGB); + Graphics2D gt = (Graphics2D) fontImage.getGraphics(); + if (antiAlias == true) { + gt.setRenderingHint(RenderingHints.KEY_ANTIALIASING, + RenderingHints.VALUE_ANTIALIAS_ON); + } + gt.setFont(font); + + gt.setColor(Color.WHITE); + int charx = 3; + int chary = 1; + gt.drawString(String.valueOf(ch), (charx), (chary) + + fontMetrics.getAscent()); + + return fontImage; + + } + + private void createSet(char[] customCharsArray) { + // If there are custom chars then I expand the font texture twice + if (customCharsArray != null && customCharsArray.length > 0) { + textureWidth *= 2; + } + + // In any case this should be done in other way. Texture with size 512x512 + // can maintain only 256 characters with resolution of 32x32. The texture + // size should be calculated dynamicaly by looking at character sizes. + + try { + + BufferedImage imgTemp = new BufferedImage(textureWidth, textureHeight, BufferedImage.TYPE_INT_ARGB); + Graphics2D g = (Graphics2D) imgTemp.getGraphics(); + + g.setColor(new Color(0, 0, 0, 1)); + g.fillRect(0, 0, textureWidth, textureHeight); + + int rowHeight = 0; + int positionX = 0; + int positionY = 0; + + int customCharsLength = (customCharsArray != null) ? customCharsArray.length : 0; + + for (int i = 0; i < 256 + customCharsLength; i++) { + + // get 0-255 characters and then custom characters + char ch = (i < 256) ? (char) i : customCharsArray[i - 256]; + + BufferedImage fontImage = getFontImage(ch); + + IntObject newIntObject = new IntObject(); + + newIntObject.width = fontImage.getWidth(); + newIntObject.height = fontImage.getHeight(); + + if (positionX + newIntObject.width >= textureWidth) { + positionX = 0; + positionY += rowHeight; + rowHeight = 0; + } + + newIntObject.storedX = positionX; + newIntObject.storedY = positionY; + + if (newIntObject.height > fontHeight) { + fontHeight = newIntObject.height; + } + + if (newIntObject.height > rowHeight) { + rowHeight = newIntObject.height; + } + + // Draw it here + g.drawImage(fontImage, positionX, positionY, null); + + positionX += newIntObject.width; + + if (i < 256) { // standard characters + charArray[i] = newIntObject; + } else { // custom characters + customChars.put(new Character(ch), newIntObject); + } + + fontImage = null; + } + + fontTextureID = loadImage(imgTemp); + + + //.getTexture(font.toString(), imgTemp); + + } catch (Exception e) { + System.err.println("Failed to create font."); + e.printStackTrace(); + } + } + + private void drawQuad(float drawX, float drawY, float drawX2, float drawY2, + float srcX, float srcY, float srcX2, float srcY2) { + float DrawWidth = drawX2 - drawX; + float DrawHeight = drawY2 - drawY; + float TextureSrcX = srcX / textureWidth; + float TextureSrcY = srcY / textureHeight; + float SrcWidth = srcX2 - srcX; + float SrcHeight = srcY2 - srcY; + float RenderWidth = (SrcWidth / textureWidth); + float RenderHeight = (SrcHeight / textureHeight); + + GL11.glTexCoord2f(TextureSrcX, TextureSrcY); + GL11.glVertex2f(drawX, drawY); + GL11.glTexCoord2f(TextureSrcX, TextureSrcY + RenderHeight); + GL11.glVertex2f(drawX, drawY + DrawHeight); + GL11.glTexCoord2f(TextureSrcX + RenderWidth, TextureSrcY + RenderHeight); + GL11.glVertex2f(drawX + DrawWidth, drawY + DrawHeight); + GL11.glTexCoord2f(TextureSrcX + RenderWidth, TextureSrcY); + GL11.glVertex2f(drawX + DrawWidth, drawY); + } + + public int getWidth(String whatchars) { + int totalwidth = 0; + IntObject intObject = null; + int currentChar = 0; + for (int i = 0; i < whatchars.length(); i++) { + currentChar = whatchars.charAt(i); + if (currentChar < 256) { + intObject = charArray[currentChar]; + } else { + intObject = (IntObject) customChars.get(new Character((char) currentChar)); + } + + if (intObject != null) + totalwidth += intObject.width; + } + return totalwidth; + } + + public int getHeight() { + return fontHeight; + } + + + public int getHeight(String HeightString) { + return fontHeight; + } + + public int getLineHeight() { + return fontHeight; + } + + public void drawString(float x, float y, + String whatchars, float scaleX, float scaleY) { + drawString(x, y, whatchars, 0, whatchars.length() - 1, scaleX, scaleY, ALIGN_LEFT); + } + + public void drawString(float x, float y, + String whatchars, float scaleX, float scaleY, int format) { + drawString(x, y, whatchars, 0, whatchars.length() - 1, scaleX, scaleY, format); + } + + + public void drawString(float x, float y, + String whatchars, int startIndex, int endIndex, + float scaleX, float scaleY, + int format + ) { + + IntObject intObject = null; + int charCurrent; + + + int totalwidth = 0; + int i = startIndex, d, c; + float startY = 0; + + + switch (format) { + case ALIGN_RIGHT: { + d = -1; + c = correctR; + + while (i < endIndex) { + if (whatchars.charAt(i) == '\n') startY -= fontHeight; + i++; + } + break; + } + case ALIGN_CENTER: { + for (int l = startIndex; l <= endIndex; l++) { + charCurrent = whatchars.charAt(l); + if (charCurrent == '\n') break; + if (charCurrent < 256) { + intObject = charArray[charCurrent]; + } else { + intObject = (IntObject) customChars.get(new Character((char) charCurrent)); + } + totalwidth += intObject.width - correctL; + } + totalwidth /= -2; + } + case ALIGN_LEFT: + default: { + d = 1; + c = correctL; + break; + } + + } + + GL11.glEnable(GL11.GL_TEXTURE_2D); + GL11.glBindTexture(GL11.GL_TEXTURE_2D, fontTextureID); + GL11.glBegin(GL11.GL_QUADS); + + while (i >= startIndex && i <= endIndex) { + + charCurrent = whatchars.charAt(i); + if (charCurrent < 256) { + intObject = charArray[charCurrent]; + } else { + intObject = (IntObject) customChars.get((char) charCurrent); + } + + if (intObject != null) { + if (d < 0) totalwidth += (intObject.width - c) * d; + if (charCurrent == '\n') { + startY -= fontHeight * d; + totalwidth = 0; + if (format == ALIGN_CENTER) { + for (int l = i + 1; l <= endIndex; l++) { + charCurrent = whatchars.charAt(l); + if (charCurrent == '\n') break; + if (charCurrent < 256) { + intObject = charArray[charCurrent]; + } else { + intObject = (IntObject) customChars.get((char) charCurrent); + } + totalwidth += intObject.width - correctL; + } + totalwidth /= -2; + } + //if center get next lines total width/2; + } else { + drawQuad((totalwidth + intObject.width) * scaleX + x, startY * scaleY + y, + totalwidth * scaleX + x, + (startY + intObject.height) * scaleY + y, intObject.storedX + intObject.width, + intObject.storedY + intObject.height, intObject.storedX, + intObject.storedY); + if (d > 0) totalwidth += (intObject.width - c) * d; + } + i += d; + + } + } + GL11.glDisable(GL11.GL_TEXTURE_2D); + GL11.glEnd(); + } + + public static int loadImage(BufferedImage bufferedImage) { + try { + short width = (short) bufferedImage.getWidth(); + short height = (short) bufferedImage.getHeight(); + //textureLoader.bpp = bufferedImage.getColorModel().hasAlpha() ? (byte)32 : (byte)24; + int bpp = (byte) bufferedImage.getColorModel().getPixelSize(); + ByteBuffer byteBuffer; + DataBuffer db = bufferedImage.getData().getDataBuffer(); + if (db instanceof DataBufferInt) { + int intI[] = ((DataBufferInt) (bufferedImage.getData().getDataBuffer())).getData(); + byte newI[] = new byte[intI.length * 4]; + for (int i = 0; i < intI.length; i++) { + byte b[] = intToByteArray(intI[i]); + int newIndex = i * 4; + + newI[newIndex] = b[1]; + newI[newIndex + 1] = b[2]; + newI[newIndex + 2] = b[3]; + newI[newIndex + 3] = b[0]; + } + + byteBuffer = ByteBuffer.allocateDirect( + width * height * (bpp / 8)) + .order(ByteOrder.nativeOrder()) + .put(newI); + } else { + byteBuffer = ByteBuffer.allocateDirect( + width * height * (bpp / 8)) + .order(ByteOrder.nativeOrder()) + .put(((DataBufferByte) (bufferedImage.getData().getDataBuffer())).getData()); + } + byteBuffer.flip(); + + + int internalFormat = GL11.GL_RGBA8, + format = GL11.GL_RGBA; + IntBuffer textureId = BufferUtils.createIntBuffer(1); + ; + GL11.glGenTextures(textureId); + GL11.glBindTexture(GL11.GL_TEXTURE_2D, textureId.get(0)); + + + GL11.glTexParameteri(GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_WRAP_S, GL11.GL_CLAMP); + GL11.glTexParameteri(GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_WRAP_T, GL11.GL_CLAMP); + + GL11.glTexParameteri(GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_MAG_FILTER, GL11.GL_LINEAR); + GL11.glTexParameteri(GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_MIN_FILTER, GL11.GL_LINEAR); + + GL11.glTexEnvf(GL11.GL_TEXTURE_ENV, GL11.GL_TEXTURE_ENV_MODE, GL11.GL_MODULATE); + + + GLU.gluBuild2DMipmaps(GL11.GL_TEXTURE_2D, + internalFormat, + width, + height, + format, + GL11.GL_UNSIGNED_BYTE, + byteBuffer); + return textureId.get(0); + + } catch (Exception e) { + e.printStackTrace(); + System.exit(-1); + } + + return -1; + } + + public static boolean isSupported(String fontname) { + Font font[] = getFonts(); + for (int i = font.length - 1; i >= 0; i--) { + if (font[i].getName().equalsIgnoreCase(fontname)) + return true; + } + return false; + } + + public static Font[] getFonts() { + return GraphicsEnvironment.getLocalGraphicsEnvironment().getAllFonts(); + } + + public static byte[] intToByteArray(int value) { + return new byte[]{ + (byte) (value >>> 24), + (byte) (value >>> 16), + (byte) (value >>> 8), + (byte) value}; + } + + public void destroy() { + IntBuffer scratch = BufferUtils.createIntBuffer(1); + scratch.put(0, fontTextureID); + GL11.glBindTexture(GL11.GL_TEXTURE_2D, 0); + GL11.glDeleteTextures(scratch); + } +} |