2016-06-22 4 views
0

私は長方形のオブジェクトを表示しようとしているので、回転の外観が変わらず、OpenSceneGraphがズームされます。私はosg::AutoTransformが私のために働くはずであることを発見しました。スクリーンにオートローテーションされたオブジェクトをピクセルグリッドに合わせる方法は?

しかし、次のコードでは、デフォルトのLINEARの代わりにNEARESTにテクスチャフィルタを設定しても、壊れた結果が表示されます。 LINEARの場合、結果は単にぼやけていますが、NEARESTではテクセルラインが不足することがあります。

#include <osg/Node> 
#include <osgViewer/Viewer> 
#include <osg/Texture2D> 
#include <osg/Geode> 
#include <osg/AutoTransform> 

osg::ref_ptr<osg::Node> createFixedSizeTexture(osg::Image *image,int W,int H) 
{ 
    osg::Vec3Array& verts = *new osg::Vec3Array(4); 
    verts[0] = osg::Vec3(-W/2., -H/2., 0); 
    verts[1] = osg::Vec3(+W/2., -H/2., 0); 
    verts[2] = osg::Vec3(+W/2., +H/2., 0); 
    verts[3] = osg::Vec3(-W/2., +H/2., 0); 

    osg::Vec2Array& texcoords = *new osg::Vec2Array(4); 
    texcoords[0].set(0,0); 
    texcoords[1].set(1,0); 
    texcoords[2].set(1,1); 
    texcoords[3].set(0,1); 

    osg::Geometry*const geometry = new osg::Geometry; 
    geometry->setVertexArray(&verts); 
    geometry->setTexCoordArray(0, &texcoords); 
    geometry->addPrimitiveSet(new osg::DrawArrays(GL_QUADS, 0, 4)); 

    osg::Texture2D*const texture = new osg::Texture2D(image); 
    texture->setResizeNonPowerOfTwoHint(false); 

    geometry->getOrCreateStateSet()->setTextureAttributeAndModes(0, texture, osg::StateAttribute::ON); 

    osg::Geode*const geode = new osg::Geode; 
    geode->addDrawable(geometry); 

    return geode; 
} 

int main() 
{ 
    static constexpr int W=21, H=15; 
    unsigned bits[W*H]; 
    for(int x=0;x<W;++x) 
     for(int y=0;y<H;++y) 
      bits[x+W*y] = (x&y&1)*0xffffffff; 

    osg::Image *formImage = new osg::Image; 
    formImage->setImage(W, H, 1, GL_RGBA, GL_RGBA, GL_UNSIGNED_BYTE, 
         reinterpret_cast<unsigned char*>(bits), osg::Image::NO_DELETE); 

    osg::AutoTransform *at = new osg::AutoTransform; 
    at->setAutoScaleToScreen(true); 
    at->setAutoRotateMode(osg::AutoTransform::ROTATE_TO_SCREEN); 
    at->addChild(createFixedSizeTexture(formImage,W,H)); 

    osgViewer::Viewer viewer; 
    viewer.setUpViewInWindow(0, 0, 800, 600); 
    viewer.setSceneData(at); 

    return viewer.run(); 
} 

これは、最終オブジェクトの非整数の画面座標によるものです。ですから、これを修正するには、オブジェクトをピクセルグリッドに合わせる必要があります。どうすればこれを達成できますか?

答えて

0

ミスアライメントの問題は、スクリーン座標を整数に丸めるカスタム頂点シェーダを使用することで解決できます。

次の例は、OPのコードに基づいており、その代わりにosg::AutoTransformを使用することは、回転及びスケーリングを除去+画面を丸めるの全部が単一のシェーダ座標ん

#include <osg/Node> 
#include <osgViewer/Viewer> 
#include <osg/Texture2D> 
#include <osg/Geode> 

