🎯

threejs-loaders

🎯Skill

from toilahuongg/shopify-agents-kit

VibeIndex|
What it does

Loads and manages 3D assets in Three.js, handling GLTF models, textures, and environments with comprehensive loading progress tracking.

πŸ“¦

Part of

toilahuongg/shopify-agents-kit(35 items)

threejs-loaders

Installation

npm installInstall npm package
npm install -g shopify-cc-kit
npm installInstall npm package
npm install shopify-cc-kit
πŸ“– Extracted from docs: toilahuongg/shopify-agents-kit
1Installs
-
AddedFeb 4, 2026

Skill Details

SKILL.md

Three.js asset loading - GLTF, textures, images, models, async patterns. Use when loading 3D models, textures, HDR environments, or managing loading progress.

Overview

# Three.js Loaders

Quick Start

```javascript

import * as THREE from "three";

import { GLTFLoader } from "three/addons/loaders/GLTFLoader.js";

// Load GLTF model

const loader = new GLTFLoader();

loader.load("model.glb", (gltf) => {

scene.add(gltf.scene);

});

// Load texture

const textureLoader = new THREE.TextureLoader();

const texture = textureLoader.load("texture.jpg");

```

LoadingManager

Coordinate multiple loaders and track progress.

```javascript

const manager = new THREE.LoadingManager();

// Callbacks

manager.onStart = (url, loaded, total) => {

console.log(Started loading: ${url});

};

manager.onLoad = () => {

console.log("All assets loaded!");

startGame();

};

manager.onProgress = (url, loaded, total) => {

const progress = (loaded / total) * 100;

console.log(Loading: ${progress.toFixed(1)}%);

updateProgressBar(progress);

};

manager.onError = (url) => {

console.error(Error loading: ${url});

};

// Use manager with loaders

const textureLoader = new THREE.TextureLoader(manager);

const gltfLoader = new GLTFLoader(manager);

// Load assets

textureLoader.load("texture1.jpg");

textureLoader.load("texture2.jpg");

gltfLoader.load("model.glb");

// onLoad fires when ALL are complete

```

Texture Loading

TextureLoader

```javascript

const loader = new THREE.TextureLoader();

// Callback style

loader.load(

"texture.jpg",

(texture) => {

// onLoad

material.map = texture;

material.needsUpdate = true;

},

undefined, // onProgress - not supported for image loading

(error) => {

// onError

console.error("Error loading texture", error);

},

);

// Synchronous (returns texture, loads async)

const texture = loader.load("texture.jpg");

material.map = texture;

```

Texture Configuration

```javascript

const texture = loader.load("texture.jpg", (tex) => {

// Color space (important for color accuracy)

tex.colorSpace = THREE.SRGBColorSpace; // For color/albedo maps

// tex.colorSpace = THREE.LinearSRGBColorSpace; // For data maps (normal, roughness)

// Wrapping

tex.wrapS = THREE.RepeatWrapping;

tex.wrapT = THREE.RepeatWrapping;

// ClampToEdgeWrapping, RepeatWrapping, MirroredRepeatWrapping

// Repeat/offset

tex.repeat.set(2, 2);

tex.offset.set(0.5, 0.5);

tex.rotation = Math.PI / 4;

tex.center.set(0.5, 0.5);

// Filtering

tex.minFilter = THREE.LinearMipmapLinearFilter; // Default

tex.magFilter = THREE.LinearFilter; // Default

// NearestFilter - pixelated

// LinearFilter - smooth

// LinearMipmapLinearFilter - smooth with mipmaps

// Anisotropic filtering (sharper at angles)

tex.anisotropy = renderer.capabilities.getMaxAnisotropy();

// Flip Y (usually true for standard textures)

tex.flipY = true;

tex.needsUpdate = true;

});

```

CubeTextureLoader

For environment maps and skyboxes.

```javascript

const loader = new THREE.CubeTextureLoader();

// Load 6 faces

const cubeTexture = loader.load([

"px.jpg",

"nx.jpg", // positive/negative X

"py.jpg",

"ny.jpg", // positive/negative Y

"pz.jpg",

"nz.jpg", // positive/negative Z

]);

// Use as background

scene.background = cubeTexture;

// Use as environment map

scene.environment = cubeTexture;

material.envMap = cubeTexture;

```

