WebGL-クイックガイド

数年前、Javaアプリケーションは、アプレットとJOGLの組み合わせとして、GPU(Graphical Processing Unit)をアドレス指定することにより、Web上の3Dグラフィックスを処理するために使用されていました。アプレットを実行するにはJVMが必要なため、Javaアプレットに依存することが困難になりました。数年後、人々はJavaアプレットの使用をやめました。

Adobe(Flash、AIR)が提供するStage3D APIは、GPUハードウェアアクセラレーションアーキテクチャを提供しました。これらのテクノロジーを使用して、プログラマーはWebブラウザーおよびIOSおよびAndroidプラットフォームで2Dおよび3D機能を備えたアプリケーションを開発できます。Flashはプロプライエタリソフトウェアであったため、Web標準としては使用されませんでした。

2011年3月、WebGLがリリースされました。これは、JVMなしで実行できるオープンウェアです。それは完全にウェブブラウザによって制御されます。

HTML 5の新しいリリースには、2D Canvas、WebGL、SVG、3D CSS変換、SMILなどの3Dグラフィックスをサポートするいくつかの機能があります。このチュートリアルでは、WebGLの基本について説明します。

OpenGLとは何ですか?

OpenGL(Open Graphics Library)は、2Dおよび3Dグラフィックス用のクロス言語のクロスプラットフォームAPIです。これはコマンドのコレクションです。OpenGL4.5は、OpenGLの最新バージョンです。次の表に、OpenGLに関連する一連のテクノロジを示します。

API 使用したテクノロジー
OpenGL ES これは、コンソール、電話、電化製品、車両などの組み込みシステム上の2Dおよび3Dグラフィックス用のライブラリです。OpenGL ES3.1は最新バージョンです。クロノスグループwww.khronos.orgによって維持されています
JOGL これはOpenGLのJavaバインディングです。JOGL 4.5は最新バージョンであり、jogamp.orgによって保守されています。
WebGL これはOpenGLのJavaScriptバインディングです。WebGL 1.0は最新バージョンであり、khronosグループによって管理されています。
OpenGLSL OpenGL Shading Language。これは、OpenGL2.0以降のコンパニオンであるプログラミング言語です。これは、コアOpenGL4.4仕様の一部です。これは、携帯電話やタブレットに存在するような組み込みシステム向けに特別に調整されたAPIです。

Note − WebGLでは、GLSLを使用してシェーダーを記述します。

WebGLとは何ですか?

WebGL(Web Graphics Library)は、Web上の3Dグラフィックスの新しい標準であり、2Dグラフィックスとインタラクティブな3Dグラフィックスをレンダリングすることを目的として設計されています。これは、電話やその他のモバイルデバイス用の低レベルの3DAPIであるOpenGLのES2.0ライブラリから派生しています。WebGLは、ES 2.0(Embedded Systems)と同様の機能を提供し、最新の3Dグラフィックハードウェアで適切に機能します。

HTML5で使用できるJavaScriptAPIです。WebGLコードは、HTML5の<canvas>タグ内に記述されています。これは、インターネットブラウザが使用されていたコンピュータのグラフィックプロセッシングユニット(GPU)にアクセスできるようにする仕様です。

WebGLを開発したのは誰か

名前の付いたアメリカとセルビアのソフトウェアエンジニア Vladimir Vukicevic 基礎作業を行い、WebGLの作成を主導しました

  • 2007年に、ウラジミールは OpenGL HTMLドキュメントのCanvas要素のプロトタイプ。

  • 2011年3月、KronosGroupはWebGLを作成しました。

レンダリング

レンダリングは、コンピュータープログラムを使用してモデルから画像を生成するプロセスです。グラフィックスでは、仮想シーンは、レンダリングプログラムを介して渡される、ジオメトリ、視点、テクスチャ、ライティング、シェーディングなどの情報を使用して記述されます。このレンダリングプログラムの出力はデジタル画像になります。

レンダリングには2つのタイプがあります-

  • Software Rendering −すべてのレンダリング計算はCPUの助けを借りて行われます。

  • Hardware Rendering −すべてのグラフィックス計算はGPU(グラフィックスプロセッシングユニット)によって実行されます。

レンダリングはローカルまたはリモートで実行できます。レンダリングする画像が複雑すぎる場合、レンダリングは、複雑なシーンをレンダリングするために必要な十分なハードウェアリソースを備えた専用サーバー上でリモートで実行されます。とも呼ばれますserver-based rendering。レンダリングは、CPUによってローカルで実行することもできます。それはとして呼ばれますclient-based rendering

WebGLは、クライアントベースのレンダリングアプローチに従って3Dシーンをレンダリングします。画像を取得するために必要なすべての処理は、クライアントのグラフィックハードウェアを使用してローカルで実行されます。

GPU

NVIDIAによると、GPUは「変換、照明、三角形のセットアップ/クリッピング、およびレンダリングエンジンが統合されたシングルチッププロセッサであり、1秒あたり最低1,000万ポリゴンを処理できます」。シーケンシャル処理用に最適化されたいくつかのコアを備えたマルチコアプロセッサとは異なり、GPUは、並列ワークロードを効率的に処理する数千の小さなコアで構成されています。したがって、GPUは、ディスプレイへの出力を目的としたフレームバッファー(完全なフレームデータを含むRAMの一部)での画像の作成を高速化します。

GPUアクセラレーションコンピューティング

GPUアクセラレーションコンピューティングでは、アプリケーションはCPUにロードされます。それが遭遇するときはいつでもcompute-intensiveコードの一部、次にコードのその部分がロードされ、GPUで実行されます。これにより、システムはグラフィックを効率的に処理できます。

GPUには個別のメモリがあり、コードのごく一部の複数のコピーを一度に実行します。GPUは、中央メモリではなく、ローカルメモリにあるすべてのデータを処理します。したがって、GPUで処理する必要のあるデータは、GPUメモリにロード/コピーしてから処理する必要があります。

上記のアーキテクチャを備えたシステムでは、3Dプログラムの処理を高速化するために、CPUとGPU間の通信オーバーヘッドを削減する必要があります。このため、GPUと繰り返し通信するのではなく、すべてのデータをコピーしてGPUに保持する必要があります。

サポートされているブラウザ

次の表に、WebGLをサポートするブラウザのリストを示します。

ウェブブラウザー

ブラウザ名 バージョン サポート
Iインターネット Explorer 11歳以上 完全なサポート
グーグルクローム 39以上 完全なサポート
サファリ 8 完全なサポート
Firefox 36歳以上 部分的なサポート
オペラ 27歳以上 部分的なサポート

モバイルブラウザ

ブラウザ名 バージョン サポート
Chrome for Android 42 部分的なサポート
Androidブラウザ 40 部分的なサポート
IOSサファリ 8.3 完全なサポート
Opera Miniは 8 サポートしていません
ブラックベリーブラウザ 10 完全なサポート
IEモバイル 10 部分的なサポート

WebGLの利点

WebGLを使用する利点は次のとおりです-

  • JavaScript programming−WebGLアプリケーションはJavaScriptで記述されています。これらのアプリケーションを使用すると、HTMLドキュメントの他の要素と直接対話できます。他のJavaScriptライブラリ(JQueryなど)やHTMLテクノロジを使用して、WebGLアプリケーションを強化することもできます。

  • Increasing support with mobile browsers − WebGLは、iOSサファリ、Androidブラウザー、Chrome forAndroidなどのモバイルブラウザーもサポートします。

  • Open source−WebGLはオープンソースです。ライブラリのソースコードにアクセスして、ライブラリがどのように機能し、どのように開発されたかを理解できます。

  • No need for compilation− JavaScriptは、半分のプログラミングと半分のHTMLコンポーネントです。このスクリプトを実行するために、ファイルをコンパイルする必要はありません。代わりに、任意のブラウザを使用してファイルを直接開き、結果を確認できます。WebGLアプリケーションはJavaScriptを使用して開発されているため、WebGLアプリケーションもコンパイルする必要はありません。

  • Automatic memory management−JavaScriptは自動メモリ管理をサポートしています。手動でメモリを割り当てる必要はありません。WebGLはJavaScriptのこの機能を継承します。

  • Easy to set up−WebGLはHTML5に統合されているため、追加の設定は必要ありません。WebGLアプリケーションを作成するために必要なのは、テキストエディタとWebブラウザだけです。

環境設定

WebGLに別の環境を設定する必要はありません。WebGLをサポートするブラウザには、WebGL用の独自のセットアップが組み込まれています。

HTML-5は、Web上でグラフィカルアプリケーションを作成するために、2D Canvas、WebGL、SVG、3D CSS変換、SMILなどの豊富な機能セットを提供します。WebGLアプリケーションを作成するには、HTML-5の既存のcanvas要素を使用します。この章では、HTML-52Dキャンバス要素の概要を説明します。

HTML5キャンバス

HTML-5 <canvas>JavaScriptを使用してグラフィックを描画するための簡単で強力なオプションを提供します。グラフの描画、写真の構図の作成、または単純な(それほど単純ではない)アニメーションの実行に使用できます。

これは簡単です <canvas> 2つの特定の属性のみを持つ要素 width そして height さらに、id、name、classなどのすべてのコアHTML-5属性。

構文

HTMLキャンバスタグの構文を以下に示します。キャンバスの名前を二重引用符(“”)で囲む必要があります。

<canvas id = "mycanvas" width = "100" height = "100"></canvas>

キャンバス属性

キャンバスタグには、id、width、heightの3つの属性があります。

  • Id− idは、Document Object Model(DOM)のcanvas要素の識別子を表します

  • Width −幅はキャンバスの幅を表します。

  • Height −高さはキャンバスの高さを表します。

これらの属性は、キャンバスのサイズを決定します。プログラマーがcanvasタグの下でそれらを指定していない場合、Firefox、Chrome、Web Kitなどのブラウザーは、デフォルトで、サイズ300×150のcanvas要素を提供します。

例-キャンバスを作成する

次のコードは、キャンバスを作成する方法を示しています。CSSを使用して、キャンバスに色付きの境界線を付けました。

<html>
   <head>
      <style>
         #mycanvas{border:1px solid red;}
      </style>
   </head>
   <body>
      <canvas id = "mycanvas" width = "100" height = "100"></canvas>
   </body>
</html>

実行すると、上記のコードは次の出力を生成します-

レンダリングコンテキスト

<canvas>は最初は空白です。キャンバス要素に何かを表示するには、スクリプト言語を使用する必要があります。このスクリプト言語は、レンダリングコンテキストにアクセスし、それを利用する必要があります。

キャンバス要素には、と呼ばれるDOMメソッドがあります getContext()、レンダリングコンテキストとその描画関数を取得するために使用されます。このメソッドは、コンテキストのタイプという1つのパラメーターを取ります2d

次のコードは、必要なコンテキストを取得するために記述されます。このスクリプトは、以下に示すようにbodyタグ内に記述できます。

<!DOCTYPE HTML>
<html>
   <body>
      <canvas id = "mycanvas" width = "600" height = "200"></canvas>

      <script>
         var canvas = document.getElementById('mycanvas');
         var context = canvas.getContext('2d');
			
         context.font = '20pt Calibri';
         context.fillStyle = 'green';
         context.fillText('Welcome to Tutorialspoint', 70, 70);
      </script>
   </body>
</html>

実行すると、上記のコードは次の出力を生成します-

HTML-5 2D Canvasのその他の例については、次のリンクHTML- 5Canvasを確認してください。

WebGLコンテキスト

HTML5 Canvasは、WebGLアプリケーションの作成にも使用されます。キャンバス要素にWebGLレンダリングコンテキストを作成するには、文字列を渡す必要がありますexperimental-webgl、 の代わりに 2dcanvas.getContext()方法。一部のブラウザは 'のみをサポートしますwebgl'。

<!DOCTYPE html>
<html>
   <canvas id = 'my_canvas'></canvas>
	
   <script>
      var canvas = document.getElementById('my_canvas');
      var gl = canvas.getContext('experimental-webgl');
      gl.clearColor(0.9,0.9,0.8,1);
      gl.clear(gl.COLOR_BUFFER_BIT);
   </script>
</html>

実行すると、上記のコードは次の出力を生成します-

WebGLは、ほとんどが3DAPIではなく低レベルのラスタライズAPIです。WebGLを使用して画像を描画するには、画像を表すベクトルを渡す必要があります。次に、OpenGL SLを使用して、指定されたベクトルをピクセル形式に変換し、画像を画面に表示します。WebGLアプリケーションの作成には、この章で説明する一連の手順が含まれます。

WebGL-座標系

他の3Dシステムと同様に、WebGLにはx、y、z軸があります。 z 軸は意味します depth。WebGLの座標は、(1、1、1)および(-1、-1、-1)に制限されています。つまり、WebGLグラフィックスを投影する画面を立方体と見なすと、立方体の1つの角は(1、1、1)になり、反対側の角は(-1、-1、-1)になります。WebGLは、これらの境界を超えて描画されたものを表示しません。

次の図は、WebGL座標系を示しています。z軸は深さを示します。zの正の値は、オブジェクトが画面/ビューアの近くにあることを示し、zの負の値は、オブジェクトが画面から離れていることを示します。同様に、xの正の値は、オブジェクトが画面の右側にあることを示し、負の値は、オブジェクトが左側にあることを示します。同様に、yの正の値と負の値は、オブジェクトが画面の上部にあるか下部にあるかを示します。

WebGLグラフィックス

キャンバスオブジェクトのWebGLコンテキストを取得した後、JavaScriptのWebGLAPIを使用してグラフィック要素の描画を開始できます。

WebGLを開始する前に知っておく必要のある基本的な用語を次に示します。

頂点

一般に、ポリゴンなどのオブジェクトを描画するには、平面上にポイントをマークし、それらを結合して目的のポリゴンを形成します。Avertex3Dオブジェクトのエッジの結合を定義するポイントです。これは、それぞれx、y、z軸を表す3つの浮動小数点値で表されます。

次の例では、次の頂点を持つ三角形を描画しています-(0.5、0.5)、(-0.5、0.5)、(-0.5、-0.5)。

Note − JavaScript配列を使用してこれらの頂点を手動で保存し、頂点バッファーを使用してWebGLレンダリングパイプラインに渡す必要があります。

インデックス

WebGLでは、頂点を識別するために数値が使用されます。これらの数値はインデックスと呼ばれます。これらのインデックスは、WebGLでメッシュを描画するために使用されます。

Note −頂点と同様に、JavaScript配列を使用してインデックスを格納し、インデックスバッファを使用してWebGLレンダリングパイプラインに渡します。

配列

OpenGLやJoGLとは異なり、WebGLには、頂点を直接レンダリングするための事前定義されたメソッドはありません。JavaScript配列を使用して手動で保存する必要があります。

var vertices = [ 0.5, 0.5, 0.1,-0.5, 0.5,-0.5]

バッファ