const char*const vertexShader=R"(
    #version 120 

    uniform vec2 screenSize; 

    vec4 roundCoords(vec4 clipPos) 
    { 
     vec2 halfScreenSize=screenSize/2.; 
     vec2 ndcPos=clipPos.xy/clipPos.w;   // transform to normalized device coordinates 
     vec2 screenPos=(ndcPos+1)*halfScreenSize; // transform from NDC space to screen space 
     vec2 screenPosRounded=floor(screenPos);  // round the screen coordinates 
     ndcPos=screenPosRounded.xy/halfScreenSize-1; // transform back to NDC space 
     clipPos.xy=ndcPos*clipPos.w;     // transform back to clip space 
     return clipPos; 
    } 

    void main() 
    { 
     gl_TexCoord[0]=gl_MultiTexCoord0; 
     gl_FrontColor=gl_Color; 

     vec4 translCol=gl_ModelViewProjectionMatrix[3]; 
     // Prevent rotation and unneeded scaling 
     mat4 mvp=mat4(vec4(2./screenSize.x, 0,  0,0), 
         vec4(  0, 2./screenSize.y,0,0), 
         vec4(  0,   0,  1,0), 
         vec4(translCol.xyz/translCol.w, 1)); 

     gl_Position=roundCoords(mvp*gl_Vertex); 
    } 
)"; 

static constexpr int windowWidth=800, windowHeight=600; 

osg::ref_ptr<osg::Node> createFixedSizeTexture(osg::Image *image,int W,int H) 
{ 
    osg::Vec3Array& verts = *new osg::Vec3Array(4); 
    verts[0] = osg::Vec3(-W/2., -H/2., 0); 
    verts[1] = osg::Vec3(+W/2., -H/2., 0); 
    verts[2] = osg::Vec3(+W/2., +H/2., 0); 
    verts[3] = osg::Vec3(-W/2., +H/2., 0); 

    osg::Vec2Array& texcoords = *new osg::Vec2Array(4); 
    texcoords[0].set(0,0); 
    texcoords[1].set(1,0); 
    texcoords[2].set(1,1); 
    texcoords[3].set(0,1); 

    osg::Geometry*const geometry = new osg::Geometry; 
    geometry->setVertexArray(&verts); 
    geometry->setTexCoordArray(0, &texcoords); 
    geometry->addPrimitiveSet(new osg::DrawArrays(GL_QUADS, 0, 4)); 

    osg::Texture2D*const texture = new osg::Texture2D(image); 
    texture->setResizeNonPowerOfTwoHint(false); 

    geometry->getOrCreateStateSet()->setTextureAttributeAndModes(0, texture, osg::StateAttribute::ON); 

    osg::Geode*const geode = new osg::Geode; 
    geode->addDrawable(geometry); 

    osg::Program*const program = new osg::Program; 
    program->addShader(new osg::Shader(osg::Shader::VERTEX, vertexShader)); 

    osg::StateSet*const set = geode->getOrCreateStateSet(); 
    set->setAttributeAndModes(program, osg::StateAttribute::ON); 
    set->addUniform(new osg::Uniform("screenSize" , osg::Vec2(windowWidth,windowHeight))); 

    return geode; 
} 

int main() 
{ 
    static constexpr int W=21, H=15; 
    unsigned bits[W*H]; 
    for(int x=0;x<W;++x) 
     for(int y=0;y<H;++y) 
      bits[x+W*y] = (x&y&1)*0xffffffff; 

    osg::Image *formImage = new osg::Image; 
    formImage->setImage(W, H, 1, GL_RGBA, GL_RGBA, GL_UNSIGNED_BYTE, 
         reinterpret_cast<unsigned char*>(bits), osg::Image::NO_DELETE); 

    osgViewer::Viewer viewer; 
    viewer.setUpViewInWindow(0, 0, windowWidth, windowHeight); 
    viewer.setSceneData(createFixedSizeTexture(formImage,W,H)); 

    return viewer.run(); 
} 
関連する問題