Как избавиться от чего-то странного при рендеринге шрифта sdf freetypegl? - PullRequest
0 голосов
/ 31 декабря 2018

Я использую freetypegl с режимом рендеринга поля со знаком для отображения текста.Это трудно захватить, поэтому мне пришлось записывать видео (было трудно захватить даже запись fullhd 60 кадров в секунду).Кроме того, мне пришлось записать еще несколько вершин, я не знаю, почему это работает, но большая проблема проявляется, когда окно scene рендерится.Когда данных меньше, проблема почти невидима.

Кроме того, проблема сразу исчезает, когда я открываю Material или любой другой сворачивающийся заголовок

В видео это показано справасторона - Layout окно.Когда мы смотрим поближе, текст - fps, gui window и т. Д. Делают странную гимнастику.

Проблема, конечно, связана с текстом, я сделал тот же тест без рендеринга текста, и все было в порядке.

https://www.youtube.com/watch?v=D5i761KVq0Y&feature=youtu.be

void RenderData::DrawString(const std::string& text, const Vector2& position, uint color, float scale, const Font& font)
{
    using namespace ftgl;
    texture_font_t* ftFont = font.getFTFont();

    float textSlot = FindTexture(font.getTexture().get());

    Vector2 finalPosition = position;

    for (const auto& c : text) {
        AddGlyph(finalPosition, c, textSlot, color, scale, ftFont);
    }
}


void RenderData::AddGlyph(Vector2& position, const char string, float textureSlot, uint color, float scale, texture_font_t* ftFont)
{
    texture_glyph_t* glyph = texture_font_get_glyph(ftFont, &string);
    if (glyph) {
        if (string) {
            position.x += texture_glyph_get_kerning(glyph, &string - 1) * scale;
        }

        float x0 = position.x + static_cast<float>(glyph->offset_x) * scale,
              y0 = position.y + (ftFont->ascender + ftFont->descender - static_cast<float>(glyph->offset_y)) * scale,
              x1 = x0 + static_cast<float>(glyph->width) * scale,
              y1 = y0 + static_cast<float>(glyph->height) * scale,

              u0 = glyph->s0,
              v0 = glyph->t0,
              u1 = glyph->s1,
              v1 = glyph->t1;

        //adds vertex data 
        AddVertexData(Vector2(x0, y0), Vector2(u0, v0), color, textureSlot);
        AddVertexData(Vector2(x0, y1), Vector2(u0, v1), color, textureSlot);
        AddVertexData(Vector2(x1, y1), Vector2(u1, v1), color, textureSlot);
        AddVertexData(Vector2(x1, y0), Vector2(u1, v0), color, textureSlot);

        //adds indexes with rect pattern - index + 0; index + 1; index + 2; 
        //index + 2; index + 3; index + 0
        AddRectElements();

        position.x += glyph->advance_x * scale;
        position.y += glyph->advance_y * scale;
    }
}


float RenderData::FindTexture(Texture* t)
{
    float result = 0.0f;
    bool found = false;
    for (uint i = 0; i < texture.size(); ++i) {
        if (texture[i] == t) {
            result = static_cast<float>(i + 1);
            found = true;
            break;
        }
    }

    if (!found) {
        const int maxTexturePerDrawCall = 32;
        if (texture.size() >= maxTexturePerDrawCall ) {
            AddDrawCall();
            Clear();
        }
        texture.push_back(t);
        result = static_cast<float>(texture.size());
    }

    return result;
}

Font::Font(const std::string& name, const std::string& filename, float size)
    : name(name), filename(filename), size(size)
{
    FTAtlas = ftgl::texture_atlas_new(512, 512, 1);
    FTFont = ftgl::texture_font_new_from_file(FTAtlas, size, filename.c_str());
    FTFont->rendermode = RENDER_SIGNED_DISTANCE_FIELD;

    TextureParameters parameters = { RED, TextureFilter::LINEAR, LINEAR, CLAMP_TO_EDGE };
    texture = texture->Create(512, 512, parameters);
    texture->setData(FTAtlas->data);
}

и фрагментный шейдер

#version 460 core

layout (location = 0) out vec4 f_color;                                                             

layout(location = 1) uniform float width;
layout(location = 2) uniform float edge;

layout(location = 5) uniform sampler2D u_textureSampler[32];

in vec2 te_uv;                                              
in vec4 te_color;
in float te_textID;

void main()
{
    float finalAlpha = 1.0f;
    if(te_textID > 0) {
        int texID = int(te_textID - 0.5);   
        float distance = 1.0 - texture2D(u_textureSampler[texID], te_uv).r,
              alpha = 1.0 - smoothstep(width, width + edge, distance);

        finalAlpha = alpha;
    }    

    f_color = vec4(te_color.rgb, te_color.a * finalAlpha);
}

ничего не меняется, когда я переключаю его в обычный режим рендеринга вместо sdf, но, возможно, с sdf естьнаоборот (это выглядит ужасно при обычном рендере).

Почему текст ведет себя странно?Как это исправить?

...