忍者ブログ

Memeplexes

プログラミング、3DCGとその他いろいろについて

かんたんXNA その24 アルファ・ブレンディング

このページは古いです
最新版はこちら

XNAでは物を半透明に描くことが出来ます。
ゼリーのように、後ろのものが透けて見えるのです。

これをアルファ・ブレンディングと言います。
不透明度(アルファ)を考慮する描画だからです。

アルファ・ブレンディングを使うには、まず
RenderState.AlphaBlendEnableプロパティをtrueにセットします。

public bool AlphaBlendEnable { get; set; }

これをtrueにセットするとアルファ・ブレンディングが
使えるようになるのですが、このまま描画してもやはり
半透明にはなりません。

そのまま、いつもと変わりなく描画されてしまいます。

これは、まだどのようなアルファ・ブレンディングを使うかを
指定していないからです。
(アルファ・ブレンディングにもいろいろな種類があるということです。)

デフォルトの設定では、背景は全く考慮されず
塗りつぶされるアルファ・ブレンディングを行います。
つまり、普通の描画と変わりません!

ポリゴンをスライムのように透明にするには、
RenderState.DestinationBlendプロパティを使います。

public Blend DestinationBlend { get; set; }

Blend列挙体
メンバ名 説明
Zero × 0
One × 1
SourceColor × "これから描くポリゴンの色"

※正確には掛け算と言うより、「それぞれの色の要素が”これから書くポリゴンの色”の対応する色の要素倍される」です。

例えば、(1, 1, 1)と(0.1, 0.2, 0.3)なら(0.1, 0.2, 0.3)。
(0.1, 0.2, 0.3)と(0.1, 0.2, 0.3)なら(0.01, 0.04, 0.09)
InverseSourceColor × (白 - "これから描くポリゴンの色")
SourceAlpha × "これから描くポリゴンの不透明度"
InverseSourceAlpha × (1 - "これから描くポリゴンの不透明度")
DestinationAlpha × "背景の不透明度"
InverseDestinationAlpha × (1 - "背景の不透明度")
DestinationColor × 背景の色
InverseDestinationColor × (白 - "背景の色")
SourceAlphaSaturation × Min("これから描くポリゴンの不透明度", 1 - 描画前の不透明度)
BothInverseSourceAlpha これはDestinationBlendプロパティにセットする必要はありません。これをSourceBlendプロパティにセットするとDestinationBlendの中身は上書きされます(ただしプロパティの値には反映されず、Zeroのままですが)。

この値をセットすると、
最終的な色 =
描画前の色 × 描画するポリゴンの不透明度
+ 描画するポリゴンの色 × (1 - 描画するポリゴンの不透明度)

となります。
BlendFactor × "RenderState.BlendFactorプロパティにセットした色"
InverseBlendFactor × (白 - "RenderState.BlendFactorプロパティにセットした色")
ただしこのモードは、GraphicsDeviceCapabilitiesクラスでSourceBlendCapabilitiesプロパティかDestinationBlendCapabilitiesプロパティのSupportsBlendFactorプロパティ(こいつはgetだけでsetはできません)がtrueのときにしか使えません。
BothSourceAlpha  Obsoleteです。他のメンバを使いましょう。


このプロパティは、ポリゴンをその上に描画する前の背景が
どの程度結果に影響するかを表します。
デフォルトではBlend.Zeroで全く影響されません。
つまり、背景は全く描画結果に現れないのです。(不透明に見えます)

"最終的な色" = "これから描画するポリゴンの色"

もし透明なポリゴンを描こうと思えば、
これを、例えばBlend.Oneに指定してやればいいでしょう。
そうすれば、

"最終的な色" = "描画する前の色" + "描画するポリゴンの色"

となり、半透明に見えます。
using Microsoft.Xna.Framework;
using Microsoft.Xna.Framework.Graphics;


class Triangle
{
    VertexPositionColor[] vertices = new VertexPositionColor[3];

    public Triangle(Vector3 p1, Vector3 p2, Vector3 p3, Color color)
    {
        vertices[0] = new VertexPositionColor(p1, color);
        vertices[1] = new VertexPositionColor(p2, color);
        vertices[2] = new VertexPositionColor(p3, color);
    }

    public void Draw(GraphicsDevice graphicsDevice, BasicEffect effect)
    {
        graphicsDevice.VertexDeclaration = new VertexDeclaration(
            graphicsDevice,
            VertexPositionColor.VertexElements);
        effect.VertexColorEnabled = true;

        effect.Begin();

        foreach (EffectPass pass in effect.CurrentTechnique.Passes)
        {
            pass.Begin();

            graphicsDevice.DrawUserPrimitives<VertexPositionColor>(
                PrimitiveType.TriangleList,
                vertices,
                0,
                1
                );

            pass.End();
        }

        effect.End();
    }
}

class MyGame : Game
{
    GraphicsDeviceManager graphics;
    BasicEffect basicEffect;

    Triangle redTriangle = new Triangle(
        new Vector3(1, 0.5f, -1),
        new Vector3(0, -1, -1),
        new Vector3(-1, 0.5f, -1),
        Color.Red
        );


    public MyGame()
    {
        graphics = new GraphicsDeviceManager(this);
    }

    protected override void LoadGraphicsContent(bool loadAllContent)
    {
        if (loadAllContent)
        {
            basicEffect = new BasicEffect(graphics.GraphicsDevice, null);
            basicEffect.View = Matrix.CreateLookAt(
                new Vector3(0, 0, 3),
                new Vector3(0, 0, 0),
                new Vector3(0, 1, 0)
                );
            basicEffect.Projection = Matrix.CreatePerspectiveFieldOfView(
                MathHelper.ToRadians(45),
                Window.ClientBounds.Width / (float)Window.ClientBounds.Height,
                1, 100
                );
        }
    }