バッファは、データを保持するWebGLのメモリ領域です。描画バッファ、フレームバッファ、ベテックスバッファ、インデックスバッファなど、さまざまなバッファがあります。ザ・vertex buffer そして index buffer モデルのジオメトリを記述および処理するために使用されます。

頂点バッファオブジェクトは頂点に関するデータを格納し、インデックスバッファオブジェクトはインデックスに関するデータを格納します。頂点を配列に格納した後、これらのBufferオブジェクトを使用してWegGLグラフィックパイプラインに頂点を渡します。

Frame bufferシーンデータを保持するグラフィックメモリの一部です。このバッファには、サーフェスの幅と高さ(ピクセル単位)、各ピクセルの色、深度、ステンシルバッファなどの詳細が含まれています。

メッシュ

2Dまたは3Dオブジェクトを描画するために、WebGLAPIは2つのメソッドを提供します。 drawArrays() そして drawElements()。これらの2つのメソッドは、modeこれを使用して、描画するオブジェクトを選択できます。このフィールドで提供されるオプションは、点、線、および三角形に制限されています。

これらの2つの方法を使用して3Dオブジェクトを描画するには、点、線、または三角形を使用して1つ以上のプリミティブポリゴンを作成する必要があります。その後、これらのプリミティブポリゴンを使用して、メッシュを形成できます。

プリミティブポリゴンを使用して描画された3Dオブジェクトは、 mesh。WebGLは、3Dグラフィカルオブジェクトを描画するいくつかの方法を提供しますが、ユーザーは通常、メッシュを描画することを好みます。

次の例では、2つの三角形→{1、2、3}と{4、1、3}を使用して正方形を描いたことがわかります。

シェーダープログラム

通常、メッシュの作成には三角形を使用します。WebGLはGPUアクセラレーションコンピューティングを使用するため、これらの三角形に関する情報をCPUからGPUに転送する必要があり、通信のオーバーヘッドが大きくなります。

WebGLは、通信オーバーヘッドを削減するためのソリューションを提供します。GPU上で実行されるESSL(Embedded System Shader Language)を使用するため、クライアントシステム上でグラフィカル要素を描画するために必要なすべてのプログラムを作成します。shader programs (OpenGLESシェーディング言語を使用して作成したプログラム/ GLSL)。

これらのシェーダーはGPU用のプログラムであり、シェーダープログラムの作成に使用される言語はGLSLです。これらのシェーダーでは、頂点、変換、マテリアル、ライト、およびカメラが相互作用して特定のイメージを作成する方法を正確に定義します。

つまり、メッシュのピクセルを取得するアルゴリズムを実装するスニペットです。シェーダーについては、後の章で詳しく説明します。シェーダーには、頂点シェーダーとフラグメントシェーダーの2種類があります。

頂点シェーダー

頂点シェーダーは、すべての頂点で呼び出されるプログラムコードです。これは、ジオメトリ(例:三角形)をある場所から別の場所に変換(移動)するために使用されます。頂点座標、法線、色、テクスチャ座標など、各頂点のデータ(頂点ごとのデータ)を処理します。

の中に ES GL頂点シェーダーのコードでは、プログラマーはデータを処理するための属性を定義する必要があります。これらの属性はVertex Buffer Object JavaScriptで書かれています。

次のタスクは、頂点シェーダーを使用して実行できます-

  • 頂点変換
  • 通常の変換と正規化
  • テクスチャ座標の生成
  • テクスチャ座標変換
  • Lighting
  • カラー素材の用途

フラグメントシェーダー(ピクセルシェーダー)

メッシュは複数の三角形で形成され、各三角形の表面は fragment。フラグメントシェーダーは、すべてのフラグメントのすべてのピクセルで実行されるコードです。個々のピクセルの色を計算して塗りつぶすために書かれています。

次のタスクは、フラグメントシェーダーを使用して実行できます-

  • 補間された値の操作
  • テクスチャへのアクセス
  • テクスチャアプリケーション
  • Fog
  • 色の合計

OpenGL ESSL変数

の完全な形式 OpenGL ES SLOpenGL組み込みシステムシェーディング言語です。シェーダープログラムでデータを処理するために、ESSLは3種類の変数を提供します。それらは次のとおりです-

  • Attributes−これらの変数は、頂点シェーダープログラムの入力値を保持します。属性は、頂点ごとのデータを含む頂点バッファオブジェクトを指します。頂点シェーダーが呼び出されるたびに、属性は異なる頂点のVBOを指します。

  • Uniforms −これらの変数は、ライトの位置、テクスチャ座標、色など、頂点シェーダーとフラグメントシェーダーの両方に共通の入力データを保持します。

  • Varyings −これらの変数は、頂点シェーダーからフラグメントシェーダーにデータを渡すために使用されます。

これだけの基本事項を踏まえて、グラフィックスパイプラインについて説明します。

3Dグラフィックスをレンダリングするには、一連の手順に従う必要があります。これらの手順は、graphics pipeline または rendering pipeline。次の図は、WebGLグラフィックパイプラインを示しています。

次のセクションでは、パイプラインの各ステップの役割について1つずつ説明します。

JavaScript

WebGLアプリケーションの開発中に、GPUと通信するためのシェーダー言語コードを記述します。JavaScriptは、プログラムの制御コードを記述するために使用されます。これには、次のアクションが含まれます。

  • Initialize WebGL − JavaScriptは、WebGLコンテキストを初期化するために使用されます。

  • Create arrays −ジオメトリのデータを保持するJavaScript配列を作成します。

  • Buffer objects −配列をパラメーターとして渡すことにより、バッファーオブジェクト(頂点とインデックス)を作成します。

  • Shaders − JavaScriptを使用してシェーダーを作成、コンパイル、リンクします。

  • Attributes − JavaScriptを使用して、属性を作成して有効にし、バッファオブジェクトに関連付けることができます。

  • Uniforms −JavaScriptを使用してユニフォームを関連付けることもできます。

  • Transformation matrix − JavaScriptを使用して、変換行列を作成できます。

最初に、必要なジオメトリのデータを作成し、それらをバッファーの形式でシェーダーに渡します。シェーダー言語の属性変数は、頂点シェーダーへの入力として渡されるバッファーオブジェクトを指します。

頂点シェーダー

メソッドを呼び出してレンダリングプロセスを開始するとき drawElements() そして drawArray()、頂点シェーダーは、頂点バッファオブジェクトで提供される各頂点に対して実行されます。プリミティブポリゴンの各頂点の位置を計算し、それを変化するポリゴンに格納しますgl_position。また、次のような他の属性も計算しますcolor, texture coordinates、および vertices 通常は頂点に関連付けられています。

プリミティブアセンブリ

各頂点の位置とその他の詳細を計算した後、次のフェーズは primitive assembly stage。ここで三角形が組み立てられ、ラスタライザに渡されます。

ラスタライズ

ラスタライズステップでは、プリミティブの最終画像のピクセルが決定されます。2つのステップがあります-

  • Culling−最初に、ポリゴンの方向(正面向きか背面向きか)が決定されます。ビューエリアに表示されていない不適切な方向の三角形はすべて破棄されます。このプロセスはカリングと呼ばれます。

  • Clipping−三角形の一部が表示領域の外側にある場合、表示領域の外側の部分は削除されます。このプロセスはクリッピングとして知られています。

フラグメントシェーダー

フラグメントシェーダーは

  • さまざまな変数の頂点シェーダーからのデータ、
  • ラスタライズ段階からのプリミティブ、そして
  • 頂点間の各ピクセルのカラー値を計算します。

フラグメントシェーダーは、各フラグメントのすべてのピクセルのカラー値を格納します。これらの色の値は、フラグメント操作中にアクセスできます。これについては、次に説明します。

フラグメント操作

フラグメント操作は、プリミティブの各ピクセルの色を決定した後に実行されます。これらのフラグメント操作には、次のものが含まれる場合があります。

  • Depth
  • カラーバッファブレンド
  • Dithering

すべてのフラグメントが処理されると、2D画像が形成され、画面に表示されます。ザ・frame buffer レンダリングパイプラインの最終的な宛先です。

フレームバッファ

フレームバッファは、シーンデータを保持するグラフィックメモリの一部です。このバッファには、サーフェスの幅と高さ(ピクセル単位)、各ピクセルの色、深度とステンシルバッファなどの詳細が含まれています。

WebGLとWebGLパイプライン(グラフィックアプリケーションをレンダリングするための手順)の基本について説明しました。この章では、サンプルアプリケーションを使用して、WebGLを使用して三角形を作成し、アプリケーションで実行される手順を確認します。

WebGLアプリケーションの構造

WebGLアプリケーションコードは、JavaScriptとOpenGLシェーダー言語を組み合わせたものです。

  • CPUと通信するにはJavaScriptが必要です
  • GPUと通信するには、OpenGLシェーダー言語が必要です。

サンプルアプリケーション

ここで、WebGLを使用して2D座標で単純な三角形を描画する方法を学ぶための簡単な例を見てみましょう。

<!doctype html>
<html>
   <body>
      <canvas width = "300" height = "300" id = "my_Canvas"></canvas>
		
      <script>
         /* Step1: Prepare the canvas and get WebGL context */

         var canvas = document.getElementById('my_Canvas');
         var gl = canvas.getContext('experimental-webgl');

         /* Step2: Define the geometry and store it in buffer objects */

         var vertices = [-0.5, 0.5, -0.5, -0.5, 0.0, -0.5,];

         // Create a new buffer object
         var vertex_buffer = gl.createBuffer();

         // Bind an empty array buffer to it
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         
         // Pass the vertices data to the buffer
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Unbind the buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, null);

         /* Step3: Create and compile Shader programs */

         // Vertex shader source code
         var vertCode =
            'attribute vec2 coordinates;' + 
            'void main(void) {' + ' gl_Position = vec4(coordinates,0.0, 1.0);' + '}';

         //Create a vertex shader object
         var vertShader = gl.createShader(gl.VERTEX_SHADER);

         //Attach vertex shader source code
         gl.shaderSource(vertShader, vertCode);

         //Compile the vertex shader
         gl.compileShader(vertShader);

         //Fragment shader source code
         var fragCode = 'void main(void) {' + 'gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' + '}';

         // Create fragment shader object
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);

         // Attach fragment shader source code
         gl.shaderSource(fragShader, fragCode);

         // Compile the fragment shader
         gl.compileShader(fragShader);

         // Create a shader program object to store combined shader program
         var shaderProgram = gl.createProgram();

         // Attach a vertex shader
         gl.attachShader(shaderProgram, vertShader); 
         
         // Attach a fragment shader
         gl.attachShader(shaderProgram, fragShader);

         // Link both programs
         gl.linkProgram(shaderProgram);

         // Use the combined shader program object
         gl.useProgram(shaderProgram);

         /* Step 4: Associate the shader programs to buffer objects */

         //Bind vertex buffer object
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         //Get the attribute location
         var coord = gl.getAttribLocation(shaderProgram, "coordinates");

         //point an attribute to the currently bound VBO
         gl.vertexAttribPointer(coord, 2, gl.FLOAT, false, 0, 0);

         //Enable the attribute
         gl.enableVertexAttribArray(coord);

         /* Step5: Drawing the required object (triangle) */

         // Clear the canvas
         gl.clearColor(0.5, 0.5, 0.5, 0.9);

         // Enable the depth test
         gl.enable(gl.DEPTH_TEST); 
         
         // Clear the color buffer bit
         gl.clear(gl.COLOR_BUFFER_BIT);

         // Set the view port
         gl.viewport(0,0,canvas.width,canvas.height);

         // Draw the triangle
         gl.drawArrays(gl.TRIANGLES, 0, 3);
      </script>
   </body>
</html>

次の結果が得られます-

上記のプログラムを注意深く観察すると、WebGLを使用して単純な三角形を描くための5つの連続した手順に従いました。手順は次のとおりです-

Step 1 − Prepare the canvas and get WebGL rendering context

現在のHTMLキャンバスオブジェクトを取得し、そのWebGLレンダリングコンテキストを取得します。

Step 2 − Define the geometry and store it in buffer objects

頂点、インデックス、色などのジオメトリの属性を定義し、JavaScript配列に格納します。次に、1つ以上のバッファオブジェクトを作成し、データを含む配列をそれぞれのバッファオブジェクトに渡します。この例では、三角形の頂点をJavaScript配列に格納し、この配列を頂点バッファーオブジェクトに渡します。

Step 3 − Create and compile Shader programs

頂点シェーダーとフラグメントシェーダープログラムを作成してコンパイルし、これら2つのプログラムをリンクして結合プログラムを作成します。

Step 4 − Associate the shader programs with buffer objects

バッファオブジェクトと結合されたシェーダープログラムを関連付けます。

Step 5 − Drawing the required object (triangle)

このステップには、色のクリア、バッファビットのクリア、深度テストの有効化、ビューポートの設定などの操作が含まれます。最後に、次のいずれかの方法を使用して必要なプリミティブを描画する必要があります。 drawArrays() または drawElements()

これらの手順はすべて、このチュートリアルでさらに説明されています。

WebGLアプリケーションを作成するための最初のステップは、WebGLレンダリングコンテキストオブジェクトを取得することです。このオブジェクトはWebGL描画バッファと相互作用し、すべてのWebGLメソッドを呼び出すことができます。WebGLコンテキストを取得するには、次の操作を実行します。

  • HTML-5キャンバスを作成する
  • キャンバスIDを取得する
  • WebGLを取得する

HTML-5Canvas要素の作成

第5章では、HTML-5キャンバス要素を作成する方法について説明しました。HTML-5ドキュメントの本文内で、キャンバスを作成して名前を付け、パラメーターとして属性IDに渡します。幅と高さの属性を使用して、キャンバスの寸法を定義できます(オプション)。

次の例は、500×500の寸法のキャンバス要素を作成する方法を示しています。可視性のためにCSSを使用してキャンバスへの境界線を作成しました。次のコードをコピーして、名前の付いたファイルに貼り付けますmy_canvas.html

<!DOCTYPE HTML>
<html>
   <head>
      <style>
         #mycanvas{border:1px solid blue;}
      </style>
   </head>
   <body>
      <canvas id = "mycanvas" width = "300" height = "300"></canvas>
   </body>
</html>

次の結果が得られます-

キャンバスIDを取得する

キャンバスを作成したら、WebGLコンテキストを取得する必要があります。WebGL描画コンテキストを取得するために最初に行うことは、現在のキャンバス要素のIDを取得することです。

Canvas IDは、DOM(Document Object Model)メソッドを呼び出すことによって取得されます getElementById()。このメソッドはパラメータとして文字列値を受け入れるため、現在のキャンバスの名前を渡します。

たとえば、キャンバス名が my_canvas、次にキャンバスIDが次のように取得されます-

var canvas = document.getElementById('my_Canvas');

WebGL描画コンテキストを取得する

