mirror of https://github.com/adamdruppe/arsd.git
opengl3 limited font wip
This commit is contained in:
parent
368d1d8c59
commit
25e46e9d2d
198
ttf.d
198
ttf.d
|
@ -138,9 +138,10 @@ struct TtfFont {
|
||||||
// ~this() {}
|
// ~this() {}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Version of OpenGL you want it to use. Currently only one option.
|
/// Version of OpenGL you want it to use. Currently only two options.
|
||||||
enum OpenGlFontGLVersion {
|
enum OpenGlFontGLVersion {
|
||||||
old /// old style glBegin/glEnd stuff
|
old, /// old style glBegin/glEnd stuff
|
||||||
|
shader, /// newer style shader stuff
|
||||||
}
|
}
|
||||||
|
|
||||||
/+
|
/+
|
||||||
|
@ -160,20 +161,17 @@ struct DrawingTextContext {
|
||||||
const int bottom; /// ditto
|
const int bottom; /// ditto
|
||||||
}
|
}
|
||||||
|
|
||||||
/++
|
abstract class OpenGlLimitedFontBase() {
|
||||||
Note that the constructor calls OpenGL functions and thus this must be called AFTER
|
void createShaders() {}
|
||||||
the context creation, e.g. on simpledisplay's window first visible delegate.
|
abstract uint glFormat();
|
||||||
|
abstract void startDrawing(Color color);
|
||||||
|
abstract void addQuad(
|
||||||
|
float s0, float t0, float x0, float y0,
|
||||||
|
float s1, float t1, float x1, float y1
|
||||||
|
);
|
||||||
|
abstract void finishDrawing();
|
||||||
|
|
||||||
Any text with characters outside the range you bake in the constructor are simply
|
|
||||||
ignored - that's why it is called "limited" font. The [TtfFont] struct can generate
|
|
||||||
any string on-demand which is more flexible, and even faster for strings repeated
|
|
||||||
frequently, but slower for frequently-changing or one-off strings. That's what this
|
|
||||||
class is made for.
|
|
||||||
|
|
||||||
History:
|
|
||||||
Added April 24, 2020
|
|
||||||
+/
|
|
||||||
class OpenGlLimitedFont(OpenGlFontGLVersion ver = OpenGlFontGLVersion.old) {
|
|
||||||
// FIXME: does this kern?
|
// FIXME: does this kern?
|
||||||
// FIXME: it would be cool if it did per-letter transforms too like word art. make it tangent to some baseline
|
// FIXME: it would be cool if it did per-letter transforms too like word art. make it tangent to some baseline
|
||||||
|
|
||||||
|
@ -225,9 +223,7 @@ class OpenGlLimitedFont(OpenGlFontGLVersion ver = OpenGlFontGLVersion.old) {
|
||||||
bool actuallyDraw = color != Color.transparent;
|
bool actuallyDraw = color != Color.transparent;
|
||||||
|
|
||||||
if(actuallyDraw) {
|
if(actuallyDraw) {
|
||||||
glBindTexture(GL_TEXTURE_2D, _tex);
|
startDrawing(color);
|
||||||
|
|
||||||
glColor4f(cast(float)color.r/255.0, cast(float)color.g/255.0, cast(float)color.b/255.0, cast(float)color.a / 255.0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool newWord = true;
|
bool newWord = true;
|
||||||
|
@ -330,19 +326,14 @@ class OpenGlLimitedFont(OpenGlFontGLVersion ver = OpenGlFontGLVersion.old) {
|
||||||
auto t1 = b.y1 * iph;
|
auto t1 = b.y1 * iph;
|
||||||
|
|
||||||
if(actuallyDraw) {
|
if(actuallyDraw) {
|
||||||
glBegin(GL_QUADS);
|
addQuad(s0, t0, x0, y0, s1, t1, x1, y1);
|
||||||
glTexCoord2f(s0, t0); glVertex2i(x0, y0);
|
|
||||||
glTexCoord2f(s1, t0); glVertex2i(x1, y0);
|
|
||||||
glTexCoord2f(s1, t1); glVertex2i(x1, y1);
|
|
||||||
glTexCoord2f(s0, t1); glVertex2i(x0, y1);
|
|
||||||
glEnd();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
context.x += b.xadvance;
|
context.x += b.xadvance;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(actuallyDraw)
|
if(actuallyDraw)
|
||||||
glBindTexture(GL_TEXTURE_2D, 0); // unbind the texture
|
finishDrawing();
|
||||||
}
|
}
|
||||||
|
|
||||||
private {
|
private {
|
||||||
|
@ -414,6 +405,8 @@ class OpenGlLimitedFont(OpenGlFontGLVersion ver = OpenGlFontGLVersion.old) {
|
||||||
this.descent = descent;
|
this.descent = descent;
|
||||||
this.line_gap = line_gap;
|
this.line_gap = line_gap;
|
||||||
|
|
||||||
|
assert(openGLCurrentContext() !is null);
|
||||||
|
|
||||||
glGenTextures(1, &_tex);
|
glGenTextures(1, &_tex);
|
||||||
glBindTexture(GL_TEXTURE_2D, _tex);
|
glBindTexture(GL_TEXTURE_2D, _tex);
|
||||||
glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
|
glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
|
||||||
|
@ -423,18 +416,169 @@ class OpenGlLimitedFont(OpenGlFontGLVersion ver = OpenGlFontGLVersion.old) {
|
||||||
glTexImage2D(
|
glTexImage2D(
|
||||||
GL_TEXTURE_2D,
|
GL_TEXTURE_2D,
|
||||||
0,
|
0,
|
||||||
GL_ALPHA,
|
glFormat,
|
||||||
width,
|
width,
|
||||||
height,
|
height,
|
||||||
0,
|
0,
|
||||||
GL_ALPHA,
|
glFormat,
|
||||||
GL_UNSIGNED_BYTE,
|
GL_UNSIGNED_BYTE,
|
||||||
buffer.ptr);
|
buffer.ptr);
|
||||||
|
|
||||||
assert(!glGetError());
|
checkGlError();
|
||||||
|
|
||||||
glBindTexture(GL_TEXTURE_2D, 0);
|
glBindTexture(GL_TEXTURE_2D, 0);
|
||||||
|
|
||||||
|
createShaders();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/++
|
||||||
|
Note that the constructor calls OpenGL functions and thus this must be called AFTER
|
||||||
|
the context creation, e.g. on simpledisplay's window first visible delegate.
|
||||||
|
|
||||||
|
Any text with characters outside the range you bake in the constructor are simply
|
||||||
|
ignored - that's why it is called "limited" font. The [TtfFont] struct can generate
|
||||||
|
any string on-demand which is more flexible, and even faster for strings repeated
|
||||||
|
frequently, but slower for frequently-changing or one-off strings. That's what this
|
||||||
|
class is made for.
|
||||||
|
|
||||||
|
History:
|
||||||
|
Added April 24, 2020
|
||||||
|
+/
|
||||||
|
final class OpenGlLimitedFont(OpenGlFontGLVersion ver = OpenGlFontGLVersion.old) : OpenGlLimitedFontBase!() {
|
||||||
|
import arsd.color;
|
||||||
|
import arsd.simpledisplay;
|
||||||
|
|
||||||
|
public this(const ubyte[] ttfData, float fontPixelHeight, dchar firstChar = 32, dchar lastChar = 127) {
|
||||||
|
super(ttfData, fontPixelHeight, firstChar, lastChar);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
override uint glFormat() {
|
||||||
|
// on new-style opengl this is the required value
|
||||||
|
static if(ver == OpenGlFontGLVersion.shader)
|
||||||
|
return GL_RED;
|
||||||
|
else
|
||||||
|
return GL_ALPHA;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
static if(ver == OpenGlFontGLVersion.shader) {
|
||||||
|
private OpenGlShader shader;
|
||||||
|
private static struct Vertex {
|
||||||
|
this(float a, float b, float c, float d, float e = 0.0) {
|
||||||
|
t[0] = a;
|
||||||
|
t[1] = b;
|
||||||
|
xyz[0] = c;
|
||||||
|
xyz[1] = d;
|
||||||
|
xyz[2] = e;
|
||||||
|
}
|
||||||
|
float[2] t;
|
||||||
|
float[3] xyz;
|
||||||
|
}
|
||||||
|
private GlObject!Vertex glo;
|
||||||
|
// GL_DYNAMIC_DRAW FIXME
|
||||||
|
private Vertex[] vertixes;
|
||||||
|
private uint[] indexes;
|
||||||
|
|
||||||
|
public BasicMatrix!(4, 4) mymatrix;
|
||||||
|
public BasicMatrix!(4, 4) projection;
|
||||||
|
|
||||||
|
override void createShaders() {
|
||||||
|
mymatrix.loadIdentity();
|
||||||
|
projection.loadIdentity();
|
||||||
|
|
||||||
|
shader = new OpenGlShader(
|
||||||
|
OpenGlShader.Source(GL_VERTEX_SHADER, `
|
||||||
|
#version 330 core
|
||||||
|
|
||||||
|
`~glo.generateShaderDefinitions()~`
|
||||||
|
|
||||||
|
out vec2 texCoord;
|
||||||
|
|
||||||
|
uniform mat4 mymatrix;
|
||||||
|
uniform mat4 projection;
|
||||||
|
|
||||||
|
void main() {
|
||||||
|
gl_Position = projection * mymatrix * vec4(xyz, 1.0);
|
||||||
|
texCoord = t;
|
||||||
|
}
|
||||||
|
`),
|
||||||
|
OpenGlShader.Source(GL_FRAGMENT_SHADER, `
|
||||||
|
#version 330 core
|
||||||
|
|
||||||
|
in vec2 texCoord;
|
||||||
|
|
||||||
|
out vec4 FragColor;
|
||||||
|
|
||||||
|
uniform vec4 color;
|
||||||
|
uniform sampler2D tex;
|
||||||
|
|
||||||
|
void main() {
|
||||||
|
FragColor = color * vec4(1, 1, 1, texture(tex, texCoord).r);
|
||||||
|
}
|
||||||
|
`),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override void startDrawing(Color color) {
|
||||||
|
glBindTexture(GL_TEXTURE_2D, _tex);
|
||||||
|
|
||||||
|
static if(ver == OpenGlFontGLVersion.shader) {
|
||||||
|
shader.use();
|
||||||
|
shader.uniforms.color() = OGL.vec4f(cast(float)color.r/255.0, cast(float)color.g/255.0, cast(float)color.b/255.0, cast(float)color.a / 255.0);
|
||||||
|
|
||||||
|
shader.uniforms.mymatrix() = OGL.mat4x4f(mymatrix.data);
|
||||||
|
shader.uniforms.projection() = OGL.mat4x4f(projection.data);
|
||||||
|
} else {
|
||||||
|
glColor4f(cast(float)color.r/255.0, cast(float)color.g/255.0, cast(float)color.b/255.0, cast(float)color.a / 255.0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
override void addQuad(
|
||||||
|
float s0, float t0, float x0, float y0,
|
||||||
|
float s1, float t1, float x1, float y1
|
||||||
|
) {
|
||||||
|
static if(ver == OpenGlFontGLVersion.shader) {
|
||||||
|
auto idx = cast(int) vertixes.length;
|
||||||
|
vertixes ~= [
|
||||||
|
Vertex(s0, t0, x0, y0),
|
||||||
|
Vertex(s1, t0, x1, y0),
|
||||||
|
Vertex(s1, t1, x1, y1),
|
||||||
|
Vertex(s0, t1, x0, y1),
|
||||||
|
];
|
||||||
|
|
||||||
|
indexes ~= [
|
||||||
|
idx + 0,
|
||||||
|
idx + 1,
|
||||||
|
idx + 3,
|
||||||
|
idx + 1,
|
||||||
|
idx + 2,
|
||||||
|
idx + 3,
|
||||||
|
];
|
||||||
|
} else {
|
||||||
|
glBegin(GL_QUADS);
|
||||||
|
glTexCoord2f(s0, t0); glVertex2f(x0, y0);
|
||||||
|
glTexCoord2f(s1, t0); glVertex2f(x1, y0);
|
||||||
|
glTexCoord2f(s1, t1); glVertex2f(x1, y1);
|
||||||
|
glTexCoord2f(s0, t1); glVertex2f(x0, y1);
|
||||||
|
glEnd();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
override void finishDrawing() {
|
||||||
|
static if(ver == OpenGlFontGLVersion.shader) {
|
||||||
|
glo = new typeof(glo)(vertixes, indexes);
|
||||||
|
glo.draw();
|
||||||
|
vertixes = vertixes[0..0];
|
||||||
|
vertixes.assumeSafeAppend();
|
||||||
|
indexes = indexes[0..0];
|
||||||
|
indexes.assumeSafeAppend();
|
||||||
|
}
|
||||||
|
glBindTexture(GL_TEXTURE_2D, 0); // unbind the texture
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue