Agent skill
webgl-expert
Expert guide for WebGL API development including 3D graphics, shaders (GLSL), rendering pipeline, textures, buffers, performance optimization, and canvas rendering. Use when working with WebGL, 3D graphics, canvas rendering, shaders, GPU programming, or when user mentions WebGL, OpenGL ES, GLSL, vertex shaders, fragment shaders, texture mapping, or 3D web graphics.
Install this agent skill to your Project
npx add-skill https://github.com/ronnycoding/.claude/tree/main/skills/webgl-expert
SKILL.md
WebGL Expert
Expert guide for WebGL (Web Graphics Library) API development, covering both WebGL 1.0 and WebGL 2.0 for high-performance 2D and 3D graphics rendering in web browsers.
Overview
WebGL is a JavaScript API that enables hardware-accelerated 3D graphics rendering within HTML canvas elements without requiring plugins. It closely conforms to OpenGL ES 2.0 (WebGL 1.0) and OpenGL ES 3.0 (WebGL 2.0) standards.
Key capabilities:
- Hardware-accelerated 2D and 3D rendering
- Programmable shader pipeline (GLSL)
- Texture mapping and advanced materials
- Lighting and transformation systems
- High-performance graphics for games and visualizations
- Cross-platform compatibility (all modern browsers)
Core Interfaces
WebGLRenderingContext (WebGL 1.0)
The foundational interface for WebGL operations, obtained via canvas context:
const canvas = document.querySelector('canvas');
const gl = canvas.getContext('webgl') || canvas.getContext('experimental-webgl');
if (!gl) {
console.error('WebGL not supported');
}
WebGL2RenderingContext (WebGL 2.0)
Enhanced interface with advanced features:
const gl = canvas.getContext('webgl2');
if (!gl) {
console.log('WebGL 2 not supported, falling back to WebGL 1');
gl = canvas.getContext('webgl');
}
WebGL 2 exclusive features:
- 3D textures
- Sampler objects
- Uniform Buffer Objects (UBO)
- Transform Feedback
- Vertex Array Objects (VAO) - core feature
- Instanced rendering
- Multiple render targets
- Integer textures and attributes
- Query objects
- Occlusion queries
Rendering Pipeline
1. Shader Creation and Compilation
Shaders are programs written in GLSL (OpenGL Shading Language) that run on the GPU:
Vertex Shader - Processes each vertex:
attribute vec3 aPosition;
attribute vec2 aTexCoord;
uniform mat4 uModelViewProjection;
varying vec2 vTexCoord;
void main() {
gl_Position = uModelViewProjection * vec4(aPosition, 1.0);
vTexCoord = aTexCoord;
}
Fragment Shader - Determines pixel colors:
precision mediump float;
varying vec2 vTexCoord;
uniform sampler2D uTexture;
void main() {
gl_FragColor = texture2D(uTexture, vTexCoord);
}
JavaScript shader setup:
function createShader(gl, type, source) {
const shader = gl.createShader(type);
gl.shaderSource(shader, source);
gl.compileShader(shader);
if (!gl.getShaderParameter(shader, gl.COMPILE_STATUS)) {
console.error('Shader compilation error:', gl.getShaderInfoLog(shader));
gl.deleteShader(shader);
return null;
}
return shader;
}
function createProgram(gl, vertexShader, fragmentShader) {
const program = gl.createProgram();
gl.attachShader(program, vertexShader);
gl.attachShader(program, fragmentShader);
gl.linkProgram(program);
if (!gl.getProgramParameter(program, gl.LINK_STATUS)) {
console.error('Program linking error:', gl.getProgramInfoLog(program));
gl.deleteProgram(program);
return null;
}
return program;
}
2. Buffer Management
Buffers store vertex data (positions, colors, normals, texture coordinates):
// Create buffer
const positionBuffer = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer);
// Upload data
const positions = new Float32Array([
-1.0, -1.0, 0.0,
1.0, -1.0, 0.0,
0.0, 1.0, 0.0
]);
gl.bufferData(gl.ARRAY_BUFFER, positions, gl.STATIC_DRAW);
// Set up attribute pointer
const positionLocation = gl.getAttribLocation(program, 'aPosition');
gl.enableVertexAttribArray(positionLocation);
gl.vertexAttribPointer(positionLocation, 3, gl.FLOAT, false, 0, 0);
Buffer usage patterns:
gl.STATIC_DRAW- Data doesn't changegl.DYNAMIC_DRAW- Data changes occasionallygl.STREAM_DRAW- Data changes every frame
3. Texture Handling
function loadTexture(gl, url) {
const texture = gl.createTexture();
gl.bindTexture(gl.TEXTURE_2D, texture);
// Placeholder until image loads
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, 1, 1, 0, gl.RGBA, gl.UNSIGNED_BYTE,
new Uint8Array([255, 0, 255, 255]));
const image = new Image();
image.onload = () => {
gl.bindTexture(gl.TEXTURE_2D, texture);
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, image);
// Generate mipmaps if power of 2
if (isPowerOf2(image.width) && isPowerOf2(image.height)) {
gl.generateMipmap(gl.TEXTURE_2D);
} else {
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR);
}
};
image.src = url;
return texture;
}
function isPowerOf2(value) {
return (value & (value - 1)) === 0;
}
4. Rendering Loop
function render(gl, program) {
// Clear canvas
gl.clearColor(0.0, 0.0, 0.0, 1.0);
gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);
// Enable depth testing
gl.enable(gl.DEPTH_TEST);
gl.depthFunc(gl.LEQUAL);
// Use program
gl.useProgram(program);
// Set uniforms
const projectionMatrix = mat4.create();
mat4.perspective(projectionMatrix, Math.PI / 4, canvas.width / canvas.height, 0.1, 100.0);
const uniformLocation = gl.getUniformLocation(program, 'uModelViewProjection');
gl.uniformMatrix4fv(uniformLocation, false, projectionMatrix);
// Draw
gl.drawArrays(gl.TRIANGLES, 0, 3);
// Animation loop
requestAnimationFrame(() => render(gl, program));
}
Matrix Mathematics
WebGL uses column-major matrices for transformations. Recommended libraries:
- glMatrix - Fast matrix/vector operations
- three.js - High-level 3D library with built-in math
Common transformations:
// Model matrix (object transform)
const modelMatrix = mat4.create();
mat4.translate(modelMatrix, modelMatrix, [x, y, z]);
mat4.rotate(modelMatrix, modelMatrix, angle, [0, 1, 0]);
mat4.scale(modelMatrix, modelMatrix, [sx, sy, sz]);
// View matrix (camera)
const viewMatrix = mat4.create();
mat4.lookAt(viewMatrix, eyePosition, targetPosition, upVector);
// Projection matrix
const projectionMatrix = mat4.create();
mat4.perspective(projectionMatrix, fov, aspect, near, far);
// Combined MVP matrix
const mvpMatrix = mat4.create();
mat4.multiply(mvpMatrix, projectionMatrix, viewMatrix);
mat4.multiply(mvpMatrix, mvpMatrix, modelMatrix);
Performance Optimization
Best Practices
- Minimize state changes - Batch draw calls with similar state
- Use Vertex Array Objects (VAO) - Reduce attribute setup overhead
- Texture atlases - Combine multiple textures into one
- Instanced rendering - Draw many similar objects efficiently
- Frustum culling - Don't render objects outside view
- Level of Detail (LOD) - Use simpler models at distance
- Texture compression - Use compressed texture formats (DXT, ETC, ASTC)
- Minimize shader complexity - Keep fragment shaders simple
- Use uniform buffers (WebGL 2) - Efficient uniform data sharing
- Avoid CPU-GPU synchronization - Don't read back data frequently
Instanced Rendering (WebGL 2)
const ext = gl.getExtension('ANGLE_instanced_arrays'); // WebGL 1
// or use gl.drawArraysInstanced directly in WebGL 2
// Set up per-instance attribute
const instanceOffsetBuffer = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, instanceOffsetBuffer);
gl.bufferData(gl.ARRAY_BUFFER, offsetData, gl.STATIC_DRAW);
const offsetLocation = gl.getAttribLocation(program, 'aInstanceOffset');
gl.enableVertexAttribArray(offsetLocation);
gl.vertexAttribPointer(offsetLocation, 3, gl.FLOAT, false, 0, 0);
gl.vertexAttribDivisor(offsetLocation, 1); // Advance per instance
// Draw multiple instances
gl.drawArraysInstanced(gl.TRIANGLES, 0, vertexCount, instanceCount);
Extension System
Check for and use extensions to access advanced features:
function getExtension(gl, name) {
const ext = gl.getExtension(name);
if (!ext) {
console.warn(`Extension ${name} not supported`);
}
return ext;
}
// Common extensions
const anisotropic = getExtension(gl, 'EXT_texture_filter_anisotropic');
const floatTextures = getExtension(gl, 'OES_texture_float');
const depthTexture = getExtension(gl, 'WEBGL_depth_texture');
const drawBuffers = getExtension(gl, 'WEBGL_draw_buffers');
const loseContext = getExtension(gl, 'WEBGL_lose_context'); // for testing
Important extension categories:
- Texture formats: WEBGL_compressed_texture_s3tc, WEBGL_compressed_texture_etc
- Rendering: WEBGL_draw_buffers, EXT_blend_minmax, EXT_frag_depth
- Precision: OES_texture_float, OES_texture_half_float
- Instancing: ANGLE_instanced_arrays (WebGL 1)
- Debugging: WEBGL_debug_renderer_info, WEBGL_debug_shaders
Context Management
Context Loss Handling
canvas.addEventListener('webglcontextlost', (event) => {
event.preventDefault();
console.log('WebGL context lost');
cancelAnimationFrame(animationId);
}, false);
canvas.addEventListener('webglcontextrestored', () => {
console.log('WebGL context restored');
initWebGL(); // Recreate all resources
render();
}, false);
Context Creation Options
const gl = canvas.getContext('webgl2', {
alpha: false, // No alpha channel (better performance)
antialias: true, // Antialiasing (performance cost)
depth: true, // Depth buffer
stencil: false, // Stencil buffer
premultipliedAlpha: true, // Alpha premultiplication
preserveDrawingBuffer: false, // Keep buffer after render
powerPreference: 'high-performance', // GPU preference
failIfMajorPerformanceCaveat: false // Fallback to software
});
Common Patterns
Framebuffer Rendering (Render to Texture)
const framebuffer = gl.createFramebuffer();
gl.bindFramebuffer(gl.FRAMEBUFFER, framebuffer);
const targetTexture = gl.createTexture();
gl.bindTexture(gl.TEXTURE_2D, targetTexture);
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, width, height, 0, gl.RGBA, gl.UNSIGNED_BYTE, null);
gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, targetTexture, 0);
// Render to framebuffer
gl.bindFramebuffer(gl.FRAMEBUFFER, framebuffer);
gl.viewport(0, 0, width, height);
// ... render scene ...
// Render to canvas
gl.bindFramebuffer(gl.FRAMEBUFFER, null);
gl.viewport(0, 0, canvas.width, canvas.height);
Multiple Render Targets (WebGL 2)
const ext = gl.getExtension('WEBGL_draw_buffers'); // WebGL 1
// Fragment shader outputs to multiple targets
gl.drawBuffers([
gl.COLOR_ATTACHMENT0,
gl.COLOR_ATTACHMENT1,
gl.COLOR_ATTACHMENT2
]);
Common Pitfalls
- Not checking compilation/linking errors - Always check shader status
- Forgetting to enable attributes - Call
gl.enableVertexAttribArray() - Incorrect data types - Use
Float32Array,Uint16Array, etc. - Not handling context loss - Add event listeners
- Mixing WebGL 1 and 2 APIs - Check version compatibility
- Power-of-2 texture assumptions - Handle non-POT textures correctly
- Z-fighting - Insufficient depth buffer precision
- Coordinate system confusion - WebGL uses clip space [-1, 1]
- Premature optimization - Profile before optimizing
- Not clearing buffers - Call
gl.clear()each frame
Debugging Tools
- Browser DevTools - Check console for WebGL errors
- WebGL Inspector - Browser extension for frame capture
- Spector.js - WebGL debugging library
- gl.getError() - Check for runtime errors
- WEBGL_debug_shaders - Get translated shader source
// Error checking
const error = gl.getError();
if (error !== gl.NO_ERROR) {
console.error('WebGL error:', error);
}
Popular Libraries and Frameworks
- three.js - Comprehensive 3D library with scene graph
- Babylon.js - Game engine with physics and VR support
- PlayCanvas - Cloud-based game engine
- Pixi.js - Fast 2D WebGL renderer
- Phaser - 2D game framework
- regl - Functional WebGL wrapper
- twgl - Tiny WebGL helper library
- glMatrix - High-performance matrix/vector library
Learning Resources
- MDN WebGL Tutorial
- WebGL Fundamentals
- The Book of Shaders
- Shadertoy - Shader examples
- WebGL2 Fundamentals
Quick Reference
See reference.md for:
- Complete constant reference
- All WebGL methods
- GLSL built-in functions
- Extension compatibility matrix
See examples for:
- Basic triangle rendering
- Texture mapping
- Lighting models
- Advanced techniques
Version Compatibility
When supporting both WebGL 1 and 2:
function initWebGL(canvas) {
const gl = canvas.getContext('webgl2');
let version = 2;
if (!gl) {
gl = canvas.getContext('webgl');
version = 1;
console.log('Using WebGL 1');
}
// Feature detection
const hasVAO = version === 2 || gl.getExtension('OES_vertex_array_object');
const hasInstancing = version === 2 || gl.getExtension('ANGLE_instanced_arrays');
return { gl, version, hasVAO, hasInstancing };
}
Security Considerations
- Cross-origin textures - Use CORS properly
- Shader validation - Validate user-provided shader code
- Resource limits - Don't trust client-reported capabilities
- Timing attacks - Be aware of shader compilation timing
- Context fingerprinting - Users may block WebGL for privacy
When helping users with WebGL:
- Determine version - Check if WebGL 1 or 2 is needed
- Check requirements - Browser support, extensions needed
- Start simple - Basic rendering before advanced features
- Debug systematically - Check shaders, buffers, state in order
- Profile performance - Use browser tools to identify bottlenecks
- Consider libraries - Recommend three.js/Babylon.js for complex projects
- Validate inputs - Check for null contexts, compilation errors
- Handle context loss - Always implement recovery
- Optimize appropriately - Don't over-optimize early
- Test across devices - GPU capabilities vary significantly
Recommended Agent Skills
Expand your agent's capabilities with these related and highly-rated skills.
creating-financial-models
This skill provides an advanced financial modeling suite with DCF analysis, sensitivity testing, Monte Carlo simulations, and scenario planning for investment decisions
create-command
Guide for creating custom Claude Code slash commands with proper structure, argument handling, frontmatter configuration, and best practices. Use when the user wants to create slash commands, custom commands, reusable prompts, or mentions creating/designing/building commands.
connect-mcp-server
Guide for connecting MCP (Model Context Protocol) servers to Claude Code with HTTP, stdio, and SSE transports. Covers installation, configuration, authentication, environment variables, and security. Use when the user wants to connect MCP servers, add integrations, configure external services, or mentions MCP, servers, integrations, or external tools.
create-skill
Guide for creating well-structured Claude Code skills with proper YAML frontmatter, focused descriptions, and supporting files. Use when the user wants to create a new skill, build a custom skill, extend Claude's capabilities, or mentions creating/designing/building skills.
create-claude-plugin
Guide for creating Claude Code plugins that bundle skills, agents, commands, hooks, and MCP servers for distribution. Covers plugin structure, marketplace.json format, installation, testing, and publishing. Use when the user wants to create plugins, distribute skills/agents, build marketplaces, or mentions creating/packaging/publishing plugins.
create-hooks
Guide for creating Claude Code hooks with proper configuration, shell commands, event handling, and security practices. Use when the user wants to create hooks, automate workflows, add event handlers, format code automatically, protect files, log actions, or mentions creating/configuring/building hooks.
Didn't find tool you were looking for?