HDR/EXR Loading

```javascript

import { RGBELoader } from "three/addons/loaders/RGBELoader.js";

import { EXRLoader } from "three/addons/loaders/EXRLoader.js";

// HDR

const rgbeLoader = new RGBELoader();

rgbeLoader.load("environment.hdr", (texture) => {

texture.mapping = THREE.EquirectangularReflectionMapping;

scene.environment = texture;

scene.background = texture;

});

// EXR

const exrLoader = new EXRLoader();

exrLoader.load("environment.exr", (texture) => {

texture.mapping = THREE.EquirectangularReflectionMapping;

scene.environment = texture;

});

```

PMREMGenerator

Generate prefiltered environment maps for PBR.

```javascript

import { RGBELoader } from "three/addons/loaders/RGBELoader.js";

const pmremGenerator = new THREE.PMREMGenerator(renderer);

pmremGenerator.compileEquirectangularShader();

new RGBELoader().load("environment.hdr", (texture) => {

const envMap = pmremGenerator.fromEquirectangular(texture).texture;

scene.environment = envMap;

scene.background = envMap;

texture.dispose();

pmremGenerator.dispose();

});

```

GLTF/GLB Loading

The most common 3D format for web.

```javascript

import { GLTFLoader } from "three/addons/loaders/GLTFLoader.js";

const loader = new GLTFLoader();

loader.load("model.glb", (gltf) => {

// The loaded scene

const model = gltf.scene;

scene.add(model);

// Animations

const animations = gltf.animations;

if (animations.length > 0) {

const mixer = new THREE.AnimationMixer(model);

animations.forEach((clip) => {

mixer.clipAction(clip).play();

});

}

// Cameras (if any)

const cameras = gltf.cameras;

// Asset info

console.log(gltf.asset); // Version, generator, etc.

// User data from Blender/etc

console.log(gltf.userData);

});

```

GLTF with Draco Compression

```javascript

import { GLTFLoader } from "three/addons/loaders/GLTFLoader.js";

import { DRACOLoader } from "three/addons/loaders/DRACOLoader.js";

const dracoLoader = new DRACOLoader();

dracoLoader.setDecoderPath(

"https://www.gstatic.com/draco/versioned/decoders/1.5.6/",

);

dracoLoader.preload();

const gltfLoader = new GLTFLoader();

gltfLoader.setDRACOLoader(dracoLoader);

gltfLoader.load("compressed-model.glb", (gltf) => {

scene.add(gltf.scene);

});

```

GLTF with KTX2 Textures

```javascript

import { GLTFLoader } from "three/addons/loaders/GLTFLoader.js";

import { KTX2Loader } from "three/addons/loaders/KTX2Loader.js";

const ktx2Loader = new KTX2Loader();

ktx2Loader.setTranscoderPath(

"https://cdn.jsdelivr.net/npm/three@0.160.0/examples/jsm/libs/basis/",

);

ktx2Loader.detectSupport(renderer);

const gltfLoader = new GLTFLoader();

gltfLoader.setKTX2Loader(ktx2Loader);

gltfLoader.load("model-with-ktx2.glb", (gltf) => {

scene.add(gltf.scene);

});

```

Process GLTF Content

```javascript

loader.load("model.glb", (gltf) => {

const model = gltf.scene;

// Enable shadows

model.traverse((child) => {

if (child.isMesh) {

child.castShadow = true;

child.receiveShadow = true;

}

});

// Find specific mesh

const head = model.getObjectByName("Head");

// Adjust materials

model.traverse((child) => {

if (child.isMesh && child.material) {

child.material.envMapIntensity = 0.5;

}

});

// Center and scale

const box = new THREE.Box3().setFromObject(model);

const center = box.getCenter(new THREE.Vector3());

const size = box.getSize(new THREE.Vector3());

model.position.sub(center);

const maxDim = Math.max(size.x, size.y, size.z);

model.scale.setScalar(1 / maxDim);

scene.add(model);

});

```

Other Model Formats

OBJ + MTL

