mirror of
https://github.com/nuxt/nuxt.git
synced 2025-01-19 01:45:53 +00:00
122 lines
2.8 KiB
Vue
122 lines
2.8 KiB
Vue
|
<template>
|
||
|
<canvas class="webgl" />
|
||
|
</template>
|
||
|
.webgl {
|
||
|
outline: none;
|
||
|
}
|
||
|
</style>
|
||
|
|
||
|
<script>
|
||
|
import * as THREE from 'three'
|
||
|
import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls.js'
|
||
|
import { GLTFLoader } from 'three/examples/jsm/loaders/GLTFLoader.js'
|
||
|
|
||
|
export default {
|
||
|
mounted () {
|
||
|
// Canvas
|
||
|
const canvas = document.querySelector('canvas.webgl')
|
||
|
|
||
|
// Scene
|
||
|
const scene = new THREE.Scene()
|
||
|
|
||
|
// Models
|
||
|
let gem
|
||
|
let light
|
||
|
|
||
|
const gltfLoader = new GLTFLoader()
|
||
|
gltfLoader.load(
|
||
|
'/3D/gem.gltf',
|
||
|
(gltf) => {
|
||
|
console.log(gltf)
|
||
|
// Gem
|
||
|
gem = gltf.scene.children[6]
|
||
|
console.log(gltf.scene.children[6])
|
||
|
|
||
|
// Material setup
|
||
|
const textureLoader = new THREE.TextureLoader()
|
||
|
const roughnessTexture = textureLoader.load('/3D/roughness.jpeg')
|
||
|
gem.material.roughnessMap = roughnessTexture
|
||
|
gem.material.displacementScale = 0.15
|
||
|
gem.material.emissiveIntensity = 1
|
||
|
gem.material.refractionRatio = 1
|
||
|
gem.rotation.z = 0.5
|
||
|
scene.add(gem)
|
||
|
|
||
|
light = gltf.scene.children[0]
|
||
|
scene.add(light)
|
||
|
}
|
||
|
)
|
||
|
|
||
|
// Lights
|
||
|
const ambientLight = new THREE.AmbientLight(0xFFFFFF, 2)
|
||
|
|
||
|
scene.add(ambientLight)
|
||
|
|
||
|
const directionalLight = new THREE.DirectionalLight(0xFFFFFF, 3)
|
||
|
directionalLight.position.set(1, 1, 1)
|
||
|
scene.add(directionalLight)
|
||
|
|
||
|
// Settings
|
||
|
const sizes = {
|
||
|
width: 200,
|
||
|
height: 200
|
||
|
}
|
||
|
|
||
|
// Base camera
|
||
|
const camera = new THREE.PerspectiveCamera(
|
||
|
75,
|
||
|
sizes.width / sizes.height,
|
||
|
0.1,
|
||
|
100
|
||
|
)
|
||
|
camera.position.set(2, 2, 6)
|
||
|
scene.add(camera)
|
||
|
|
||
|
// Controls
|
||
|
const controls = new OrbitControls(camera, canvas)
|
||
|
controls.enableZoom = false
|
||
|
controls.target.set(0, 0.75, 0)
|
||
|
controls.enableDamping = true
|
||
|
// Lock Y Axis
|
||
|
controls.minPolarAngle = Math.PI / 2
|
||
|
controls.maxPolarAngle = Math.PI / 2
|
||
|
|
||
|
// Render
|
||
|
const renderer = new THREE.WebGLRenderer({
|
||
|
antialiasing: true,
|
||
|
canvas,
|
||
|
alpha: true
|
||
|
})
|
||
|
renderer.setClearColor(0x000000, 0)
|
||
|
renderer.shadowMap.enabled = true
|
||
|
renderer.shadowMap.type = THREE.PCFSoftShadowMap
|
||
|
renderer.setSize(sizes.width, sizes.height)
|
||
|
renderer.setPixelRatio(Math.min(window.devicePixelRatio, 2))
|
||
|
|
||
|
// Animations
|
||
|
const clock = new THREE.Clock()
|
||
|
let previousTime = 0
|
||
|
|
||
|
const tick = () => {
|
||
|
const elapsedTime = clock.getElapsedTime()
|
||
|
const deltaTime = elapsedTime - previousTime
|
||
|
previousTime = elapsedTime
|
||
|
if (gem) {
|
||
|
gem.rotation.y = 0.5 * elapsedTime
|
||
|
}
|
||
|
|
||
|
// Update controls
|
||
|
controls.update()
|
||
|
|
||
|
// Render
|
||
|
renderer.render(scene, camera)
|
||
|
|
||
|
// Call tick again on the next frame
|
||
|
window.requestAnimationFrame(tick)
|
||
|
}
|
||
|
|
||
|
tick()
|
||
|
}
|
||
|
}
|
||
|
</script>
|