WebGLRenderingContextオブジェクト(またはWebGL描画コンテキストオブジェクトまたは単にWebGLコンテキスト)を取得するには、 getContext() 現在の方法 HTMLCanvasElement。getContext()の構文は次のとおりです-

canvas.getContext(contextType, contextAttributes);

文字列を渡す webgl または experimental-webgl として contentType。ザ・contextAttributesパラメータはオプションです。(この手順を進めるときは、ブラウザーがWebGLバージョン1(OpenGL ES 2.0)を実装していることを確認してください)。

次のコードスニペットは、WebGLレンダリングコンテキストを取得する方法を示しています。ここにgl 取得したコンテキストオブジェクトへの参照変数です。

var canvas = document.getElementById('my_Canvas');
var gl = canvas.getContext('experimental-webgl');

WebGLContextAttributes

パラメータ WebGLContextAttributes必須ではありません。このパラメーターは、以下にリストされているブール値を受け入れるさまざまなオプションを提供します-

シニア番号 属性と説明
1

Alpha

その値がtrueの場合、キャンバスにアルファバッファを提供します。

デフォルトでは、その値はtrueです。

2

depth

その値がtrueの場合、少なくとも16ビットの深度バッファーを含む描画バッファーを取得します。

デフォルトでは、その値はtrueです。

3

stencil

その値がtrueの場合、少なくとも8ビットのステンシルバッファーを含む描画バッファーを取得します。

デフォルトでは、その値はfalseです。

4

antialias

その値がtrueの場合、アンチエイリアシングを実行する描画バッファーを取得します。

デフォルトでは、その値はtrueです。

5

premultipliedAlpha

その値がtrueの場合、アルファが事前に乗算された色を含む描画バッファーを取得します。

デフォルトでは、その値はtrueです。

6

preserveDrawingBuffer

その値がtrueの場合、バッファーはクリアされず、作成者によってクリアまたは上書きされるまでその値が保持されます。

デフォルトでは、その値はfalseです。

次のコードスニペットは、ステンシルバッファを使用してWebGLコンテキストを作成する方法を示しています。 anti-aliasing

var canvas = document.getElementById('canvas1');
var context = canvas.getContext('webgl', { antialias: false, stencil: true });

WebGLRenderingContextの作成時に、描画バッファーが作成されます。Contextオブジェクトは、OpenGLの状態を管理し、描画バッファにレンダリングします。

WebGLRenderingContext

これは、WebGLの主要なインターフェイスです。これは、WebGL描画コンテキストを表します。このインターフェイスには、図面バッファでさまざまなタスクを実行するために使用されるすべてのメソッドが含まれています。このインターフェースの属性を次の表に示します。

シニア番号 属性と説明
1

Canvas

これは、このコンテキストを作成したcanvas要素への参照です。

2

drawingBufferWidth

この属性は、描画バッファの実際の幅を表します。HTMLCanvasElementのwidth属性とは異なる場合があります。

3

drawingBufferHeight

この属性は、描画バッファの実際の高さを表します。HTMLCanvasElementのheight属性とは異なる場合があります。

WebGLコンテキストを取得したら、プリミティブ(描画するオブジェクト)のジオメトリを定義して保存する必要があります。WebGLでは、JavaScript配列を使用して、ジオメトリの詳細(頂点、インデックス、プリミティブの色など)を定義します。これらの詳細をシェーダープログラムに渡すには、バッファーオブジェクトを作成し、データを含むJavaScript配列をそれぞれのバッファーに格納(アタッチ)する必要があります。

Note: 後で、これらのバッファオブジェクトは、シェーダープログラム(頂点シェーダー)の属性に関連付けられます。

必要な形状の定義

頂点を使用して描画された2Dまたは3Dモデルは、 mesh。メッシュの各ファセットは、polygon ポリゴンは3つ以上の頂点で構成されています。

WebGLレンダリングコンテキストでモデルを描画するには、JavaScript配列を使用して頂点とインデックスを定義する必要があります。たとえば、図に示すように、座標{(5,5)、(-5,5)、(-5、-5)}にある三角形を作成する場合は、次の配列を作成できます。 -としての頂点

var vertices = [
   0.5,0.5,    //Vertex 1
   0.5,-0.5,   //Vertex 2
   -0.5,-0.5,  //Vertex 3
];

同様に、インデックスの配列を作成できます。上記の三角形のインデックスのインデックスは[0、1、2]になり、次のように定義できます。

var indices = [ 0,1,2 ]

インデックスをよりよく理解するために、正方形のようなより複雑なモデルを検討してください。正方形を2つの三角形のセットとして表すことができます。(0,3,1)と(3,1,2)が正方形を描くために使用する2つの三角形である場合、インデックスは次のように定義されます。

var indices = [0,3,1,3,1,2];

Note

プリミティブを描画するために、WebGLは次の2つの方法を提供します-

  • drawArrays() −このメソッドを使用している間、JavaScript配列を使用してプリミティブの頂点を渡します。

  • drawElements() −このメソッドを使用している間、JavaScript配列を使用してプリミティブの頂点とインデックスの両方を渡します。

バッファオブジェクト

バッファオブジェクトは、システムに割り当てられたメモリ領域を示すWebGLによって提供されるメカニズムです。これらのバッファオブジェクトには、頂点、インデックス、色などに対応する、描画するモデルのデータを格納できます。

これらのバッファオブジェクトを使用すると、属性変数の1つを介して複数のデータをシェーダープログラム(頂点シェーダー)に渡すことができます。これらのバッファオブジェクトはGPUメモリに存在するため、直接レンダリングでき、パフォーマンスが向上します。

ジオメトリを処理するには、2種類のバッファオブジェクトがあります。彼らは-

  • Vertex buffer object (VBO)−レンダリングされるグラフィカルモデルの頂点ごとのデータを保持します。WebGLの頂点バッファオブジェクトを使用して、頂点座標、法線、色、テクスチャ座標などの頂点に関するデータを保存および処理します。

  • Index buffer objects (IBO) −レンダリングされるグラフィカルモデルのインデックス(インデックスデータ)を保持します。

必要なジオメトリを定義してJavaScript配列に格納した後、これらの配列をバッファオブジェクトに渡す必要があります。ここから、データがシェーダープログラムに渡されます。データをバッファに保存するには、次の手順に従います。

  • 空のバッファを作成します。

  • 適切な配列オブジェクトを空のバッファにバインドします。

  • データ(頂点/インデックス)のいずれかを使用してバッファに渡します typed arrays

  • バッファーのバインドを解除します(オプション)。

バッファの作成

空のバッファオブジェクトを作成するために、WebGLはと呼ばれるメソッドを提供します createBuffer()。作成が成功した場合、このメソッドは新しく作成されたバッファオブジェクトを返します。それ以外の場合は、失敗した場合にnull値を返します。

WebGLはステートマシンとして動作します。バッファーが作成されると、それ以降のバッファー操作は、バインドを解除するまで現在のバッファーで実行されます。次のコードを使用してバッファを作成します-

var vertex_buffer = gl.createBuffer();

Notegl 現在のWebGLコンテキストへの参照変数です。

バッファをバインドします

空のバッファオブジェクトを作成したら、適切な配列バッファ(ターゲット)をそれにバインドする必要があります。WebGLはと呼ばれるメソッドを提供しますbindBuffer() この目的のために。

構文

の構文 bindBuffer() 方法は次のとおりです-

void bindBuffer (enum target, Object buffer)

このメソッドは2つのパラメーターを受け入れ、それらについて以下で説明します。

target−最初の変数は、空のバッファーにバインドするバッファーのタイプを表す列挙値です。このパラメーターのオプションとして、2つの事前定義された列挙値があります。彼らは-

  • ARRAY_BUFFER これは頂点データを表します。

  • ELEMENT_ARRAY_BUFFER これはインデックスデータを表します。

Object buffer− 2つ目は、前の手順で作成したバッファオブジェクトへの参照変数です。参照変数は、頂点バッファオブジェクトまたはインデックスバッファオブジェクトにすることができます。

次のコードスニペットは、bindBuffer()メソッドの使用方法を示しています。

//vertex buffer
var vertex_buffer = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

//Index buffer
var Index_Buffer = gl.createBuffer();
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);

バッファへのデータの受け渡し

次のステップは、データ(頂点/インデックス)をバッファーに渡すことです。これまで、データは配列の形式であり、バッファーに渡す前に、WebGL型付き配列の1つでラップする必要があります。WebGLはという名前のメソッドを提供しますbufferData() この目的のために。

構文

bufferData()メソッドの構文は次のとおりです-

void bufferData (enum target, Object data, enum usage)

このメソッドは3つのパラメーターを受け入れ、それらについて以下で説明します。

target −最初のパラメータは、使用した配列バッファのタイプを表す列挙値です。このパラメータのオプションは次のとおりです。

  • ARRAY_BUFFER これは vertex data

  • ELEMENT_ARRAY_BUFFER これは index data

Object data− 2番目のパラメータは、バッファオブジェクトに書き込まれるデータを含むオブジェクト値です。ここでは、を使用してデータを渡す必要がありますtyped arrays

Usage−このメソッドの3番目のパラメーターは、バッファーオブジェクトデータ(保存されたデータ)を使用して形状を描画する方法を指定する列挙型変数です。このパラメータには、以下に示す3つのオプションがあります。

  • gl.STATIC_DRAW −データは一度指定され、何度も使用されます。

  • gl.STREAM_DRAW −データは1回指定され、数回使用されます。

  • gl.DYNAMIC_DRAW −データは繰り返し指定され、何度も使用されます。

次のコードスニペットは、 bufferData()方法。頂点とインデックスは、それぞれ頂点とインデックスのデータを保持する配列であると想定します。

//vertex buffer
gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

//Index buffer
gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);

型付き配列

WebGLは、と呼ばれる特別なタイプの配列を提供します typed arraysインデックス頂点やテクスチャなどのデータ要素を転送します。これらの型付き配列は、大量のデータを格納し、ネイティブバイナリ形式で処理するため、パフォーマンスが向上します。WebGLで使用される型付き配列は、Int8Array、Uint8Array、Int16Array、Uint16Array、Int32Array、UInt32Array、Float32Array、およびFloat64Arrayです。

Note

  • 一般的に、頂点データの保存には、 Float32Array; インデックスデータを保存するには、Uint16Array

  • を使用してJavaScript配列と同じように型付き配列を作成できます new キーワード。

バッファのバインドを解除します

使用後は、バッファーのバインドを解除することをお勧めします。これは、以下に示すように、バッファオブジェクトの代わりにnull値を渡すことで実行できます。

gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, null);

WebGLは、バッファ操作を実行するために次のメソッドを提供します-

シニア番号 メソッドと説明
1

