UNPKG

@google/model-viewer

Version:

Easily display interactive 3D models on the web and in AR!

315 lines (293 loc) 12.5 kB
/* @license * Copyright 2019 Google LLC. All Rights Reserved. * Licensed under the Apache License, Version 2.0 (the 'License'); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an 'AS IS' BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { BackSide, BoxBufferGeometry, CubeCamera, EquirectangularReflectionMapping, EventDispatcher, HalfFloatType, LinearEncoding, Mesh, NoBlending, NoToneMapping, RGBAFormat, Scene, ShaderMaterial, sRGBEncoding, TextureLoader, Vector3, WebGLCubeRenderTarget } from 'three'; import { RGBELoader } from 'three/examples/jsm/loaders/RGBELoader.js'; import { deserializeUrl, timePasses } from '../utilities.js'; import EnvironmentScene from './EnvironmentScene.js'; import EnvironmentSceneAlt from './EnvironmentSceneAlt.js'; const GENERATED_SIGMA = 0.04; // The maximum length of the blur for loop. Smaller sigmas will use fewer // samples and exit early, but not recompile the shader. const MAX_SAMPLES = 20; const HDR_FILE_RE = /\.hdr(\.js)?$/; const ldrLoader = new TextureLoader(); const hdrLoader = new RGBELoader(); hdrLoader.setDataType(HalfFloatType); export default class TextureUtils extends EventDispatcher { constructor(threeRenderer) { super(); this.threeRenderer = threeRenderer; this.generatedEnvironmentMap = null; this.generatedEnvironmentMapAlt = null; this.skyboxCache = new Map(); this.blurMaterial = null; this.blurScene = null; } async load(url, progressCallback = () => { }) { try { const isHDR = HDR_FILE_RE.test(url); const loader = isHDR ? hdrLoader : ldrLoader; const texture = await new Promise((resolve, reject) => loader.load(url, resolve, (event) => { progressCallback(event.loaded / event.total * 0.9); }, reject)); progressCallback(1.0); texture.name = url; texture.mapping = EquirectangularReflectionMapping; if (!isHDR) { texture.encoding = sRGBEncoding; } return texture; } finally { if (progressCallback) { progressCallback(1); } } } /** * Returns a { skybox, environmentMap } object with the targets/textures * accordingly. `skybox` is a WebGLRenderCubeTarget, and `environmentMap` * is a Texture from a WebGLRenderCubeTarget. */ async generateEnvironmentMapAndSkybox(skyboxUrl = null, environmentMapUrl = null, options = {}) { const { progressTracker } = options; const useAltEnvironment = environmentMapUrl === 'neutral'; if (useAltEnvironment === true) { environmentMapUrl = null; } environmentMapUrl = deserializeUrl(environmentMapUrl); let skyboxLoads = Promise.resolve(null); let environmentMapLoads; // If we have a skybox URL, attempt to load it as a cubemap if (!!skyboxUrl) { skyboxLoads = this.loadEquirectFromUrl(skyboxUrl, progressTracker); } if (!!environmentMapUrl) { // We have an available environment map URL environmentMapLoads = this.loadEquirectFromUrl(environmentMapUrl, progressTracker); } else if (!!skyboxUrl) { // Fallback to deriving the environment map from an available skybox environmentMapLoads = this.loadEquirectFromUrl(skyboxUrl, progressTracker); } else { // Fallback to generating the environment map environmentMapLoads = useAltEnvironment === true ? this.loadGeneratedEnvironmentMapAlt() : this.loadGeneratedEnvironmentMap(); } let [environmentMap, skybox] = await Promise.all([environmentMapLoads, skyboxLoads]); if (environmentMap == null) { throw new Error('Failed to load environment map.'); } return { environmentMap, skybox }; } /** * Loads an equirect Texture from a given URL, for use as a skybox. */ async loadEquirectFromUrl(url, progressTracker) { if (!this.skyboxCache.has(url)) { const progressCallback = progressTracker ? progressTracker.beginActivity() : () => { }; const skyboxMapLoads = this.load(url, progressCallback); this.skyboxCache.set(url, skyboxMapLoads); } return this.skyboxCache.get(url); } async GenerateEnvironmentMap(scene, name) { await timePasses(); const renderer = this.threeRenderer; const cubeTarget = new WebGLCubeRenderTarget(256, { generateMipmaps: false, type: HalfFloatType, format: RGBAFormat, encoding: LinearEncoding, depthBuffer: true }); const cubeCamera = new CubeCamera(0.1, 100, cubeTarget); const generatedEnvironmentMap = cubeCamera.renderTarget.texture; generatedEnvironmentMap.name = name; const outputEncoding = renderer.outputEncoding; const toneMapping = renderer.toneMapping; renderer.toneMapping = NoToneMapping; renderer.outputEncoding = LinearEncoding; cubeCamera.update(renderer, scene); this.blurCubemap(cubeTarget, GENERATED_SIGMA); renderer.toneMapping = toneMapping; renderer.outputEncoding = outputEncoding; return generatedEnvironmentMap; } /** * Loads a dynamically generated environment map. */ async loadGeneratedEnvironmentMap() { if (this.generatedEnvironmentMap == null) { this.generatedEnvironmentMap = this.GenerateEnvironmentMap(new EnvironmentScene(), 'default'); } return this.generatedEnvironmentMap; } /** * Loads a dynamically generated environment map, designed to be neutral and * color-preserving. Shows less contrast around the different sides of the * object. */ async loadGeneratedEnvironmentMapAlt() { if (this.generatedEnvironmentMapAlt == null) { this.generatedEnvironmentMapAlt = this.GenerateEnvironmentMap(new EnvironmentSceneAlt(), 'neutral'); } return this.generatedEnvironmentMapAlt; } blurCubemap(cubeTarget, sigma) { if (this.blurMaterial == null) { this.blurMaterial = this.getBlurShader(MAX_SAMPLES); const box = new BoxBufferGeometry(); const blurMesh = new Mesh(box, this.blurMaterial); this.blurScene = new Scene(); this.blurScene.add(blurMesh); } const tempTarget = cubeTarget.clone(); this.halfblur(cubeTarget, tempTarget, sigma, 'latitudinal'); this.halfblur(tempTarget, cubeTarget, sigma, 'longitudinal'); // Disposing this target after we're done with it somehow corrupts Safari's // whole graphics driver. It's random, but occurs more frequently on // lower-powered GPUs (macbooks with intel graphics, older iPhones). It goes // beyond just messing up the PMREM, as it also occasionally causes // visible corruption on the canvas and even on the rest of the page. /** tempTarget.dispose(); */ } halfblur(targetIn, targetOut, sigmaRadians, direction) { // Number of standard deviations at which to cut off the discrete // approximation. const STANDARD_DEVIATIONS = 3; const pixels = targetIn.width; const radiansPerPixel = isFinite(sigmaRadians) ? Math.PI / (2 * pixels) : 2 * Math.PI / (2 * MAX_SAMPLES - 1); const sigmaPixels = sigmaRadians / radiansPerPixel; const samples = isFinite(sigmaRadians) ? 1 + Math.floor(STANDARD_DEVIATIONS * sigmaPixels) : MAX_SAMPLES; if (samples > MAX_SAMPLES) { console.warn(`sigmaRadians, ${sigmaRadians}, is too large and will clip, as it requested ${samples} samples when the maximum is set to ${MAX_SAMPLES}`); } const weights = []; let sum = 0; for (let i = 0; i < MAX_SAMPLES; ++i) { const x = i / sigmaPixels; const weight = Math.exp(-x * x / 2); weights.push(weight); if (i == 0) { sum += weight; } else if (i < samples) { sum += 2 * weight; } } for (let i = 0; i < weights.length; i++) { weights[i] = weights[i] / sum; } const blurUniforms = this.blurMaterial.uniforms; blurUniforms['envMap'].value = targetIn.texture; blurUniforms['samples'].value = samples; blurUniforms['weights'].value = weights; blurUniforms['latitudinal'].value = direction === 'latitudinal'; blurUniforms['dTheta'].value = radiansPerPixel; const cubeCamera = new CubeCamera(0.1, 100, targetOut); cubeCamera.update(this.threeRenderer, this.blurScene); } getBlurShader(maxSamples) { const weights = new Float32Array(maxSamples); const poleAxis = new Vector3(0, 1, 0); const shaderMaterial = new ShaderMaterial({ name: 'SphericalGaussianBlur', defines: { 'n': maxSamples }, uniforms: { 'envMap': { value: null }, 'samples': { value: 1 }, 'weights': { value: weights }, 'latitudinal': { value: false }, 'dTheta': { value: 0 }, 'poleAxis': { value: poleAxis } }, vertexShader: /* glsl */ ` varying vec3 vOutputDirection; void main() { vOutputDirection = vec3( position ); gl_Position = projectionMatrix * modelViewMatrix * vec4( position, 1.0 ); } `, fragmentShader: /* glsl */ ` varying vec3 vOutputDirection; uniform samplerCube envMap; uniform int samples; uniform float weights[ n ]; uniform bool latitudinal; uniform float dTheta; uniform vec3 poleAxis; vec3 getSample( float theta, vec3 axis ) { float cosTheta = cos( theta ); // Rodrigues' axis-angle rotation vec3 sampleDirection = vOutputDirection * cosTheta + cross( axis, vOutputDirection ) * sin( theta ) + axis * dot( axis, vOutputDirection ) * ( 1.0 - cosTheta ); return vec3( textureCube( envMap, sampleDirection ) ); } void main() { vec3 axis = latitudinal ? poleAxis : cross( poleAxis, vOutputDirection ); if ( all( equal( axis, vec3( 0.0 ) ) ) ) { axis = vec3( vOutputDirection.z, 0.0, - vOutputDirection.x ); } axis = normalize( axis ); gl_FragColor = vec4( 0.0, 0.0, 0.0, 1.0 ); gl_FragColor.rgb += weights[ 0 ] * getSample( 0.0, axis ); for ( int i = 1; i < n; i++ ) { if ( i >= samples ) { break; } float theta = dTheta * float( i ); gl_FragColor.rgb += weights[ i ] * getSample( -1.0 * theta, axis ); gl_FragColor.rgb += weights[ i ] * getSample( theta, axis ); } } `, blending: NoBlending, depthTest: false, depthWrite: false, side: BackSide }); return shaderMaterial; } async dispose() { for (const [, promise] of this.skyboxCache) { const skybox = await promise; skybox.dispose(); } if (this.generatedEnvironmentMap != null) { (await this.generatedEnvironmentMap).dispose(); this.generatedEnvironmentMap = null; } if (this.generatedEnvironmentMapAlt != null) { (await this.generatedEnvironmentMapAlt).dispose(); this.generatedEnvironmentMapAlt = null; } if (this.blurMaterial != null) { this.blurMaterial.dispose(); } } } //# sourceMappingURL=TextureUtils.js.map