    protected override void Draw(GameTime gameTime)
    {
        graphics.GraphicsDevice.Clear(Color.CornflowerBlue);

        graphics.GraphicsDevice.RenderState.AlphaBlendEnable = true;
        graphics.GraphicsDevice.RenderState.DestinationBlend = Blend.One;
        redTriangle.Draw(graphics.GraphicsDevice, basicEffect);
    }
}


destBlendOne.JPG
ここでは赤い三角形を描画しているのですが、
アルファ・ブレンディングの効果で半透明になり、
ピンクっぽくなっています。

背景のコーンフラワーブルーと赤が
加算されているのです。
(ここでは三角形は1つだけですが、
別に奥にさらに別のポリゴンがあっても問題ありません。
実はこのサンプルプログラム、最初は青い三角形と赤い三角形を2つ
重ねて描画する予定だったのですが、
アルファ・ブレンディングの効果を知るだけなら1つだけでいい事に気付き、
こうなりました。シンプル・イズ・ベストです。)


さて、このサンプルプログラムはアルファ・ブレンディングを
銘打っておきながら不透明度(アルファ)を利用していません。
ただ単純にポリゴンの色を加算しているだけです。

サギです。

不透明値を変更しても実際に描画されるポリゴンの
色には変わりがないのですから!

ポリゴンの色の不透明度のデータを変更したら、
実際に描画される色も変更されて欲しいものです。

それに、この描画の仕方はゼリーやスライムというよりもむしろ
自分で光っている透明な物体、光るゼリーに近いのです。
(光るゼリーなんて見たことありませんが、
他に形容するのは難しいですからね)


アルファが本当に考慮されるアルファ・ブレンディングを行うには、
RenderState.SourceBlendプロパティも使います。

public Blend SourceBlend [ get; set; }

このプロパティは、これから描画するポリゴンの色が
どの程度最終的な色に反映されるかを表します。

デフォルトはBlend.Oneで、これから描くポリゴンの色が
100%反映されるようになっています。
using Microsoft.Xna.Framework;
using Microsoft.Xna.Framework.Graphics;


class Triangle
{
    VertexPositionColor[] vertices = new VertexPositionColor[3];

    public Triangle(Vector3 p1, Vector3 p2, Vector3 p3, Color color)
    {
        vertices[0] = new VertexPositionColor(p1, color);
        vertices[1] = new VertexPositionColor(p2, color);
        vertices[2] = new VertexPositionColor(p3, color);
    }

    public void Draw(GraphicsDevice graphicsDevice, BasicEffect effect)
    {
        graphicsDevice.VertexDeclaration = new VertexDeclaration(
            graphicsDevice,
            VertexPositionColor.VertexElements);
        effect.VertexColorEnabled = true;

        effect.Begin();

        foreach (EffectPass pass in effect.CurrentTechnique.Passes)
        {
            pass.Begin();

            graphicsDevice.DrawUserPrimitives<VertexPositionColor>(
                PrimitiveType.TriangleList,
                vertices,
                0,
                1
                );

            pass.End();
        }

        effect.End();
    }
}

class MyGame : Game
{
    GraphicsDeviceManager graphics
    BasicEffect basicEffect;

    Triangle redTriangle = new Triangle(
        new Vector3(1, 0.5f, -1),
        new Vector3(0, -1, -1),
        new Vector3(-1, 0.5f, -1),
        new Color(255, 0, 0, 20)
        );


    public MyGame()
    {
        graphics = new GraphicsDeviceManager(this);
    }

    protected override void LoadGraphicsContent(bool loadAllContent)
    {
        if (loadAllContent)
        {
            basicEffect = new BasicEffect(graphics.GraphicsDevice, null);
            basicEffect.View = Matrix.CreateLookAt(
                new Vector3(0, 0, 3),
                new Vector3(0, 0, 0),
                new Vector3(0, 1, 0)
                );
            basicEffect.Projection = Matrix.CreatePerspectiveFieldOfView(
                MathHelper.ToRadians(45),
                Window.ClientBounds.Width / (float)Window.ClientBounds.Height,
                1, 100
                );
        }
    }

    protected override void Draw(GameTime gameTime)
    {
        graphics.GraphicsDevice.Clear(Color.CornflowerBlue);

        graphics.GraphicsDevice.RenderState.AlphaBlendEnable = true;
        graphics.GraphicsDevice.RenderState.DestinationBlend = Blend.InverseSourceAlpha;
        graphics.GraphicsDevice.RenderState.SourceBlend = Blend.SourceAlpha;
        redTriangle.Draw(graphics.GraphicsDevice, basicEffect);
    }
}

alpha20.JPG
このサンプルプログラムは、不透明度が255中20の
赤い三角形を描画しています。

「最終的に描画される色」 
        = 不透明度 × 「描画前の色」 + (1 - 不透明度)×「描画するポリゴンの色」


ぼんやりとしていますが、不透明度を上げていくと
徐々にはっきりしてきます。

alpha40.JPG40/255
alpha80.JPG80/255
alpha160.JPG160/255
alpha255.JPG255/255

不透明度が255の三角形は普通に描画したものと変わりません。
これが前のサンプルとは違うところです。

前のサンプルは背景と三角の色を単純に足していましたが、
このサンプルではポリゴンの色の不透明度が考慮されて足されます。

背景の色と三角形の色の強さの合計は1であり、
2つの色が綱引きをしているような感じになります。
片方の色が強くなればもう片方の色は弱くなるのです。
まさにゼロサムゲームです。(ここではサム(和)は1ですが)

拍手[1回]

PR