ボイド bindBuffer(列挙型ターゲット、オブジェクトバッファ

ターゲット-ARRAY_BUFFER、ELEMENT_ARRAY_BUFFER

2

ボイド bufferData(列挙型ターゲット、長いサイズ、列挙型の使用法

ターゲット-ARRAY_BUFFER、ELEMENT_ARRAY_BUFFER

使用法-STATIC_DRAW、STREAM_DRAW、DYNAMIC_DRAW

3

ボイド bufferData(列挙型ターゲット、オブジェクトデータ、列挙型使用法

ターゲット使用法-と同じbufferData 上記

4

ボイド bufferSubData(列挙型ターゲット、長いオフセット、オブジェクトデータ

ターゲット-ARRAY_BUFFER、ELEMENT_ARRAY_BUFFER

5 オブジェクト createBuffer()
6 ボイド deleteBuffer(オブジェクトバッファ
7

どれか getBufferParameter(列挙型ターゲット、列挙型pname

ターゲット-ARRAY_BUFFER、ELEMENT_ ARRAY_BUFFER

pname − BUFFER_SIZE、BUFFER_USAGE

8 ブール isBuffer(オブジェクトバッファ

シェーダーはGPUで実行されるプログラムです。シェーダーはOpenGLESシェーダー言語(ES SLとして知られています)で書かれています。ES SLには、独自の変数、データ型、修飾子、組み込みの入力および出力があります。

データ型

次の表に、OpenGL ESSLが提供する基本的なデータ型を示します。

シニア番号 タイプと説明
1

void

空の値を表します。

2

bool

trueまたはfalseを受け入れます。

3

int

これは符号付き整数データ型です。

4

float

これは浮動スカラーデータ型です。

5

vec2, vec3, vec4

n成分浮動小数点ベクトル

6

bvec2, bvec3, bvec4

ブールベクトル

7

ivec2, ivec3, ivec4

符号付き整数ベクトル

8

mat2, mat3, mat4

2x2、3x3、4x4フロートマトリックス

9

sampler2D

2Dテクスチャにアクセスする

10

samplerCube

キューブマップされたテクスチャにアクセスする

修飾子

OpenGL ESSLには3つの主要な修飾子があります-

シニア番号 修飾子と説明
1

attribute

この修飾子は、頂点シェーダーと頂点ごとのデータのOpenGLESの間のリンクとして機能します。この属性の値は、頂点シェーダーを実行するたびに変更されます。

2

uniform

この修飾子は、シェーダープログラムとWebGLアプリケーションをリンクします。属性修飾子とは異なり、ユニフォームの値は変更されません。ユニフォームは読み取り専用です。それらを任意の基本データ型で使用して、変数を宣言できます。

Example −均一 vec4 lightPosition;

3

varying

この修飾子は、補間されたデータの頂点シェーダーとフラグメントシェーダーの間のリンクを形成します。次のデータ型で使用できます-float、vec2、vec3、vec4、mat2、mat3、mat4、または配列。

Example −変化する vec3 正常;

頂点シェーダー

頂点シェーダーは、すべての頂点で呼び出されるプログラムコードです。ジオメトリ(例:三角形)をある場所から別の場所に変換(移動)します。頂点座標、法線、色、テクスチャ座標など、各頂点のデータ(頂点ごとのデータ)を処理します。

頂点シェーダーのESGLコードでは、プログラマーはデータを処理するための属性を定義する必要があります。これらの属性は、JavaScriptで記述された頂点バッファオブジェクトを指します。次のタスクは、頂点シェーダーと頂点変換を使用して実行できます。

  • 頂点変換
  • 通常の変換と正規化
  • テクスチャ座標の生成
  • テクスチャ座標変換
  • Lighting
  • カラー素材の用途

事前定義された変数

OpenGL ES SLは、頂点シェーダー用に次の定義済み変数を提供します-

シニア番号 変数と説明
1

highp vec4 gl_Position;

頂点の位置を保持します。

2

mediump float gl_PointSize;

変換されたポイントサイズを保持します。この変数の単位はピクセルです。

サンプルコード

頂点シェーダーの次のサンプルコードを見てください。三角形の頂点を処理します。

attribute vec2 coordinates;

void main(void) {
   gl_Position = vec4(coordinates, 0.0, 1.0);
};

上記のコードを注意深く観察すると、次の名前の属性変数が宣言されています。 coordinates。(この変数は、メソッドを使用して頂点バッファオブジェクトに関連付けられますgetAttribLocation()。属性coordinates シェーダープログラムオブジェクトとともに、このメソッドにパラメーターとして渡されます。)

指定された頂点シェーダープログラムの2番目のステップでは、 gl_position 変数が定義されています。

gl_Position

gl_Positionは、頂点シェーダープログラムでのみ使用できる事前定義された変数です。頂点の位置が含まれています。上記のコードでは、coordinates属性はベクトルの形式で渡されます。頂点シェーダーは頂点ごとの操作であるため、gl_position値は頂点ごとに計算されます。

その後、gl_position値は、プリミティブのアセンブリ、クリッピング、カリング、および頂点処理が終了した後にプリミティブを操作するその他の固定機能操作によって使用されます。

頂点シェーダーのすべての可能な操作のための頂点シェーダープログラムを作成できます。これについては、このチュートリアルで個別に説明します。

フラグメントシェーダー

A mesh は複数の三角形で形成され、各三角形の表面は fragment。フラグメントシェーダーは、各フラグメントのすべてのピクセルで実行されるコードです。これは、個々のピクセルの色を計算して塗りつぶすために作成されています。次のタスクは、フラグメントシェーダーを使用して実行できます-

  • 補間された値の操作
  • テクスチャへのアクセス
  • テクスチャアプリケーション
  • Fog
  • 色の合計

事前定義された変数

OpenGL ES SLは、フラグメントシェーダー用に次の定義済み変数を提供します-

シニア番号 変数と説明
1

中程度のvec4 gl_FragCoord;

フレームバッファ内のフラグメント位置を保持します。

2

bool gl_FrontFacing;

正面向きのプリミティブに属するフラグメントを保持します。

3

mediump vec2 gl_PointCoord;

ポイント内のフラグメント位置を保持します(ポイントラスタライズのみ)。

4

mediump vec4 gl_FragColor;

シェーダーの出力フラグメントカラー値を保持します

5

mediump vec4 gl_FragData[n]

カラーアタッチメントのフラグメントカラーを保持します n

サンプルコード

次のフラグメントシェーダーのサンプルコードは、三角形のすべてのピクセルに色を適用する方法を示しています。

void main(void) {
   gl_FragColor = vec4(0, 0.8, 0, 1);
}

上記のコードでは、 color値は変数に格納されますgl.FragColor。フラグメントシェーダープログラムは、固定関数変数を使用して出力をパイプラインに渡します。FragColorはその1つです。この変数は、モデルのピクセルのカラー値を保持します。

シェーダープログラムの保存とコンパイル

シェーダーは独立したプログラムであるため、別のスクリプトとして記述してアプリケーションで使用できます。または、直接保存することもできますstring 以下に示すフォーマット。

var vertCode =
   'attribute vec2 coordinates;' +
	
   'void main(void) {' +
      ' gl_Position = vec4(coordinates, 0.0, 1.0);' +
   '}';

シェーダーのコンパイル

コンパイルには次の3つのステップが含まれます-

  • シェーダーオブジェクトの作成
  • 作成したシェーダーオブジェクトにソースコードを添付する
  • プログラムのコンパイル

頂点シェーダーの作成

空のシェーダーを作成するために、WebGLはと呼ばれるメソッドを提供します createShader()。シェーダーオブジェクトを作成して返します。その構文は次のとおりです-

Object createShader (enum type)

構文に見られるように、このメソッドは事前定義された列挙値をパラメーターとして受け入れます。これには2つのオプションがあります-

  • gl.VERTEX_SHADER 頂点シェーダーを作成するため

  • gl.FRAGMENT_SHADER フラグメントシェーダーを作成するため。

ソースをシェーダーにアタッチする

メソッドを使用して、作成したシェーダーオブジェクトにソースコードを添付できます。 shaderSource()。その構文は次のとおりです-

void shaderSource(Object shader, string source)

このメソッドは2つのパラメーターを受け入れます-

  • shader −作成したシェーダーオブジェクトを1つのパラメーターとして渡す必要があります。

  • Source −シェーダープログラムコードを文字列形式で渡す必要があります。

プログラムのコンパイル

プログラムをコンパイルするには、メソッドを使用する必要があります compileShader()。その構文は次のとおりです-

compileShader(Object shader)

このメソッドは、シェーダープログラムオブジェクトをパラメーターとして受け入れます。シェーダープログラムオブジェクトを作成したら、それにソースコードを添付して、そのオブジェクトをこのメソッドに渡します。

次のコードスニペットは、頂点シェーダーとフラグメントシェーダーを作成およびコンパイルして三角形を作成する方法を示しています。

// Vertex Shader
var vertCode =
   'attribute vec3 coordinates;' +
	
   'void main(void) {' +
      ' gl_Position = vec4(coordinates, 1.0);' +
   '}';

var vertShader = gl.createShader(gl.VERTEX_SHADER);
gl.shaderSource(vertShader, vertCode);
gl.compileShader(vertShader);
 
// Fragment Shader
var fragCode =
   'void main(void) {' +
      ' gl_FragColor = vec4(0, 0.8, 0, 1);' +
   '}';

var fragShader = gl.createShader(gl.FRAGMENT_SHADER);
gl.shaderSource(fragShader, fragCode);
gl.compileShader(fragShader);

複合プログラム

両方のシェーダープログラムを作成してコンパイルした後、両方のシェーダー(頂点とフラグメント)を含む結合プログラムを作成する必要があります。次の手順に従う必要があります-

  • プログラムオブジェクトを作成する
  • 両方のシェーダーを取り付けます
  • 両方のシェーダーをリンクします
  • プログラムを使用する

プログラムオブジェクトを作成する

メソッドを使用してプログラムオブジェクトを作成します createProgram()。空のプログラムオブジェクトを返します。これがその構文です-

createProgram();

シェーダーを取り付ける

メソッドを使用して、作成したプログラムオブジェクトにシェーダーをアタッチします attachShader()。その構文は次のとおりです-

attachShader(Object program, Object shader);

このメソッドは2つのパラメーターを受け入れます-

  • Program −作成した空のプログラムオブジェクトを1つのパラメータとして渡します。

  • Shader −コンパイルされたシェーダープログラム(頂点シェーダー、フラグメントシェーダー)の1つを渡す

Note −この方法を使用して両方のシェーダーをアタッチする必要があります。

シェーダーをリンクする

メソッドを使用してシェーダーをリンクします linkProgram()、シェーダーをアタッチしたプログラムオブジェクトを渡すことによって。その構文は次のとおりです-

linkProgram(shaderProgram);

プログラムを使用する

WebGLはと呼ばれるメソッドを提供します useProgram()。リンクされたプログラムを渡す必要があります。その構文は次のとおりです-

useProgram(shaderProgram);

次のコードスニペットは、結合されたシェーダープログラムを作成、リンク、および使用する方法を示しています。

var shaderProgram = gl.createProgram();
gl.attachShader(shaderProgram, vertShader);
gl.attachShader(shaderProgram, fragShader);
gl.linkProgram(shaderProgram);
gl.useProgram(shaderProgram);

頂点シェーダープログラムの各属性は、頂点バッファオブジェクトを指します。頂点バッファオブジェクトを作成した後、プログラマはそれらを頂点シェーダープログラムの属性に関連付ける必要があります。各属性は、データ値を抽出する1つの頂点バッファーオブジェクトのみを指し、これらの属性はシェーダープログラムに渡されます。

頂点バッファオブジェクトを頂点シェーダープログラムの属性に関連付けるには、以下の手順に従う必要があります。

  • 属性の場所を取得する
  • 属性を頂点バッファオブジェクトにポイントします
  • 属性を有効にする

属性の場所を取得する

WebGLはと呼ばれるメソッドを提供します getAttribLocation()これは属性の場所を返します。その構文は次のとおりです-

ulong getAttribLocation(Object program, string name)

このメソッドは、頂点シェーダープログラムオブジェクトと頂点シェーダープログラムの属性値を受け入れます。

次のコードスニペットは、このメソッドの使用方法を示しています。

var coordinatesVar = gl.getAttribLocation(shader_program, "coordinates");

ここに、 shader_program シェーダープログラムのオブジェクトであり、 coordinates 頂点シェーダープログラムの属性です。

属性をVBOにポイントする

バッファオブジェクトを属性変数に割り当てるために、WebGLはと呼ばれるメソッドを提供します vertexAttribPointer()。このメソッドの構文は次のとおりです-

void vertexAttribPointer(location, int size, enum type, bool normalized, long stride, long offset)

このメソッドは6つのパラメーターを受け入れ、それらについて以下で説明します。

  • Location−属性変数の保存場所を指定します。このオプションでは、によって返される値を渡す必要がありますgetAttribLocation() 方法。

  • Size −バッファオブジェクトの頂点ごとのコンポーネント数を指定します。

  • Type −データの種類を指定します。

  • Normalized−これはブール値です。trueの場合、非浮動データは[0、1]に正規化されます。それ以外の場合は、[-1、1]に正規化されます。

  • Stride −異なる頂点データ要素間のバイト数を指定するか、デフォルトのストライドの場合はゼロを指定します。

  • Offset−頂点データの保存元のバイトを示すために、バッファオブジェクトのオフセット(バイト単位)を指定します。データが最初から保存されている場合、オフセットは0です。

次のスニペットは、使用方法を示しています vertexAttribPointer() プログラム内-

gl.vertexAttribPointer(coordinatesVar, 3, gl.FLOAT, false, 0, 0);

属性の有効化

頂点シェーダー属性をアクティブにして、頂点シェーダーのバッファーオブジェクトにアクセスします。この操作のために、WebGLはenableVertexAttribArray()方法。このメソッドは、属性の場所をパラメーターとして受け入れます。プログラムでこのメソッドを使用する方法は次のとおりです-

gl.enableVertexAttribArray(coordinatesVar);

バッファーをシェーダーに関連付けた後、最後のステップは必要なプリミティブを描画することです。WebGLには、次の2つの方法があります。drawArrays() そして drawElements() モデルを描画します。

drawArrays()

drawArrays()は、頂点を使用してモデルを描画するために使用される方法です。これがその構文です-

void drawArrays(enum mode, int first, long count)

このメソッドは、次の3つのパラメーターを取ります-

  • mode− WebGLでは、モデルはプリミティブ型を使用して描画されます。モードを使用して、プログラマーはWebGLによって提供されるプリミティブ型の1つを選択する必要があります。このオプションに指定できる値は、-gl.POINTS、gl.LINE_STRIP、gl.LINE_LOOP、gl.LINES、gl.TRIANGLE_STRIP、gl.TRIANGLE_FAN、およびgl.TRIANGLESです。

  • first−このオプションは、有効な配列の開始要素を指定します。負の値にすることはできません。

  • count −このオプションは、レンダリングされる要素の数を指定します。

を使用してモデルを描画する場合 drawArrays() メソッド、次にWebGLは、形状をレンダリングするときに、頂点座標が定義されている順序でジオメトリを作成します。

を使用して単一の三角形を描画する場合 drawArray() メソッドの場合、3つの頂点を渡して、 drawArrays() 以下に示すように、メソッド。

var vertices = [-0.5,-0.5, -0.25,0.5, 0.0,-0.5,];
gl.drawArrays(gl.TRIANGLES, 0, 3);

以下に示すような三角形が生成されます。

連続する三角形を描画する場合、頂点バッファに次の3つの頂点を順番に渡し、レンダリングする要素の数を6として指定する必要があります。

var vertices = [-0.5,-0.5, -0.25,0.5, 0.0,-0.5, 0.0,-0.5, 0.25,0.5, 0.5,-0.5,];
gl.drawArrays(gl.TRIANGLES, 0, 6);

以下に示すように、連続した三角形が生成されます。

drawElements()

drawElements()は、頂点とインデックスを使用してモデルを描画するために使用される方法です。その構文は次のとおりです-

void drawElements(enum mode, long count, enum type, long offset)

このメソッドは、次の4つのパラメーターを取ります-

  • mode− WebGLモデルは、プリミティブ型を使用して描画されます。モードを使用して、プログラマーはWebGLによって提供されるプリミティブ型の1つを選択する必要があります。このオプションで可能な値のリストは、-gl.POINTS、gl.LINE_STRIP、gl.LINE_LOOP、gl.LINES、gl.TRIANGLE_STRIP、gl.TRIANGLE_FAN、およびgl.TRIANGLESです。

  • count −このオプションは、レンダリングされる要素の数を指定します。

  • type −このオプションは、UNSIGNED_BYTEまたはUNSIGNED_SHORTでなければならないインデックスのデータ型を指定します。

  • offset−このオプションは、レンダリングの開始点を指定します。通常、これは最初の要素(0)です。

を使用してモデルを描画する場合 drawElements()メソッドの場合、頂点バッファオブジェクトとともにインデックスバッファオブジェクトも作成する必要があります。この方法を使用すると、頂点データは1回処理され、インデックスに記載されている回数だけ使用されます。

インデックスを使用して単一の三角形を描画する場合は、頂点とともにインデックスを渡し、を呼び出す必要があります。 drawElements() 以下に示す方法。

var vertices = [ -0.5,-0.5,0.0, -0.25,0.5,0.0, 0.0,-0.5,0.0 ];
var indices = [0,1,2];

gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);

次の出力が生成されます-

を使用して伝染性の三角形を描きたい場合 drawElements() メソッドでは、他の頂点を追加し、残りの頂点のインデックスに言及するだけです。

var vertices = [
   -0.5,-0.5,0.0,
   -0.25,0.5,0.0,
   0.0,-0.5,0.0,
   0.25,0.5,0.0,
   0.5,-0.5,0.0 
];

var indices = [0,1,2,2,3,4];

gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);

次の出力が生成されます-

必要な操作

プリミティブを描画する前に、以下で説明するいくつかの操作を実行する必要があります。

キャンバスをクリアする

まず、を使用してキャンバスをクリアする必要があります clearColor()方法。目的の色のRGBA値をパラメーターとしてこのメ​​ソッドに渡すことができます。次に、WebGLはキャンバスをクリアし、指定された色で塗りつぶします。したがって、この方法を使用して背景色を設定できます。

次の例を見てください。ここでは、灰色のRGBA値を渡します。

gl.clearColor(0.5, 0.5, .5, 1);

深度テストを有効にする

を使用して深度テストを有効にします enable() 以下に示すように、メソッド。

gl.enable(gl.DEPTH_TEST);

カラーバッファビットをクリアする

を使用して、色と深度バッファをクリアします。 clear() 以下に示すように、メソッド。

gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);

ビューポートを設定する

ビューポートは、描画バッファのレンダリング結果を含む長方形の表示可能領域を表します。ビューポートの寸法は、を使用して設定できます。viewport()方法。次のコードでは、ビューポートの寸法がキャンバスの寸法に設定されています。

gl.viewport(0,0,canvas.width,canvas.height);

プリミティブを描画するための段階的なプロセスに従う方法については、前に(第5章で)説明しました。このプロセスを5つのステップで説明しました。新しい形状を描くたびに、これらの手順を繰り返す必要があります。この章では、WebGLで3D座標を使用して点を描画する方法について説明します。先に進む前に、5つのステップをもう一度見てみましょう。

必要な手順

ポイントを描画するWebGLアプリケーションを作成するには、次の手順が必要です。

Step 1 − Prepare the Canvas and Get the WebGL Rendering Context

このステップでは、メソッドを使用してWebGLレンダリングコンテキストオブジェクトを取得します getContext().

Step 2 − Define the Geometry and Store it in the Buffer Objects

3つのポイントを描画しているので、3D座標で3つの頂点を定義し、それらをバッファに格納します。

var vertices = [
   -0.5,0.5,0.0,
   0.0,0.5,0.0,
   -0.25,0.25,0.0, 
];

Step 3 − Create and Compile the Shader Programs

このステップでは、頂点シェーダーとフラグメントシェーダーのプログラムを作成してコンパイルし、これら2つのプログラムをリンクして結合プログラムを作成する必要があります。

  • Vertex Shader −与えられた例の頂点シェーダーでは、3D座標を格納するためのベクトル属性を定義し、それをに割り当てます。 gl_position 変数。

  • gl_pointsizeポイントにサイズを割り当てるために使用される変数です。ポイントサイズを10として割り当てます。

var vertCode = 'attribute vec3 coordinates;' +

   'void main(void) {' +
      ' gl_Position = vec4(coordinates, 1.0);' +
      'gl_PointSize = 10.0;'+
   '}';
  • Fragment Shader −フラグメントシェーダーでは、フラグメントの色をに割り当てるだけです。 gl_FragColor 変数

var fragCode = 'void main(void) {' +' gl_FragColor = vec4(1, 0.5, 0.0, 1);' +'}';

Step 4 − Associate the Shader Programs to Buffer Objects

このステップでは、バッファオブジェクトをシェーダープログラムに関連付けます。

Step 5 − Drawing the Required Object

この方法を使用します drawArrays()ポイントを描画します。描きたい点数が3点なので、カウント値は3点です。

gl.drawArrays(gl.POINTS, 0, 3)

例–WebGLを使用して3つのポイントを描画します

これが3つのポイントを描く完全なWebGLプログラムです-

<!doctype html>
<html>
   <body>
      <canvas width = "570" height = "570" id = "my_Canvas"></canvas>

      <script>
         /*================Creating a canvas=================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl'); 

         /*==========Defining and storing the geometry=======*/

         var vertices = [
            -0.5,0.5,0.0,
            0.0,0.5,0.0,
            -0.25,0.25,0.0, 
         ];

         // Create an empty buffer object to store the vertex buffer
         var vertex_buffer = gl.createBuffer();

         //Bind appropriate array buffer to it
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Pass the vertex data to the buffer
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Unbind the buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, null);

         /*=========================Shaders========================*/

         // vertex shader source code
         var vertCode =
            'attribute vec3 coordinates;' +

            'void main(void) {' +
               ' gl_Position = vec4(coordinates, 1.0);' +
               'gl_PointSize = 10.0;'+
            '}';

         // Create a vertex shader object
         var vertShader = gl.createShader(gl.VERTEX_SHADER);
         
         // Attach vertex shader source code
         gl.shaderSource(vertShader, vertCode);

         // Compile the vertex shader
         gl.compileShader(vertShader);

         // fragment shader source code
         var fragCode =
            'void main(void) {' +
               ' gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' +
            '}';

         // Create fragment shader object
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);

         // Attach fragment shader source code
         gl.shaderSource(fragShader, fragCode);

         // Compile the fragmentt shader
         gl.compileShader(fragShader);
         
         // Create a shader program object to store
         // the combined shader program
         var shaderProgram = gl.createProgram();

         // Attach a vertex shader
         gl.attachShader(shaderProgram, vertShader); 

         // Attach a fragment shader
         gl.attachShader(shaderProgram, fragShader);

         // Link both programs
         gl.linkProgram(shaderProgram);

         // Use the combined shader program object
         gl.useProgram(shaderProgram);

         /*======== Associating shaders to buffer objects ========*/

         // Bind vertex buffer object
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Get the attribute location
         var coord = gl.getAttribLocation(shaderProgram, "coordinates");

         // Point an attribute to the currently bound VBO
         gl.vertexAttribPointer(coord, 3, gl.FLOAT, false, 0, 0);

         // Enable the attribute
         gl.enableVertexAttribArray(coord);

         /*============= Drawing the primitive ===============*/

         // Clear the canvas
         gl.clearColor(0.5, 0.5, 0.5, 0.9);

         // Enable the depth test
         gl.enable(gl.DEPTH_TEST);
 
         // Clear the color buffer bit
         gl.clear(gl.COLOR_BUFFER_BIT);

         // Set the view port
         gl.viewport(0,0,canvas.width,canvas.height);

         // Draw the triangle
         gl.drawArrays(gl.POINTS, 0, 3);
      </script>
   </body>
</html>

次の結果が得られます-

前の章(第11章)では、WebGLを使用して3つのポイントを描画する方法について説明しました。第5章では、サンプルアプリケーションを使用して、三角形の描画方法を示しました。どちらの例でも、頂点のみを使用してプリミティブを描画しています。

より複雑なシェイプ/メッシュを描画するために、ジオメトリのインデックスも頂点とともにシェーダーに渡します。この章では、インデックスを使用して三角形を描画する方法を説明します。

三角形を描くために必要な手順

三角形を描画するWebGLアプリケーションを作成するには、次の手順が必要です。

Step 1 − Prepare the Canvas and Get WebGL Rendering Context

このステップでは、を使用してWebGLレンダリングコンテキストオブジェクトを取得します。 getContext()

Step 2 − Define the Geometry and Store it in Buffer Objects

インデックスを使用して三角形を描画しているため、インデックスを含む三角形の3つの頂点を渡し、それらをバッファに格納する必要があります。

var vertices = [
   -0.5,0.5,0.0,
   -0.5,-0.5,0.0,
   0.5,-0.5,0.0, 
];
	
indices = [0,1,2];

Step 3 − Create and Compile the Shader Programs

このステップでは、頂点シェーダーとフラグメントシェーダーのプログラムを作成してコンパイルし、これら2つのプログラムをリンクして結合プログラムを作成する必要があります。

  • Vertex Shader −プログラムの頂点シェーダーで、3D座標を格納するベクトル属性を定義し、それに割り当てます。 gl_position

var vertCode =
   'attribute vec3 coordinates;' +
	
   'void main(void) {' +
      ' gl_Position = vec4(coordinates, 1.0);' +
   '}';
  • Fragment Shader −フラグメントシェーダーでは、フラグメントの色をに割り当てるだけです。 gl_FragColor 変数。

var fragCode = 'void main(void) {' +
   ' gl_FragColor = vec4(1, 0.5, 0.0, 1);' +
'}';

Step 4 − Associate the Shader Programs to the Buffer Objects

このステップでは、バッファオブジェクトとシェーダープログラムを関連付けます。

Step 5 − Drawing the Required Object

インデックスを使用して三角形を描画しているので、を使用しますdrawElements()。このメソッドには、インデックスの数を渡す必要があります。の値indices.length インデックスの数を示します。

gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);

例–三角形を描く

次のプログラムコードは、インデックスを使用してWebGLで三角形を描画する方法を示しています-

<!doctype html>
<html>
   <body>
      <canvas width = "570" height = "570" id = "my_Canvas"></canvas>

      <script>
         /*============== Creating a canvas ====================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl');
      
         /*======== Defining and storing the geometry ===========*/

         var vertices = [
            -0.5,0.5,0.0,
            -0.5,-0.5,0.0,
            0.5,-0.5,0.0, 
         ];
         
         indices = [0,1,2];
         
         // Create an empty buffer object to store vertex buffer
         var vertex_buffer = gl.createBuffer();

         // Bind appropriate array buffer to it
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         
         // Pass the vertex data to the buffer
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Unbind the buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, null);

         // Create an empty buffer object to store Index buffer
         var Index_Buffer = gl.createBuffer();

         // Bind appropriate array buffer to it
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, Index_Buffer);

         // Pass the vertex data to the buffer
         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);
         
         // Unbind the buffer
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, null);

         /*================ Shaders ====================*/
         
         // Vertex shader source code
         var vertCode =
            'attribute vec3 coordinates;' +
				
            'void main(void) {' +
               ' gl_Position = vec4(coordinates, 1.0);' +
            '}';
            
         // Create a vertex shader object
         var vertShader = gl.createShader(gl.VERTEX_SHADER);

         // Attach vertex shader source code
         gl.shaderSource(vertShader, vertCode);

         // Compile the vertex shader
         gl.compileShader(vertShader);

         //fragment shader source code
         var fragCode =
            'void main(void) {' +
               ' gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' +
            '}';
            
         // Create fragment shader object
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);

         // Attach fragment shader source code
         gl.shaderSource(fragShader, fragCode); 
         
         // Compile the fragmentt shader
         gl.compileShader(fragShader);

         // Create a shader program object to store
         // the combined shader program
         var shaderProgram = gl.createProgram();

         // Attach a vertex shader
         gl.attachShader(shaderProgram, vertShader);

         // Attach a fragment shader
         gl.attachShader(shaderProgram, fragShader);

         // Link both the programs
         gl.linkProgram(shaderProgram);

         // Use the combined shader program object
         gl.useProgram(shaderProgram);

         /*======= Associating shaders to buffer objects =======*/

         // Bind vertex buffer object
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Bind index buffer object
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, Index_Buffer);
         
         // Get the attribute location
         var coord = gl.getAttribLocation(shaderProgram, "coordinates");

         // Point an attribute to the currently bound VBO
         gl.vertexAttribPointer(coord, 3, gl.FLOAT, false, 0, 0); 
         
         // Enable the attribute
         gl.enableVertexAttribArray(coord);

         /*=========Drawing the triangle===========*/

         // Clear the canvas
         gl.clearColor(0.5, 0.5, 0.5, 0.9);

         // Enable the depth test
         gl.enable(gl.DEPTH_TEST);

         // Clear the color buffer bit
         gl.clear(gl.COLOR_BUFFER_BIT);

         // Set the view port
         gl.viewport(0,0,canvas.width,canvas.height);

         // Draw the triangle
         gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);
      </script>
    </body>