```javascript

import { OBJLoader } from "three/addons/loaders/OBJLoader.js";

import { MTLLoader } from "three/addons/loaders/MTLLoader.js";

const mtlLoader = new MTLLoader();

mtlLoader.load("model.mtl", (materials) => {

materials.preload();

const objLoader = new OBJLoader();

objLoader.setMaterials(materials);

objLoader.load("model.obj", (object) => {

scene.add(object);

});

});

```

FBX

```javascript

import { FBXLoader } from "three/addons/loaders/FBXLoader.js";

const loader = new FBXLoader();

loader.load("model.fbx", (object) => {

// FBX often has large scale

object.scale.setScalar(0.01);

// Animations

const mixer = new THREE.AnimationMixer(object);

object.animations.forEach((clip) => {

mixer.clipAction(clip).play();

});

scene.add(object);

});

```

STL

```javascript

import { STLLoader } from "three/addons/loaders/STLLoader.js";

const loader = new STLLoader();

loader.load("model.stl", (geometry) => {

const material = new THREE.MeshStandardMaterial({ color: 0x888888 });

const mesh = new THREE.Mesh(geometry, material);

scene.add(mesh);

});

```

PLY

```javascript

import { PLYLoader } from "three/addons/loaders/PLYLoader.js";

const loader = new PLYLoader();

loader.load("model.ply", (geometry) => {

geometry.computeVertexNormals();

const material = new THREE.MeshStandardMaterial({ vertexColors: true });

const mesh = new THREE.Mesh(geometry, material);

scene.add(mesh);

});

```

Async/Promise Loading

Promisified Loader

```javascript

function loadModel(url) {

return new Promise((resolve, reject) => {

loader.load(url, resolve, undefined, reject);

});

}

// Usage

async function init() {

try {

const gltf = await loadModel("model.glb");

scene.add(gltf.scene);

} catch (error) {

console.error("Failed to load model:", error);

}

}

```

Load Multiple Assets

```javascript

async function loadAssets() {

const [modelGltf, envTexture, colorTexture] = await Promise.all([

loadGLTF("model.glb"),

loadRGBE("environment.hdr"),

loadTexture("color.jpg"),

]);

scene.add(modelGltf.scene);

scene.environment = envTexture;

material.map = colorTexture;

}

// Helper functions

function loadGLTF(url) {

return new Promise((resolve, reject) => {

new GLTFLoader().load(url, resolve, undefined, reject);

});

}

function loadRGBE(url) {

return new Promise((resolve, reject) => {

new RGBELoader().load(

url,

(texture) => {

texture.mapping = THREE.EquirectangularReflectionMapping;

resolve(texture);

},

undefined,

reject,

);

});

}

function loadTexture(url) {

return new Promise((resolve, reject) => {

new THREE.TextureLoader().load(url, resolve, undefined, reject);

});

}

```

Caching

Built-in Cache

```javascript

// Enable cache

THREE.Cache.enabled = true;

// Clear cache

THREE.Cache.clear();

// Manual cache management

THREE.Cache.add("key", data);

THREE.Cache.get("key");

THREE.Cache.remove("key");

```

Custom Asset Manager

```javascript

class AssetManager {

constructor() {

this.textures = new Map();

this.models = new Map();

this.gltfLoader = new GLTFLoader();

this.textureLoader = new THREE.TextureLoader();

}

async loadTexture(key, url) {

if (this.textures.has(key)) {

return this.textures.get(key);

}

const texture = await new Promise((resolve, reject) => {

this.textureLoader.load(url, resolve, undefined, reject);

});

this.textures.set(key, texture);

return texture;

}

async loadModel(key, url) {

if (this.models.has(key)) {

return this.models.get(key).clone();

}

const gltf = await new Promise((resolve, reject) => {

this.gltfLoader.load(url, resolve, undefined, reject);

});

this.models.set(key, gltf.scene);

return gltf.scene.clone();

}

dispose() {

this.textures.forEach((t) => t.dispose());

this.textures.clear();

this.models.clear();

}

}

// Usage

const assets = new AssetManager();

const texture = await assets.loadTexture("brick", "brick.jpg");

const model = await assets.loadModel("tree", "tree.glb");

```

Loading from Different Sources

Data URL / Base64

```javascript

const loader = new THREE.TextureLoader();

const texture = loader.load("...");

```

Blob URL

