9

まとめ(TL:DR版)

最終的に私たちの目標は、Google の ANGLE プロジェクトを使用して、エアスペースやドライバーの問題なしに、WPF アプリケーションで OpenGL ES コードをネイティブに (つまり、SharpGL などではなく) 利用できるようにすることです。

バックグラウンド:

DirectX よりも OpenGL で気に入っている点の 1 つは、そのクロスプラットフォーム機能です。OS X と Linux の両方、および ES 経由の Android と iOS で優れたサポートを提供します。ただし、Windows では、これを使用するとドライバーの問題が発生するか、さらに悪いことに、多くのカードで適切に実装されていません。

Google のANGLE プロジェクト、または Most-Native-Graphics-Layer-Engine に入ります。

ANGLE は、Direct3D 実装の OpenGL ES 2.0 ラッパーです。つまり、実際の OpenGL ドライバーを必要とせずに、Windows で実行する OpenGL ES 2.0 コードを記述できます。ES 互換性テストに合格するだけでなく、実際に Chrome が WebGL を含むすべてのグラフィックス レンダリングを行う方法であるため、実績のあるテクノロジであることは間違いありません。

質問:

WPF には、WPF 内で Direct3D レンダリングをホストできる D3DImage コントロールがあり、その出力を WPF レンダリング スレッドに適切にコンポストすることで、空域の問題を解決していると考えられます。私の質問は、ANGLE は Direct3D 経由で実装され、D3DImage は Direct3D レンダリングのターゲットであるため、2 つを組み合わせて、OpenGL ES コードを記述し、Windows 上の WPF アプリケーションでホストできるようにすることは可能ですか? ?

これが私たちにとっての「聖杯」です。

しかし、ANGLE は独自のものを使用したいので、D3DImage コントロールによって作成された D3D サーフェス上でレンダリングをターゲットにするように ANGLE を取得する際に壁にぶつかり続けています。これが可能かどうかさえわかりません。これについて議論している記事や参考文献はどこにも見つかりません。

繰り返しになりますが、目標は、エアスペースの問題や OpenGL ドライバーの要件なしに、WPF アプリケーションで動作する共有のクロスプラットフォーム OpenGL (または ES) コードを取得することです。ANGLE/D3DImage を使用するという私の提案は、単なる試みです。それは私がこれまでに思いついた「手段」ですが、それは目標そのものではなく、目標への潜在的な手段にすぎません。私たちを同じ解決策に導く他のものは大歓迎です。

4

1 に答える 1

4

OpenTK.GLControlを介して OpenGL レンダリングを WPF アプリケーションに統合する方法を示すgithub プロジェクトをアップロードしました。

コードは驚くほど単純です。

  1. WindowsFormsHostをWPF アプリケーションに追加する
  2. を構築OpenTK.GLControlし、WindowsFormsHost
    • GraphicsContextFlags.Defaultデスクトップ OpenGL コンテキストのパス
    • GraphicsContextFlags.EmbeddedOpenGL ES (ANGLE) コンテキストのパス
  3. 通常の OpenGL または OpenGL ES コマンドを使用してレンダリングする

ヒント: OpenTKOpenTK.GLControlは NuGet パッケージとしても利用できます。ANGLE サポートが改善された新しいリリースが明日予定されています。

WindowsFormsHost アプローチは空域の制限を受けることに注意してください。これが問題である場合は、解決策についてこちらの回答を参照してください。

// This code is public domain
#define USE_ANGLE

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Forms;
using System.Windows.Forms.Integration;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;

using OpenTK;
using OpenTK.Graphics;

#if USE_ANGLE
using OpenTK.Graphics.ES20;
#else
using OpenTK.Graphics.OpenGL;
#endif

namespace WPF.Angle
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        GLControl glControl;

        public MainWindow()
        {
            InitializeComponent();
        }

        private void WindowsFormsHost_Initialized(object sender, EventArgs e)
        {
            var flags = GraphicsContextFlags.Default;
#if USE_ANGLE
            flags = GraphicsContextFlags.Embedded;
#endif
            glControl = new GLControl(new GraphicsMode(32, 24), 2, 0, flags);
            glControl.MakeCurrent();
            glControl.Paint += GLControl_Paint;
            glControl.Dock = DockStyle.Fill;
            (sender as WindowsFormsHost).Child = glControl;
        }

        private void GLControl_Paint(object sender, PaintEventArgs e)
        {
            GL.ClearColor(
                (float)Red.Value,
                (float)Green.Value,
                (float)Blue.Value,
                1);
            GL.Clear(
                ClearBufferMask.ColorBufferBit |
                ClearBufferMask.DepthBufferBit |
                ClearBufferMask.StencilBufferBit);

            glControl.SwapBuffers();
        }

        private void Slider_ValueChanged(object sender, RoutedPropertyChangedEventArgs<double> e)
        {
            glControl.Invalidate();
        }
    }
}
于 2014-03-15T00:04:22.800 に答える