</html>

この例を実行すると、次の出力が生成されます-

前の章(第12章)では、WebGLを使用して三角形を描画する方法について説明しました。三角形に加えて、WebGLは他のさまざまな描画モードをサポートしています。この章では、WebGLでサポートされている描画モードについて説明します。

モードパラメータ

メソッドの構文を見てみましょう- drawElements() 描いて Arrays()

void drawElements(enum mode, long count, enum type, long offset);

void drawArrays(enum mode, int first, long count);

はっきりと観察すると、どちらの方法もパラメータを受け入れます mode。このパラメーターを使用して、プログラマーはWebGLで描画モードを選択できます。

WebGLが提供する描画モードを次の表に示します。

シニア番号 モードと説明
1

gl.POINTS

一連の点を描画します。

2

gl.LINES

一連の接続されていない線分(個々の線)を描画します。

3

gl.LINE_STRIP

一連の接続された線分を描画します。

4

gl.LINE_LOOP

一連の接続された線分を描画します。また、最初と最後の頂点を結合してループを形成します。

5

gl.TRIANGLES

一連の別々の三角形を描画します。

6

gl.TRIANGLE_STRIP

接続された一連の三角形をストリップ形式で描画します。

7

gl.TRIANGLE_FAN

最初の頂点を扇状に共有する一連の接続された三角形を描画します。

例–3本の平行線を引く

次の例は、を使用して3本の平行線を描画する方法を示しています。 gl.LINES

<!doctype html>
<html>
   <body>
      <canvas width = "300" height = "300" id = "my_Canvas"></canvas>

      <script>
         /*======= Creating a canvas =========*/

         var canvas = document.getElementById('my_Canvas');
         var gl = canvas.getContext('experimental-webgl');

         /*======= Defining and storing the geometry ======*/

         var vertices = [
            -0.7,-0.1,0,
            -0.3,0.6,0,
            -0.3,-0.3,0,
            0.2,0.6,0,
            0.3,-0.3,0,
            0.7,0.6,0 
         ]

         // Create an empty buffer object
         var vertex_buffer = gl.createBuffer();

         // Bind appropriate array buffer to it
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
      
         // Pass the vertex data to the buffer
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Unbind the buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, null);

         /*=================== Shaders ====================*/

         // Vertex shader source code
         var vertCode =
            'attribute vec3 coordinates;' +
            'void main(void) {' +
               ' gl_Position = vec4(coordinates, 1.0);' +
            '}';

         // Create a vertex shader object
         var vertShader = gl.createShader(gl.VERTEX_SHADER);

         // Attach vertex shader source code
         gl.shaderSource(vertShader, vertCode);

         // Compile the vertex shader
         gl.compileShader(vertShader);

         // Fragment shader source code
         var fragCode =
            'void main(void) {' +
               'gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' +
            '}';

         // Create fragment shader object
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);

         // Attach fragment shader source code
         gl.shaderSource(fragShader, fragCode);

         // Compile the fragmentt shader
         gl.compileShader(fragShader);

         // Create a shader program object to store
         // the combined shader program
         var shaderProgram = gl.createProgram();

         // Attach a vertex shader
         gl.attachShader(shaderProgram, vertShader);

         // Attach a fragment shader
         gl.attachShader(shaderProgram, fragShader);

         // Link both the programs
         gl.linkProgram(shaderProgram);

         // Use the combined shader program object
         gl.useProgram(shaderProgram);

         /*======= Associating shaders to buffer objects ======*/

         // Bind vertex buffer object
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Get the attribute location
         var coord = gl.getAttribLocation(shaderProgram, "coordinates");

         // Point an attribute to the currently bound VBO
         gl.vertexAttribPointer(coord, 3, gl.FLOAT, false, 0, 0);

         // Enable the attribute
         gl.enableVertexAttribArray(coord);

         /*============ Drawing the triangle =============*/

         // Clear the canvas
         gl.clearColor(0.5, 0.5, 0.5, 0.9);

         // Enable the depth test
         gl.enable(gl.DEPTH_TEST);

         // Clear the color and depth buffer
         gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);

         // Set the view port
         gl.viewport(0,0,canvas.width,canvas.height);

         // Draw the triangle
         gl.drawArrays(gl.LINES, 0, 6);

         // POINTS, LINE_STRIP, LINE_LOOP, LINES,
         // TRIANGLE_STRIP,TRIANGLE_FAN, TRIANGLES
      </script>
   </body>
</html>

この例を実行すると、次の出力が生成されます-

描画モード