```javascript

async function loadFromBlob(blob) {

const url = URL.createObjectURL(blob);

const texture = await loadTexture(url);

URL.revokeObjectURL(url);

return texture;

}

```

ArrayBuffer

```javascript

// From fetch

const response = await fetch("model.glb");

const buffer = await response.arrayBuffer();

// Parse with loader

const loader = new GLTFLoader();

loader.parse(buffer, "", (gltf) => {

scene.add(gltf.scene);

});

```

Custom Path/URL

```javascript

// Set base path

loader.setPath("assets/models/");

loader.load("model.glb"); // Loads from assets/models/model.glb

// Set resource path (for textures referenced in model)

loader.setResourcePath("assets/textures/");

// Custom URL modifier

manager.setURLModifier((url) => {

return https://cdn.example.com/${url};

});

```

Error Handling

```javascript

// Graceful fallback

async function loadWithFallback(primaryUrl, fallbackUrl) {

try {

return await loadModel(primaryUrl);

} catch (error) {

console.warn(Primary failed, trying fallback: ${error});

return await loadModel(fallbackUrl);

}

}

// Retry logic

async function loadWithRetry(url, maxRetries = 3) {

for (let i = 0; i < maxRetries; i++) {

try {

return await loadModel(url);

} catch (error) {

if (i === maxRetries - 1) throw error;

await new Promise((r) => setTimeout(r, 1000 * (i + 1)));

}

}

}

// Timeout

async function loadWithTimeout(url, timeout = 30000) {

const controller = new AbortController();

const timeoutId = setTimeout(() => controller.abort(), timeout);

try {

const response = await fetch(url, { signal: controller.signal });

clearTimeout(timeoutId);

return response;

} catch (error) {

if (error.name === "AbortError") {

throw new Error("Loading timed out");

}

throw error;

}

}

```

Performance Tips

  1. Use compressed formats: DRACO for geometry, KTX2/Basis for textures
  2. Load progressively: Show placeholders while loading
  3. Lazy load: Only load what's needed
  4. Use CDN: Faster asset delivery
  5. Enable cache: THREE.Cache.enabled = true

```javascript

// Progressive loading with placeholder

const placeholder = new THREE.Mesh(

new THREE.BoxGeometry(1, 1, 1),

new THREE.MeshBasicMaterial({ wireframe: true }),

);

scene.add(placeholder);

loadModel("model.glb").then((gltf) => {

scene.remove(placeholder);

scene.add(gltf.scene);

});

```

See Also

  • threejs-textures - Texture configuration
  • threejs-animation - Playing loaded animations
  • threejs-materials - Material from loaded models

More from this repository10

🎯
shopify-polaris-icons🎯Skill

Provides comprehensive guidance and examples for integrating and using Shopify Polaris Icons in commerce applications with accessibility and semantic color tones.

🎯
shopify-polaris-viz🎯Skill

Generates accessible, Shopify Admin-styled data visualizations using Polaris Viz library for React applications.

🎯
shopify-polaris-design🎯Skill

Generates Shopify Polaris-styled React components and design elements for creating consistent and professional e-commerce interfaces.

🎯
shopify-pos🎯Skill

Enables developers to create custom UI extensions for Shopify Point of Sale (POS), integrating specialized functionality across various retail interface screens.

🎯
brainstorm🎯Skill

Guides users through creative brainstorming by applying structured techniques like SCAMPER and Design Thinking to generate innovative solutions.

🎯
threejs-animation🎯Skill

Enables precise and flexible object animations in Three.js, supporting keyframe, skeletal, and procedural motion with advanced blending and playback control.

🎯
shopify-webhooks🎯Skill

Handles Shopify webhook verification, processing, and registration for real-time event synchronization with secure HMAC authentication.

🎯
shopify-extensions🎯Skill

Guides developers through building and integrating Shopify Extensions across Admin, Checkout, Theme, and Post-purchase interfaces using latest CLI and APIs.

🎯
design🎯Skill

Generates design-related tasks and recommendations for Shopify store aesthetics, layout, and visual branding using AI-powered analysis.

🎯
algorithmic-art🎯Skill

Generates algorithmic art philosophies and p5.js code sketches using computational aesthetics, seeded randomness, and generative design principles.