3.js - 漫天孔明灯(使用OrbitControls 锁定相机镜头)

main.js

javascript 复制代码
import * as THREE from 'three'
import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls'
import { RGBELoader } from 'three/examples/jsm/loaders/RGBELoader'
import { GLTFLoader } from 'three/examples/jsm/loaders/GLTFLoader'
import gsap from 'gsap'


// @ts-ignore
import vertexShader from './shader/11-03/vertex.glsl?raw'
// @ts-ignore
import fragmentShader from './shader/11-03/fragment.glsl?raw'


// ------------------------------------------------------ 	------------------------------------------------
	const scene = new THREE.Scene()
	
	const camera = new THREE.PerspectiveCamera(90, window.innerHeight / window.innerHeight, 0.1, 1000)
	camera.position.set(0, 0, 2)
	camera.aspect = window.innerWidth / window.innerHeight
	scene.add(camera)
	
	const axesHelper = new THREE.AxesHelper(5)
	scene.add(axesHelper)
// ------------------------------------------------------------------------------------------------------



`【loadAsync:异步加载hdr,返回一个Promise】`
const rgbeLoader = new RGBELoader()
rgbeLoader.loadAsync('../public/assets/2k.hdr').then(texture => {
	texture.mapping = THREE.EquirectangularReflectionMapping
	scene.background = texture
	scene.environment = texture
})

`【着色器材质】`
const shaderMaterial = new THREE.ShaderMaterial({
	vertexShader: vertexShader,
	fragmentShader: fragmentShader,
	side: THREE.DoubleSide,
	// transparent: true,
})

`【加载灯笼的glb】`
const gltfLoader = new GLTFLoader()
gltfLoader.load('../public/assets/model/flyLight.glb', gltf => {
	console.log('gltf=', gltf)
	
	for (let i = 0; i < 150; i++) {
		let flyLight = gltf.scene.clone(true)
		// console.log('flyLight=', flyLight)
		flyLight.children[1].material = shaderMaterial
	
		let x = (Math.random() - 0.5) * 300
		let z = (Math.random() - 0.5) * 300
		let y = Math.random() * 60 + 25
		flyLight.position.set(x, y, z)
	
		gsap.to(flyLight.rotation, {
			y: 2 * Math.PI, // 绕着y轴旋转360°
			duration: 10 + Math.random() * 30, // 持续时间
			repeat: -1 // gsap中,-1代表,无限次循环
		})
		
		gsap.to(flyLight.position, {
			x: '+=' + Math.random() * 5, // x坐标,增加一个在0到5之间的随机值
			y: '+=' + Math.random() * 20, // y坐标,增加一个在0到20之间的随机值
			duration: 5 + Math.random() * 10,
			yoyo: true, // 动画在每次重复时都将反向进行,即:从 a -> b 到 b -> a
			repeat: -1
		})
	
		scene.add(flyLight)
	}
})


// ------------------------------------------------------ 	------------------------------------------------

`【渲染器】`
	const renderer = new THREE.WebGLRenderer({ alpha: true })
	renderer.shadowMap.enabled = true	
	【最新版本属性名字有改变】:`renderer.outputEncoding`已经变更为`renderer.outputColorSpace`
	// renderer.outputEncoding = THREE.sRGBEncoding // 设置渲染器的输出编码方式(此API已弃用)
	renderer.outputColorSpace = THREE.SRGBColorSpace // 设置渲染器的输出编码方式(此API已启用)
	renderer.toneMapping = THREE.ACESFilmicToneMapping // 色调映射
	renderer.toneMappingExposure = 0.2 // 色调映射-曝光度
	renderer.setSize(window.innerWidth, window.innerHeight)
	document.body.appendChild(renderer.domElement)

`【轨道控制器】`
	const controls = new OrbitControls(camera, renderer.domElement)
	controls.enableDamping = true
	controls.autoRotate = true // 相机绕着y轴,顺时针自动旋转
	controls.autoRotateSpeed = 0.1 // 自动旋转的速度
	
	`最大极角 == 最小极角:代表,相机会被锁定在这个角度上,无法上下移动(锁定相机镜头)`
	controls.maxPolarAngle = (Math.PI / 3) * 2
	controls.minPolarAngle = (Math.PI / 3) * 2
	// controls.maxPolarAngle = (Math.PI / 2) * 0.95 // 稍微小于90度
	// controls.minPolarAngle = Math.PI / 4 // 45度


	function animate() {
		controls.update()
		requestAnimationFrame(animate)
		renderer.render(scene, camera)
	}
	animate()
	
	window.addEventListener('resize', () => {
		camera.aspect = window.innerWidth / window.innerHeight
		camera.updateProjectionMatrix()
		renderer.setSize(window.innerWidth, window.innerHeight)
		renderer.setPixelRatio(window.devicePixelRatio)
	})

vertex.glsl

javascript 复制代码
precision lowp float;

varying vec4 vPosition;
varying vec4 gPosition;

void main() {
    vec4 modelPosition = modelMatrix*vec4(position, 1.0);

    vPosition = modelPosition;
    
    gPosition = vec4(position, 1.0);

    gl_Position = projectionMatrix*viewMatrix*modelPosition;
}

fragment.glsl

javascript 复制代码
precision lowp float;

varying vec4 vPosition;
varying vec4 gPosition;

void main() {
    vec4 redColor = vec4(1, 0, 0, 1);
    vec4 yellowColor = vec4(1, 1, 0.5, 1);
    vec4 mixColor = mix(yellowColor, redColor, gPosition.y/3.0);

    /*
        gl_FrontFacing:
            OpenGL和WebGL中,片段着色器中的内置变量,
            用于指示当前处理的片段是否属于一个正面朝向的三角形,
        
        OpenGL和WebGL的渲染管线中,三角形可以被定义为,正面或背面,
            这取决于,它们的顶点顺序与当前设置的前脸/后脸判断标准(通常是基于顶点的绕序,如顺时针或逆时针)是否一致。
    */
    if(gl_FrontFacing) {
        // 如果,片段属于正面朝向的三角形
        gl_FragColor = vec4(mixColor.xyz-(vPosition.y-20.0)/80.0-0.1, 1);
    } else {
        gl_FragColor = vec4(mixColor.xyz, 1);
    }

    // 这个是我测试用的
    // gl_FragColor = vec4(1, 0, 0, 1);
}
相关推荐
一嘴一个橘子6 天前
3.js - 着色器设置点材质(螺旋星系特效)
three.js
Ian10259 天前
Three.js new THREE.TextureLoader()纹理贴图使用png图片显示为黑色
前端·javascript·webgl·three.js·贴图·三维
Jedi Hongbin15 天前
THREE.JS像素风格渲染
javascript·three.js·shader·后处理
unix2linux21 天前
Parade Series - 3D Modeling
python·flask·html·jquery·webgl·three.js
zj靖1 个月前
three.js 几何体、材质和网格模型
three.js
一嘴一个橘子1 个月前
3.js - 顶点着色器、片元着色器的联系
three.js
Jedi Hongbin1 个月前
Threejs&WebGPU运动残影demo
javascript·three.js
Virtual091 个月前
借助Three.js,我们也能实现一个地球仪旋转
前端·javascript·three.js
一嘴一个橘子1 个月前
3.js - 物理引擎终极
three.js