上記のプログラムで、のモードを置き換えると drawArrays() 次の描画モードのいずれかを使用すると、毎回異なる出力が生成されます。

描画モード 出力
LINE_STRIP
LINE_LOOP
TRIANGLE_STRIP
TRIANGLE_FAN
三角形

前の章では、WebGLによって提供されるさまざまな描画モードについて説明しました。インデックスを使用して、これらのモードの1つを使用してプリミティブを描画することもできます。WebGLでモデルを描画するには、これらのプリミティブの1つを選択し、必要なメッシュ(つまり、1つ以上のプリミティブを使用して形成されたモデル)を描画する必要があります。

この章では、WebGLを使用して四辺形を描画する方法を示す例を取り上げます。

四辺形を描く手順

四辺形を描画するWebGLアプリケーションを作成するには、次の手順が必要です。

Step 1 − Prepare the Canvas and Get the WebGL Rendering Context

このステップでは、を使用してWebGLレンダリングコンテキストオブジェクトを取得します。 getContext()

Step 2 − Define the Geometry and Store it in the Buffer Objects

正方形は2つの三角形を使用して描くことができます。この例では、2つの三角形(1つの共通のエッジを持つ)とインデックスの頂点を提供します。

var vertices = [
   -0.5,0.5,0.0,
   -0.5,-0.5,0.0,
   0.5,-0.5,0.0,
   0.5,0.5,0.0 
];

indices = [3,2,1,3,1,0];

Step 3 − Create and Compile the Shader Programs

このステップでは、頂点シェーダープログラムとフラグメントシェーダープログラムを作成してコンパイルし、これら2つのプログラムをリンクして結合プログラムを作成する必要があります。

  • Vertex Shader −プログラムの頂点シェーダーで、3D座標を格納するベクトル属性を定義し、それに割り当てます。 gl_position

var vertCode =
   'attribute vec3 coordinates;' +
   'void main(void) {' +
      ' gl_Position = vec4(coordinates, 1.0);' +
   '}';
  • Fragment Shader −フラグメントシェーダーでは、フラグメントの色をに割り当てるだけです。 gl_FragColor 変数。

var fragCode = 'void main(void) {' +' gl_FragColor = vec4(0.5, 0.3, 0.0, 7.5);' +'}';

Step 4 − Associate the Shader Programs to Buffer Objects

このステップでは、バッファオブジェクトをシェーダープログラムに関連付けます。

Step 5 − Drawing the Required Object

インデックスを使用して2つの三角形を描画してクワッドを形成しているため、この方法を使用します。 drawElements()。このメソッドには、インデックスの数を渡す必要があります。の値indices.length インデックスの数を示します。

gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);

例–四辺形を描く

次のプログラムは、四辺形を描画するWebGLアプリケーションを作成する方法を示しています。

<!doctype html>
<html>
   <body>
      <canvas width = "570" height = "570" id = "my_Canvas"></canvas>

      <script>
         /*============ Creating a canvas =================*/
      
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl');
      
         /*========== Defining and storing the geometry =========*/

         var vertices = [
            -0.5,0.5,0.0,
            -0.5,-0.5,0.0,
            0.5,-0.5,0.0,
            0.5,0.5,0.0 
         ];

         indices = [3,2,1,3,1,0];

         // Create an empty buffer object to store vertex buffer
         var vertex_buffer = gl.createBuffer();

         // Bind appropriate array buffer to it
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Pass the vertex data to the buffer
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Unbind the buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, null);

         // Create an empty buffer object to store Index buffer
         var Index_Buffer = gl.createBuffer();

         // Bind appropriate array buffer to it
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, Index_Buffer);

         // Pass the vertex data to the buffer
         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);

         // Unbind the buffer
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, null);

         /*====================== Shaders =======================*/

         // Vertex shader source code
         var vertCode =
            'attribute vec3 coordinates;' +
            'void main(void) {' +
               ' gl_Position = vec4(coordinates, 1.0);' +
            '}';

         // Create a vertex shader object
         var vertShader = gl.createShader(gl.VERTEX_SHADER);

         // Attach vertex shader source code
         gl.shaderSource(vertShader, vertCode);

         // Compile the vertex shader
         gl.compileShader(vertShader);

         // Fragment shader source code
         var fragCode =
            'void main(void) {' +
               ' gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' +
            '}';

         // Create fragment shader object 
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);

         // Attach fragment shader source code
         gl.shaderSource(fragShader, fragCode);

         // Compile the fragmentt shader
         gl.compileShader(fragShader);

         // Create a shader program object to
         // store the combined shader program
         var shaderProgram = gl.createProgram();

         // Attach a vertex shader
         gl.attachShader(shaderProgram, vertShader);

         // Attach a fragment shader
         gl.attachShader(shaderProgram, fragShader);

         // Link both the programs
         gl.linkProgram(shaderProgram);

         // Use the combined shader program object
         gl.useProgram(shaderProgram);

         /* ======= Associating shaders to buffer objects =======*/

         // Bind vertex buffer object
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Bind index buffer object
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, Index_Buffer); 

         // Get the attribute location
         var coord = gl.getAttribLocation(shaderProgram, "coordinates");

         // Point an attribute to the currently bound VBO
         gl.vertexAttribPointer(coord, 3, gl.FLOAT, false, 0, 0);

         // Enable the attribute
         gl.enableVertexAttribArray(coord);

         /*============= Drawing the Quad ================*/

         // Clear the canvas
         gl.clearColor(0.5, 0.5, 0.5, 0.9);

         // Enable the depth test
         gl.enable(gl.DEPTH_TEST);

         // Clear the color buffer bit
         gl.clear(gl.COLOR_BUFFER_BIT);

         // Set the view port
         gl.viewport(0,0,canvas.width,canvas.height);

         // Draw the triangle
         gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);
      </script>
   </body>
</html>

この例を実行すると、次の出力が生成されます-

これまでのすべての例では、目的の色の値をオブジェクトに割り当てることにより、オブジェクトに色を適用しました。 gl_FragColor変数。それに加えて、頂点の座標やインデックスと同じように、各頂点の色を定義できます。この章では、WebGLを使用して四辺形に色を適用する方法を示す例を取り上げます。

色を塗る

色を適用するには、JavaScript配列でRGB値を使用して各頂点の色を定義する必要があります。すべての頂点に同じ値を割り当てて、オブジェクトに固有の色を付けることができます。色を定義した後、カラーバッファーを作成してこれらの値を格納し、頂点シェーダー属性に関連付ける必要があります。

頂点シェーダーでは、座標属性(頂点の位置を保持する)とともに、次のように定義します。 attributevarying 色を処理します。

ザ・ color 属性は頂点ごとの色の値を保持し、 varyingフラグメントシェーダーへの入力として渡される変数です。したがって、を割り当てる必要がありますcolor の値 varying

フラグメントシェーダーでは、 varying に割り当てられている色の値を保持する gl_FragColor、オブジェクトの最終的な色を保持します。

色を適用する手順

クワッドを描画して色を適用するWebGLアプリケーションを作成するには、次の手順が必要です。

Step 1 − Prepare the Canvas and Get the WebGL Rendering Context

このステップでは、を使用してWebGLレンダリングコンテキストオブジェクトを取得します。 getContext()

Step 2 − Define the Geometry and Store it in the Buffer Objects

正方形は2つの三角形を使用して描くことができます。したがって、この例では、2つの三角形(1つの共通のエッジを持つ)とインデックスの頂点を提供します。色を適用したいので、色の値を保持する変数も定義され、それぞれの色の値(赤、青、緑、ピンク)が割り当てられます。

var vertices = [
   -0.5,0.5,0.0,
   -0.5,-0.5,0.0, 
   0.5,-0.5,0.0,
   0.5,0.5,0.0 
];

var colors = [ 0,0,1, 1,0,0, 0,1,0, 1,0,1,];
indices = [3,2,1,3,1,0];

Step 3 − Create and Compile the Shader Programs

このステップでは、頂点シェーダープログラムとフラグメントシェーダープログラムを作成してコンパイルし、これら2つのプログラムをリンクして結合プログラムを作成する必要があります。

  • Vertex Shader−プログラムの頂点シェーダーで、3D座標(位置)と各頂点の色を格納するためのベクトル属性を定義します。Avaring変数は、頂点シェーダーからフラグメントシェーダーにカラー値を渡すように宣言されています。そして最後に、color属性に格納されている値がに割り当てられますvarying

var vertCode = 'attribute vec3 coordinates;'+
   'attribute vec3 color;'+
   'varying vec3 vColor;'+
	
   'void main(void) {' +
      ' gl_Position = vec4(coordinates, 1.0);' +
      'vColor = color;'+
   '}';
  • Fragment Shader −フラグメントシェーダーで、 varyinggl_FragColor 変数。

var fragCode = 'precision mediump float;'+
   'varying vec3 vColor;'+
   'void main(void) {'+
      'gl_FragColor = vec4(vColor, 1.);'+
   '}';

Step 4 − Associate the Shader Programs with the Buffer Objects

このステップでは、バッファオブジェクトとシェーダープログラムを関連付けます。

Step 5 − Drawing the Required Object

インデックスを使用して、クワッドを形成する2つの三角形を描画しているため、この方法を使用します。 drawElements()。このメソッドには、インデックスの数を渡す必要があります。の値indices.length インデックスの数を示します。

gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);

例–色の適用

次のプログラムは、WebGLアプリケーションを使用してクワッドを描画し、それに色を適用する方法を示しています。

<!doctype html>
<html>
   <body>
    <canvas width = "300" height = "300" id = "my_Canvas"></canvas>

      <script>
         /*============= Creating a canvas ==================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl');
         
         /*========== Defining and storing the geometry ==========*/

         var vertices = [
            -0.5,0.5,0.0,
            -0.5,-0.5,0.0,
            0.5,-0.5,0.0,
            0.5,0.5,0.0
         ];

         var colors = [0,0,1, 1,0,0, 0,1,0, 1,0,1,];
         
         indices = [3,2,1,3,1,0];
         
         // Create an empty buffer object and store vertex data
         var vertex_buffer = gl.createBuffer();
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);
         gl.bindBuffer(gl.ARRAY_BUFFER, null);

         // Create an empty buffer object and store Index data
         var Index_Buffer = gl.createBuffer();
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, Index_Buffer);
         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, null);

         // Create an empty buffer object and store color data
         var color_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(colors), gl.STATIC_DRAW);

         /*======================= Shaders =======================*/
         
         // vertex shader source code
         var vertCode = 'attribute vec3 coordinates;'+
            'attribute vec3 color;'+
            'varying vec3 vColor;'+
            'void main(void) {' +
               ' gl_Position = vec4(coordinates, 1.0);' +
               'vColor = color;'+
            '}';
            
         // Create a vertex shader object
         var vertShader = gl.createShader(gl.VERTEX_SHADER);

         // Attach vertex shader source code
         gl.shaderSource(vertShader, vertCode);

         // Compile the vertex shader
         gl.compileShader(vertShader);


         // fragment shader source code
         var fragCode = 'precision mediump float;'+
            'varying vec3 vColor;'+
            'void main(void) {'+
               'gl_FragColor = vec4(vColor, 1.);'+
            '}';
            
         // Create fragment shader object
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);

         // Attach fragment shader source code
         gl.shaderSource(fragShader, fragCode);

         // Compile the fragmentt shader
         gl.compileShader(fragShader);

         // Create a shader program object to
         // store the combined shader program
         var shaderProgram = gl.createProgram();

         // Attach a vertex shader
         gl.attachShader(shaderProgram, vertShader);

         // Attach a fragment shader
         gl.attachShader(shaderProgram, fragShader);

         // Link both the programs
         gl.linkProgram(shaderProgram);

         // Use the combined shader program object
         gl.useProgram(shaderProgram);

         /* ======== Associating shaders to buffer objects =======*/

         // Bind vertex buffer object
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Bind index buffer object
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, Index_Buffer);

         // Get the attribute location
         var coord = gl.getAttribLocation(shaderProgram, "coordinates");

         // point an attribute to the currently bound VBO
         gl.vertexAttribPointer(coord, 3, gl.FLOAT, false, 0, 0);

         // Enable the attribute
         gl.enableVertexAttribArray(coord);

         // bind the color buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         
         // get the attribute location
         var color = gl.getAttribLocation(shaderProgram, "color");
 
         // point attribute to the volor buffer object
         gl.vertexAttribPointer(color, 3, gl.FLOAT, false,0,0) ;
 
         // enable the color attribute
         gl.enableVertexAttribArray(color);

         /*============Drawing the Quad====================*/

         // Clear the canvas
         gl.clearColor(0.5, 0.5, 0.5, 0.9);

         // Enable the depth test
         gl.enable(gl.DEPTH_TEST);

         // Clear the color buffer bit
         gl.clear(gl.COLOR_BUFFER_BIT);

         // Set the view port
         gl.viewport(0,0,canvas.width,canvas.height);

         //Draw the triangle
         gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);
      </script>
   </body>
</html>

この例を実行すると、次の出力が生成されます-

これまで、WebGLを使用してさまざまな形状を描画し、それらに色を適用する方法について説明してきました。ここで、この章では、三角形を平行移動する方法を示す例を取り上げます。

翻訳

翻訳はの1つです affine transformationsWebGLによって提供されます。平行移動を使用して、xyz平面上で三角形(任意のオブジェクト)を移動できます。三角形[a、b、c]があり、正のX軸に向かって5単位、正のY軸に向かって3単位の位置に三角形を移動するとします。その場合、新しい頂点は[a + 5、b + 3、c +0]になります。つまり、三角形を平行移動するには、各頂点に平行移動距離(tx、ty、tzなど)を追加する必要があります。

なので per-vertex operation, 頂点シェーダープログラムで実行できます。

頂点シェーダーでは、属性とともに、 coordinates(頂点位置を保持する)、並進距離(x、y、z)を保持する一様変数を定義します。後で、この一様変数を座標変数に追加し、結果をに割り当てます。gl_Position 変数。

Note −頂点シェーダーは各頂点で実行されるため、三角形のすべての頂点が変換されます。

三角形を平行移動する手順

三角形を描画して新しい位置に変換するWebGLアプリケーションを作成するには、次の手順が必要です。

Step 1 − Prepare the Canvas and Get the WebGL Rendering Context

このステップでは、を使用してWebGLレンダリングコンテキストオブジェクトを取得します。 getContext()

Step 2 − Define the Geometry and Store it in the Buffer Objects

三角形を描画しているので、三角形の3つの頂点を渡し、それらをバッファに格納する必要があります。

var vertices = [ -0.5,0.5,0.0, -0.5,-0.5,0.0, 0.5,-0.5,0.0, ];

Step 3 − Create and Compile the Shader Programs

