はじめての「WebGPU」で背景色を塗りつぶしてみよう

2025年6月25日(水)
大西 武 (オオニシ タケシ)
第4回の今回は、WebGPUを初期化して<canvas>タグを真っ黒な背景色で塗りつぶしてみる解説をします。

真っ黒な背景色で塗りつぶしてみよう

<canvas>タグが3Dを描画する矩形です。その背景色を黒色に設定すると、図3のように<canvas>タグ全体が真っ黒になります。次のサンプルコード「lib」→「UltraMotion3D.js」のように、背景色は色テクスチャに塗ります。それを<canvas>タグのビューに描いているわけです。

さらに、ポリゴンの前後関係を決める「Zバッファ」もデプステクスチャで作成します。Zバッファとはポリゴンの頂点を囲んでマテリアルを1ピクセルずつ塗っていくとき、そのピクセルの奥行き(Z座標)が以前に描いた他のポリゴンのピクセルより手前にあればZ座標を上書きしてピクセルを塗るものです。その値の型が'depth24plus'という深度24bitなわけです。Zバッファがないと奥のポリゴンが手前に描画されるなど、おかしな描画になったりします。

図3:真っ黒な<canvas>タグ

・サンプルコード「lib」→「UltraMotion3D.js」
var _passEncoder = null;

async function initWebGPU(canvas) {
  if (!navigator.gpu) {
    throw Error('WebGPU not supported.');
  }
  const adapter = await navigator.gpu.requestAdapter();
  if (!adapter) {
    throw Error('Could not request WebGPU adapter.');
  }
  _device = await adapter.requestDevice();
  _canvas = document.getElementById(canvas);
  _context = _canvas.getContext('webgpu');
  _presentationFormat = navigator.gpu.getPreferredCanvasFormat();
  _context.configure({
    device: _device,
    format: _presentationFormat,
    alphaMode: 'premultiplied',
  });
  resize();
  everyFrame();
}

function everyFrame() {
  _commandEncoder = _device.createCommandEncoder();
  const textureView = _context.getCurrentTexture().createView();
  _renderPassDescriptor = {
    colorAttachments: [
      {
        view: textureView,
        clearValue: { r: 0.0, g: 0.0, b: 0.0, a: 1.0 },
        loadOp: 'clear',
        storeOp: 'store',
      },
    ],
    depthStencilAttachment: {
      view: _depthTexture.createView(),
      depthClearValue: 1.0,
      depthLoadOp: 'clear',
      depthStoreOp: 'store',
    },
  };
  _passEncoder = _commandEncoder.beginRenderPass(_renderPassDescriptor);
  _passEncoder.end();
  _device.queue.submit([_commandEncoder.finish()]);
}

function resize() {
  _depthTexture = _device.createTexture({
    size: [_canvas.width, _canvas.height],
    format: 'depth24plus',
    usage: GPUTextureUsage.RENDER_ATTACHMENT,
  });
}

【サンプルコードの解説】
「everyFrame」関数は毎フレーム呼ばれるためこのような名前を付けましたが、今回はまだ最初の1回しか呼ばれません。
GPUに送る命令形式に変換するコマンドエンコーダーを作成して「_commandEncoder」変数に代入します。
テクスチャを作成し「textureView」変数に代入します。
レンダリングの通過を記述する「_renderPassDescriptor」変数を色を塗る色テクスチャとZバッファのデプステクスチャで作成します。色テクスチャもデプステクスチャも単なるテクスチャデータです。
「beginRenderPass」メソッドでレンダリングの通過を開始します。
「end」メソッドでレンダリングの通過を終了します。つまり最初に背景を塗りつぶした他にはポリゴンなどジオメトリを何も描画していません。
デバイスのキュー(列)に「submit」メソッドで1コマが終了したコマンドエンコーダーを登録します。
「resize」関数でキャンバスと同じ幅と高さの、Zバッファ用のテクスチャを深度24bitで作成します。

【コラム】「プログラミングと株式投資の共通点」

プログラミングで一度もバグを出さずにプログラムを完成させることはほぼ不可能でしょう。それは株式投資にも似ているのではないでしょうか。

株式投資はゲームのように上手く行くこともあれば、失敗することもあります。「いついつまでにいくら増やしたい」と思っても、到底上手く行くものではありません。とは言え、株式投資における資金を金額ではなく「得点」として見れば、マネーゲームとして楽しめるのではないかと思います。

誰だって、一度もゲームオーバーにならずにゲームをクリアできることはまずないでしょう。勝ったり負けたり。負けてる人でも株式投資を続けて行けるのは、ゲームとして楽しんでいるからではないでしょうか。

おわりに

今回は、WebGPUの導入として<canvas>タグの背景色を真っ黒に塗りつぶしてみました。塗りつぶす以外には何も描画していません。そこで次回は、シェーダーを使って三角形を1個だけ描画するための実装を解説します。

著者
大西 武 (オオニシ タケシ)
1975年香川県生まれ。大阪大学経済学部経営学科中退。プログラミング入門書など30冊以上を商業出版する作家。Microsoftで大賞やNTTドコモでグランプリなど20回以上全国区のコンテストに入賞するアーティスト。オリジナルの間違い探し「3Dクイズ」が全国放送のTVで約10回出題。
https://profile.vixar.jp

連載バックナンバー

開発言語技術解説
第4回

はじめての「WebGPU」で背景色を塗りつぶしてみよう

2025/6/25
第4回の今回は、WebGPUを初期化してタグを真っ黒な背景色で塗りつぶしてみる解説をします。
開発言語技術解説
第3回

WebGPUライブラリ「UltraMotion3D」で3DのサンプルWebコンテンツを動かしてみよう

2025/6/5
第3回の今回は、WebGPUライブラリ「UltraMotion3D」を使った3DのサンプルWebコンテンツの動かし方と「XAMPP」の使い方を解説します。
開発言語技術解説
第2回

WebGPUを使うための「HTML5+CSS+JavaScript」の文法を学ぼう

2025/5/20
第2回の今回は、WebGPUコンテンツを作る上で、最低限知っておくべきHTML5+CSS+JavaScriptの文法と、それを使った簡単なサンプルを解説します。

Think ITメルマガ会員登録受付中

Think ITでは、技術情報が詰まったメールマガジン「Think IT Weekly」の配信サービスを提供しています。メルマガ会員登録を済ませれば、メルマガだけでなく、さまざまな限定特典を入手できるようになります。

Think ITメルマガ会員のサービス内容を見る

他にもこの記事が読まれています