このステップでは、頂点シェーダープログラムとフラグメントシェーダープログラムを作成してコンパイルし、これら2つのプログラムをリンクして結合プログラムを作成する必要があります。

  • Vertex Shader−プログラムの頂点シェーダーで、3D座標を格納するためのベクトル属性を定義します。それに加えて、並進距離を格納するための一様変数を定義し、最後に、これら2つの値を加算してに割り当てます。gl_position これは、頂点の最終的な位置を保持します。

var vertCode =
   'attribute vec4 coordinates;' +
   'uniform vec4 translation;'+
   'void main(void) {' +
      ' gl_Position = coordinates + translation;' +
   '}';
  • Fragment Shader −フラグメントシェーダーでは、フラグメントの色を変数gl_FragColorに割り当てるだけです。

var fragCode = 'void main(void) {' +' gl_FragColor = vec4(1, 0.5, 0.0, 1);' +'}';

Step 4 − Associate the Shader Programs to the Buffer Objects

このステップでは、バッファオブジェクトをシェーダープログラムに関連付けます。

Step 5 − Drawing the Required Object

インデックスを使用して三角形を描画しているので、この方法を使用します drawArrays()。このメソッドには、考慮される頂点/要素の数を渡す必要があります。三角形を描いているので、パラメータとして3を渡します。

gl.drawArrays(gl.TRIANGLES, 0, 3);

例–三角形を平行移動する

次の例は、xyz平面上の三角形を平行移動する方法を示しています。

<!doctype html>
<html>
   <body>
      <canvas width = "300" height = "300" id = "my_Canvas"></canvas>
         
      <script>
         /*=================Creating a canvas=========================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl'); 
 
         /*===========Defining and storing the geometry==============*/
         var vertices = [
            -0.5,0.5,0.0, 	
            -0.5,-0.5,0.0, 	
            0.5,-0.5,0.0,   
         ];
            
         //Create an empty buffer object and store vertex data            
         var vertex_buffer = gl.createBuffer(); 
			
         //Create a new buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);   
			
         //bind it to the current buffer			
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW); 
			
         // Pass the buffer data
         gl.bindBuffer(gl.ARRAY_BUFFER, null);  
            
         /*========================Shaders============================*/
            
         //vertex shader source code 
         var vertCode =
            'attribute vec4 coordinates;' + 
            'uniform vec4 translation;'+
            'void main(void) {' +
               '  gl_Position = coordinates + translation;' +
            '}';
            
         //Create a vertex shader program object and compile it              
         var vertShader = gl.createShader(gl.VERTEX_SHADER);
         gl.shaderSource(vertShader, vertCode);
         gl.compileShader(vertShader);
            
   
         //fragment shader source code
         var fragCode =
            'void main(void) {' +
               '   gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' +
            '}';

         //Create a fragment shader program object and compile it            
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);
         gl.shaderSource(fragShader, fragCode);
         gl.compileShader(fragShader);
            
         //Create and use combiened shader program
         var shaderProgram = gl.createProgram();
         gl.attachShader(shaderProgram, vertShader);
         gl.attachShader(shaderProgram, fragShader);
         gl.linkProgram(shaderProgram);
   
         gl.useProgram(shaderProgram);
   
         /* ===========Associating shaders to buffer objects============*/
      
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);    
         var coordinatesVar = gl.getAttribLocation(shaderProgram, "coordinates");
         gl.vertexAttribPointer(coordinatesVar, 3, gl.FLOAT, false, 0, 0);   
         gl.enableVertexAttribArray(coordinatesVar); 
   
         /* ==========translation======================================*/
         var Tx = 0.5, Ty = 0.5, Tz = 0.0;
         var translation = gl.getUniformLocation(shaderProgram, 'translation');
         gl.uniform4f(translation, Tx, Ty, Tz, 0.0);
 
         /*=================Drawing the riangle and transforming it========================*/ 

         gl.clearColor(0.5, 0.5, 0.5, 0.9);
         gl.enable(gl.DEPTH_TEST);
   
         gl.clear(gl.COLOR_BUFFER_BIT);
         gl.viewport(0,0,canvas.width,canvas.height);
         gl.drawArrays(gl.TRIANGLES, 0, 3);
      </script>
    </body>
 </html>

この例を実行すると、次の出力が生成されます-

この章では、WebGLを使用して三角形のスケールを変更する方法を示す例を取り上げます。

スケーリング

スケーリングは、オブジェクトのサイズを拡大または縮小することに他なりません。たとえば、三角形にサイズ[a、b、c]の頂点がある場合、頂点[2a、2b、2c]を持つ三角形のサイズは2倍になります。したがって、三角形をスケーリングするには、各頂点にスケーリング係数を掛ける必要があります。特定の頂点をスケーリングすることもできます。

三角形をスケーリングするには、プログラムの頂点シェーダーで、均一な行列を作成し、座標値にこの行列を掛けます。後で、対角位置(最後の対角位置1)にx、y、z座標のスケーリング係数を持つ4×4の対角行列を渡します。

必要な手順

三角形をスケーリングするWebGLアプリケーションを作成するには、次の手順が必要です。

Step 1 − Prepare the Canvas and Get the WebGL Rendering Context

このステップでは、を使用してWebGLレンダリングコンテキストオブジェクトを取得します。 getContext()

Step 2 − Define the Geometry and Store it in the Buffer Objects

三角形を描画しているので、三角形の3つの頂点を渡し、それらをバッファに格納する必要があります。

var vertices = [ -0.5,0.5,0.0, -0.5,-0.5,0.0, 0.5,-0.5,0.0, ];

Step 3 − Create and Compile the Shader Programs

このステップでは、頂点シェーダープログラムとフラグメントシェーダープログラムを作成してコンパイルし、これら2つのプログラムをリンクして結合プログラムを作成する必要があります。

  • Vertex Shader−プログラムの頂点シェーダーで、3D座標を格納するためのベクトル属性を定義します。それに加えて、スケーリング係数を格納するための均一な行列を定義し、最後に、これら2つの値を乗算してに割り当てます。gl_position これは、頂点の最終的な位置を保持します。

var vertCode =
   'attribute vec4 coordinates;' +
   'uniform mat4 u_xformMatrix;' +
   'void main(void) {' +
      ' gl_Position = u_xformMatrix * coordinates;' +
   '}';
  • Fragment Shader −フラグメントシェーダーでは、フラグメントの色をに割り当てるだけです。 gl_FragColor 変数。

var fragCode = 'void main(void) {' +' gl_FragColor = vec4(1, 0.5, 0.0, 1);' +'}';

Step 4 − Associate the Shader Programs with the Buffer Objects

このステップでは、バッファオブジェクトをシェーダープログラムに関連付けます。

Step 5 − Drawing the Required Object

インデックスを使用して三角形を描画しているため、 drawArrays()方法。この方法では、考慮される頂点/要素の数を渡す必要があります。三角形を描いているので、パラメータとして3を渡します。

gl.drawArrays(gl.TRIANGLES, 0, 3);

例–三角形をスケーリングする

次の例は、三角形をスケーリングする方法を示しています-

<!doctype html>
<html>
   <body>
      <canvas width = "300" height = "300" id = "my_Canvas"></canvas>

      <script>
         /*=================Creating a canvas=========================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl'); 

         /*===========Defining and storing the geometry==============*/
         var vertices =  [
            -0.5,0.5,0.0, 	
            -0.5,-0.5,0.0, 	
            0.5,-0.5,0.0,   
         ];

         //Create an empty buffer object and store vertex data

         var vertex_buffer = gl.createBuffer();                                                     
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);                                                
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);           
         gl.bindBuffer(gl.ARRAY_BUFFER, null);  

         /*========================Shaders============================*/

         //Vertex shader source code
         var vertCode =
            'attribute vec4 coordinates;' + 
            'uniform mat4 u_xformMatrix;' +
            'void main(void) {' +
               '  gl_Position = u_xformMatrix * coordinates;' +
            '}';

         //Create a vertex shader program object and compile it                
         var vertShader = gl.createShader(gl.VERTEX_SHADER);
         gl.shaderSource(vertShader, vertCode);
         gl.compileShader(vertShader);

         //fragment shader source code
         var fragCode =
            'void main(void) {' +
               '   gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' +
            '}';

         //Create a fragment shader program object and compile it 
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);
         gl.shaderSource(fragShader, fragCode);
         gl.compileShader(fragShader);

         //Create and use combiened shader program
         var shaderProgram = gl.createProgram();
         gl.attachShader(shaderProgram, vertShader);
         gl.attachShader(shaderProgram, fragShader);
         gl.linkProgram(shaderProgram);

         gl.useProgram(shaderProgram); 

         /*===================scaling==========================*/

         var Sx = 1.0, Sy = 1.5, Sz = 1.0;
         var xformMatrix = new Float32Array([
            Sx,   0.0,  0.0,  0.0,
            0.0,  Sy,   0.0,  0.0,
            0.0,  0.0,  Sz,   0.0,
            0.0,  0.0,  0.0,  1.0  
         ]);

         var u_xformMatrix = gl.getUniformLocation(shaderProgram, 'u_xformMatrix');
         gl.uniformMatrix4fv(u_xformMatrix, false, xformMatrix);

         /* ===========Associating shaders to buffer objects============*/
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);   

         var coordinatesVar = gl.getAttribLocation(shaderProgram, "coordinates"); 
         gl.vertexAttribPointer(coordinatesVar, 3, gl.FLOAT, false, 0, 0);  
         gl.enableVertexAttribArray(coordinatesVar);

         /*=================Drawing the Quad========================*/ 
         gl.clearColor(0.5, 0.5, 0.5, 0.9);
         gl.enable(gl.DEPTH_TEST);

         gl.clear(gl.COLOR_BUFFER_BIT);
         gl.viewport(0,0,canvas.width,canvas.height);
         gl.drawArrays(gl.TRIANGLES, 0, 3);
      </script>
   </body>
</html>

この例を実行すると、次の出力が生成されます-

この章では、WebGLを使用して三角形を回転させる方法を示す例を取り上げます。

例–三角形を回転させる

次のプログラムは、WebGLを使用して三角形を回転させる方法を示しています。

<!doctype html>
<html>
   <body>
      <canvas width = "400" height = "400" id = "my_Canvas"></canvas>

      <script>
         /*=================Creating a canvas=========================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl');

         /*===========Defining and storing the geometry==============*/

         var vertices = [ -1,-1,-1, 1,-1,-1, 1, 1,-1 ];
         var colors = [ 1,1,1, 1,1,1, 1,1,1 ];
         var indices = [ 0,1,2 ];

         //Create and store data into vertex buffer
         var vertex_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         //Create and store data into color buffer
         var color_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(colors), gl.STATIC_DRAW);

         //Create and store data into index buffer
         var index_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);
         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);

         /*==========================Shaders=========================*/

         var vertCode = 'attribute vec3 position;'+
            'uniform mat4 Pmatrix;'+
            'uniform mat4 Vmatrix;'+
            'uniform mat4 Mmatrix;'+
            'attribute vec3 color;'+//the color of the point
            'varying vec3 vColor;'+

            'void main(void) { '+//pre-built function
               'gl_Position = Pmatrix*Vmatrix*Mmatrix*vec4(position, 1.);'+
               'vColor = color;'+
            '}';

         var fragCode = 'precision mediump float;'+
            'varying vec3 vColor;'+
            'void main(void) {'+
               'gl_FragColor = vec4(vColor, 1.);'+
            '}';

         var vertShader = gl.createShader(gl.VERTEX_SHADER);
         gl.shaderSource(vertShader, vertCode);
         gl.compileShader(vertShader);

         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);
         gl.shaderSource(fragShader, fragCode);
         gl.compileShader(fragShader);

         var shaderProgram = gl.createProgram();
         gl.attachShader(shaderProgram, vertShader);
         gl.attachShader(shaderProgram, fragShader);
         gl.linkProgram(shaderProgram);

         /*===========associating attributes to vertex shader ============*/

         var Pmatrix = gl.getUniformLocation(shaderProgram, "Pmatrix");
         var Vmatrix = gl.getUniformLocation(shaderProgram, "Vmatrix");
         var Mmatrix = gl.getUniformLocation(shaderProgram, "Mmatrix");
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         var position = gl.getAttribLocation(shaderProgram, "position");
         gl.vertexAttribPointer(position, 3, gl.FLOAT, false,0,0) ; //position
         gl.enableVertexAttribArray(position);
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);

         var color = gl.getAttribLocation(shaderProgram, "color");
         gl.vertexAttribPointer(color, 3, gl.FLOAT, false,0,0) ; //color
         gl.enableVertexAttribArray(color);
         gl.useProgram(shaderProgram);

         /*========================= MATRIX ========================= */

         function get_projection(angle, a, zMin, zMax) {
            var ang = Math.tan((angle*.5)*Math.PI/180);//angle*.5
            return [
               0.5/ang, 0 , 0, 0,
               0, 0.5*a/ang, 0, 0,
               0, 0, -(zMax+zMin)/(zMax-zMin), -1,
               0, 0, (-2*zMax*zMin)/(zMax-zMin), 0
            ];
         }

         var proj_matrix = get_projection(40, canvas.width/canvas.height, 1, 100);
         var mov_matrix = [1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1];
         var view_matrix = [1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1];

         //translating z
         view_matrix[14] = view_matrix[14]-6; //zoom

         /*=======================rotation========================*/
         function rotateZ(m, angle) {
            var c = Math.cos(angle);
            var s = Math.sin(angle);
            var mv0 = m[0], mv4 = m[4], mv8 = m[8]; 

            m[0] = c*m[0]-s*m[1];
            m[4] = c*m[4]-s*m[5];
            m[8] = c*m[8]-s*m[9];
            m[1] = c*m[1]+s*mv0;
            m[5] = c*m[5]+s*mv4;
            m[9] = c*m[9]+s*mv8;
         }

         /*=================Drawing===========================*/

         var time_old = 0;
         var animate = function(time) {
            var dt = time-time_old;
            rotateZ(mov_matrix, dt*0.002);
            time_old = time;

            gl.enable(gl.DEPTH_TEST);
            gl.depthFunc(gl.LEQUAL);
            gl.clearColor(0.5, 0.5, 0.5, 0.9);
            gl.clearDepth(1.0);
            gl.viewport(0.0, 0.0, canvas.width, canvas.height);
            gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);

            gl.uniformMatrix4fv(Pmatrix, false, proj_matrix);
            gl.uniformMatrix4fv(Vmatrix, false, view_matrix);
            gl.uniformMatrix4fv(Mmatrix, false, mov_matrix);

            gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);
            gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT, 0);
            window.requestAnimationFrame(animate);
         }
         animate(0);
      </script>
   </body>
</html>

この例を実行すると、次の出力が生成されます-

この章では、WebGLを使用して回転する3D立方体を描画する方法を示す例を取り上げます。

例–回転する3D立方体を描く

次のプログラムは、回転する3D立方体を描画する方法を示しています-

<!doctype html>
<html>
   <body>
      <canvas width = "570" height = "570" id = "my_Canvas"></canvas>

      <script>
         /*============= Creating a canvas =================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl');

         /*============ Defining and storing the geometry =========*/

         var vertices = [
            -1,-1,-1, 1,-1,-1, 1, 1,-1, -1, 1,-1,
            -1,-1, 1, 1,-1, 1, 1, 1, 1, -1, 1, 1,
            -1,-1,-1, -1, 1,-1, -1, 1, 1, -1,-1, 1,
            1,-1,-1, 1, 1,-1, 1, 1, 1, 1,-1, 1,
            -1,-1,-1, -1,-1, 1, 1,-1, 1, 1,-1,-1,
            -1, 1,-1, -1, 1, 1, 1, 1, 1, 1, 1,-1, 
         ];

         var colors = [
            5,3,7, 5,3,7, 5,3,7, 5,3,7,
            1,1,3, 1,1,3, 1,1,3, 1,1,3,
            0,0,1, 0,0,1, 0,0,1, 0,0,1,
            1,0,0, 1,0,0, 1,0,0, 1,0,0,
            1,1,0, 1,1,0, 1,1,0, 1,1,0,
            0,1,0, 0,1,0, 0,1,0, 0,1,0
         ];

         var indices = [
            0,1,2, 0,2,3, 4,5,6, 4,6,7,
            8,9,10, 8,10,11, 12,13,14, 12,14,15,
            16,17,18, 16,18,19, 20,21,22, 20,22,23 
         ];

         // Create and store data into vertex buffer
         var vertex_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Create and store data into color buffer
         var color_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(colors), gl.STATIC_DRAW);

         // Create and store data into index buffer
         var index_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);
         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);

         /*=================== Shaders =========================*/

         var vertCode = 'attribute vec3 position;'+
            'uniform mat4 Pmatrix;'+
            'uniform mat4 Vmatrix;'+
            'uniform mat4 Mmatrix;'+
            'attribute vec3 color;'+//the color of the point
            'varying vec3 vColor;'+

            'void main(void) { '+//pre-built function
               'gl_Position = Pmatrix*Vmatrix*Mmatrix*vec4(position, 1.);'+
               'vColor = color;'+
            '}';

         var fragCode = 'precision mediump float;'+
            'varying vec3 vColor;'+
            'void main(void) {'+
               'gl_FragColor = vec4(vColor, 1.);'+
            '}';

         var vertShader = gl.createShader(gl.VERTEX_SHADER);
         gl.shaderSource(vertShader, vertCode);
         gl.compileShader(vertShader);

         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);
         gl.shaderSource(fragShader, fragCode);
         gl.compileShader(fragShader);

         var shaderProgram = gl.createProgram();
         gl.attachShader(shaderProgram, vertShader);
         gl.attachShader(shaderProgram, fragShader);
         gl.linkProgram(shaderProgram);

         /* ====== Associating attributes to vertex shader =====*/
         var Pmatrix = gl.getUniformLocation(shaderProgram, "Pmatrix");
         var Vmatrix = gl.getUniformLocation(shaderProgram, "Vmatrix");
         var Mmatrix = gl.getUniformLocation(shaderProgram, "Mmatrix");

         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         var position = gl.getAttribLocation(shaderProgram, "position");
         gl.vertexAttribPointer(position, 3, gl.FLOAT, false,0,0) ;

         // Position
         gl.enableVertexAttribArray(position);
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         var color = gl.getAttribLocation(shaderProgram, "color");
         gl.vertexAttribPointer(color, 3, gl.FLOAT, false,0,0) ;

         // Color
         gl.enableVertexAttribArray(color);
         gl.useProgram(shaderProgram);

         /*==================== MATRIX =====================*/

         function get_projection(angle, a, zMin, zMax) {
            var ang = Math.tan((angle*.5)*Math.PI/180);//angle*.5
            return [
               0.5/ang, 0 , 0, 0,
               0, 0.5*a/ang, 0, 0,
               0, 0, -(zMax+zMin)/(zMax-zMin), -1,
               0, 0, (-2*zMax*zMin)/(zMax-zMin), 0 
            ];
         }

         var proj_matrix = get_projection(40, canvas.width/canvas.height, 1, 100);

         var mov_matrix = [1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1];
         var view_matrix = [1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1];

         // translating z
         view_matrix[14] = view_matrix[14]-6;//zoom

         /*==================== Rotation ====================*/

         function rotateZ(m, angle) {
            var c = Math.cos(angle);
            var s = Math.sin(angle);
            var mv0 = m[0], mv4 = m[4], mv8 = m[8];

            m[0] = c*m[0]-s*m[1];
            m[4] = c*m[4]-s*m[5];
            m[8] = c*m[8]-s*m[9];

            m[1]=c*m[1]+s*mv0;
            m[5]=c*m[5]+s*mv4;
            m[9]=c*m[9]+s*mv8;
         }

         function rotateX(m, angle) {
            var c = Math.cos(angle);
            var s = Math.sin(angle);
            var mv1 = m[1], mv5 = m[5], mv9 = m[9];

            m[1] = m[1]*c-m[2]*s;
            m[5] = m[5]*c-m[6]*s;
            m[9] = m[9]*c-m[10]*s;

            m[2] = m[2]*c+mv1*s;
            m[6] = m[6]*c+mv5*s;
            m[10] = m[10]*c+mv9*s;
         }

         function rotateY(m, angle) {
            var c = Math.cos(angle);
            var s = Math.sin(angle);
            var mv0 = m[0], mv4 = m[4], mv8 = m[8];

            m[0] = c*m[0]+s*m[2];
            m[4] = c*m[4]+s*m[6];
            m[8] = c*m[8]+s*m[10];

            m[2] = c*m[2]-s*mv0;
            m[6] = c*m[6]-s*mv4;
            m[10] = c*m[10]-s*mv8;
         }

         /*================= Drawing ===========================*/
         var time_old = 0;

         var animate = function(time) {

            var dt = time-time_old;
            rotateZ(mov_matrix, dt*0.005);//time
            rotateY(mov_matrix, dt*0.002);
            rotateX(mov_matrix, dt*0.003);
            time_old = time;

            gl.enable(gl.DEPTH_TEST);
            gl.depthFunc(gl.LEQUAL);
            gl.clearColor(0.5, 0.5, 0.5, 0.9);
            gl.clearDepth(1.0);

            gl.viewport(0.0, 0.0, canvas.width, canvas.height);
            gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);
            gl.uniformMatrix4fv(Pmatrix, false, proj_matrix);
            gl.uniformMatrix4fv(Vmatrix, false, view_matrix);
            gl.uniformMatrix4fv(Mmatrix, false, mov_matrix);
            gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);
            gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT, 0);

            window.requestAnimationFrame(animate);
         }
         animate(0);
      </script>
   </body>
</html>

この例を実行すると、次の出力が生成されます-

この章では、マウスコントロールを使用して回転できる3D立方体を描画する方法を示す例を取り上げます。

例–インタラクティブキューブを描く

次のプログラムは、マウスコントロールを使用して立方体を回転させる方法を示しています-

<!doctype html>
<html>
   <body>
      <canvas width = "570" height = "570" id = "my_Canvas"></canvas>

      <script>
         /*============= Creating a canvas ======================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl');

         /*========== Defining and storing the geometry ==========*/

         var vertices = [
            -1,-1,-1, 1,-1,-1, 1, 1,-1, -1, 1,-1,
            -1,-1, 1, 1,-1, 1, 1, 1, 1, -1, 1, 1,
            -1,-1,-1, -1, 1,-1, -1, 1, 1, -1,-1, 1,
            1,-1,-1, 1, 1,-1, 1, 1, 1, 1,-1, 1,
            -1,-1,-1, -1,-1, 1, 1,-1, 1, 1,-1,-1,
            -1, 1,-1, -1, 1, 1, 1, 1, 1, 1, 1,-1, 
         ];

         var colors = [
            5,3,7, 5,3,7, 5,3,7, 5,3,7,
            1,1,3, 1,1,3, 1,1,3, 1,1,3,
            0,0,1, 0,0,1, 0,0,1, 0,0,1,
            1,0,0, 1,0,0, 1,0,0, 1,0,0,
            1,1,0, 1,1,0, 1,1,0, 1,1,0,
            0,1,0, 0,1,0, 0,1,0, 0,1,0 
         ];

         var indices = [
            0,1,2, 0,2,3, 4,5,6, 4,6,7,
            8,9,10, 8,10,11, 12,13,14, 12,14,15,
            16,17,18, 16,18,19, 20,21,22, 20,22,23 
         ];

         // Create and store data into vertex buffer
         var vertex_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Create and store data into color buffer
         var color_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(colors), gl.STATIC_DRAW);

         // Create and store data into index buffer
         var index_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);
         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);

         /*=================== SHADERS =================== */

         var vertCode = 'attribute vec3 position;'+
            'uniform mat4 Pmatrix;'+
            'uniform mat4 Vmatrix;'+
            'uniform mat4 Mmatrix;'+
            'attribute vec3 color;'+//the color of the point
            'varying vec3 vColor;'+
            'void main(void) { '+//pre-built function
               'gl_Position = Pmatrix*Vmatrix*Mmatrix*vec4(position, 1.);'+
               'vColor = color;'+
            '}';

         var fragCode = 'precision mediump float;'+
            'varying vec3 vColor;'+
            'void main(void) {'+
               'gl_FragColor = vec4(vColor, 1.);'+
            '}';

         var vertShader = gl.createShader(gl.VERTEX_SHADER);
         gl.shaderSource(vertShader, vertCode);
         gl.compileShader(vertShader);

         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);
         gl.shaderSource(fragShader, fragCode);
         gl.compileShader(fragShader);

         var shaderprogram = gl.createProgram();
         gl.attachShader(shaderprogram, vertShader);
         gl.attachShader(shaderprogram, fragShader);
         gl.linkProgram(shaderprogram);

         /*======== Associating attributes to vertex shader =====*/
         var _Pmatrix = gl.getUniformLocation(shaderprogram, "Pmatrix");
         var _Vmatrix = gl.getUniformLocation(shaderprogram, "Vmatrix");
         var _Mmatrix = gl.getUniformLocation(shaderprogram, "Mmatrix");

         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         var _position = gl.getAttribLocation(shaderprogram, "position");
         gl.vertexAttribPointer(_position, 3, gl.FLOAT, false,0,0);
         gl.enableVertexAttribArray(_position);

         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         var _color = gl.getAttribLocation(shaderprogram, "color");
         gl.vertexAttribPointer(_color, 3, gl.FLOAT, false,0,0) ;
         gl.enableVertexAttribArray(_color);
         gl.useProgram(shaderprogram);

         /*==================== MATRIX ====================== */

         function get_projection(angle, a, zMin, zMax) {
            var ang = Math.tan((angle*.5)*Math.PI/180);//angle*.5
            return [
               0.5/ang, 0 , 0, 0,
               0, 0.5*a/ang, 0, 0,
               0, 0, -(zMax+zMin)/(zMax-zMin), -1,
               0, 0, (-2*zMax*zMin)/(zMax-zMin), 0 
			   ];
         }

         var proj_matrix = get_projection(40, canvas.width/canvas.height, 1, 100);
         var mo_matrix = [ 1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1 ];
         var view_matrix = [ 1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1 ];

         view_matrix[14] = view_matrix[14]-6;

         /*================= Mouse events ======================*/

         var AMORTIZATION = 0.95;
         var drag = false;
         var old_x, old_y;
         var dX = 0, dY = 0;

         var mouseDown = function(e) {
            drag = true;
            old_x = e.pageX, old_y = e.pageY;
            e.preventDefault();
            return false;
         };

         var mouseUp = function(e){
            drag = false;
         };

         var mouseMove = function(e) {
            if (!drag) return false;
            dX = (e.pageX-old_x)*2*Math.PI/canvas.width,
            dY = (e.pageY-old_y)*2*Math.PI/canvas.height;
            THETA+= dX;
            PHI+=dY;
            old_x = e.pageX, old_y = e.pageY;
            e.preventDefault();
         };

         canvas.addEventListener("mousedown", mouseDown, false);
         canvas.addEventListener("mouseup", mouseUp, false);
         canvas.addEventListener("mouseout", mouseUp, false);
         canvas.addEventListener("mousemove", mouseMove, false);

         /*=========================rotation================*/

         function rotateX(m, angle) {
            var c = Math.cos(angle);
            var s = Math.sin(angle);
            var mv1 = m[1], mv5 = m[5], mv9 = m[9];

            m[1] = m[1]*c-m[2]*s;
            m[5] = m[5]*c-m[6]*s;
            m[9] = m[9]*c-m[10]*s;

            m[2] = m[2]*c+mv1*s;
            m[6] = m[6]*c+mv5*s;
            m[10] = m[10]*c+mv9*s;
         }

         function rotateY(m, angle) {
            var c = Math.cos(angle);
            var s = Math.sin(angle);
            var mv0 = m[0], mv4 = m[4], mv8 = m[8];

            m[0] = c*m[0]+s*m[2];
            m[4] = c*m[4]+s*m[6];
            m[8] = c*m[8]+s*m[10];

            m[2] = c*m[2]-s*mv0;
            m[6] = c*m[6]-s*mv4;
            m[10] = c*m[10]-s*mv8;
         }

         /*=================== Drawing =================== */

         var THETA = 0,
         PHI = 0;
         var time_old = 0;

         var animate = function(time) {
            var dt = time-time_old;

            if (!drag) {
               dX *= AMORTIZATION, dY*=AMORTIZATION;
               THETA+=dX, PHI+=dY;
            }

            //set model matrix to I4

            mo_matrix[0] = 1, mo_matrix[1] = 0, mo_matrix[2] = 0,
            mo_matrix[3] = 0,

            mo_matrix[4] = 0, mo_matrix[5] = 1, mo_matrix[6] = 0,
            mo_matrix[7] = 0,

            mo_matrix[8] = 0, mo_matrix[9] = 0, mo_matrix[10] = 1,
            mo_matrix[11] = 0,

            mo_matrix[12] = 0, mo_matrix[13] = 0, mo_matrix[14] = 0,
            mo_matrix[15] = 1;

            rotateY(mo_matrix, THETA);
            rotateX(mo_matrix, PHI);

            time_old = time; 
            gl.enable(gl.DEPTH_TEST);

            // gl.depthFunc(gl.LEQUAL);

            gl.clearColor(0.5, 0.5, 0.5, 0.9);
            gl.clearDepth(1.0);
            gl.viewport(0.0, 0.0, canvas.width, canvas.height);
            gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);

            gl.uniformMatrix4fv(_Pmatrix, false, proj_matrix);
            gl.uniformMatrix4fv(_Vmatrix, false, view_matrix);
            gl.uniformMatrix4fv(_Mmatrix, false, mo_matrix);

            gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);
            gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT, 0);

            window.requestAnimationFrame(animate);
         }
         animate(0);
      </script>
   </body>
</html>

この例を実行すると、次の出力が生成されます-