在页面之间导航时,三个 JS 对象无法在 Nuxt 3 中加载
首先我要说的是,我还有很多东西需要学习。 我讨厌寻求帮助,但我翻遍了好几页寻找答案却毫无收获。
我确信这个问题很简单,但有人能解释一下我做错了什么吗?我的 Three JS atom 加载得很好。我没有遇到任何问题,直到我导航到另一个页面(NuxtLink)然后返回此页面。
它抛出了这个错误:
Uncaught (in promise) TypeError: Cannot read properties of null (reading 'width') at new WebGLRenderer (three.module.js:26639:23) at Atom.vue:64:1 at hook.__wdc.hook.__wdc (runtime-core.esm-bundler.js:2626:20) at callWithErrorHandling (runtime-core.esm-bundler.js:155:22) at callWithAsyncErrorHandling (runtime-core.esm-bundler.js:164:21) at hook.__weh.hook.__weh (runtime-core.esm-bundler.js:2684:29) at flushPostFlushCbs (runtime-core.esm-bundler.js:341:32) at flushJobs (runtime-core.esm-bundler.js:395:9)
据我理解,这意味着画布没有加载?我甚至尝试使用 options API 而不是 Composition API,但遇到了同样的问题。我尝试使用 onActivated 和 KeepAlive,如您所见,但它们都不起作用。
<script setup>
import * as THREE from 'three'
onActivated(() => {
// Canvas
const canvas = document.querySelector('canvas.webgl')
// Sizes
const sizes = {
width: 300,
height: 300
}
const pixelRatio = window.devicePixelRatio
/* Base */
// Scene
const scene = new THREE.Scene()
/* Props */
// Plane Background
const backgroundPlane = new THREE.PlaneGeometry(50, 50, 4, 4)
const backgroundMaterial = new THREE.MeshStandardMaterial({ emissive: "#000505", roughness:0.1, metalness: 1 })
const background = new THREE.Mesh(backgroundPlane, backgroundMaterial)
background.position.z = -5
scene.add(background)
// Atom Group
const atom = new THREE.Group()
scene.add(atom)
atom.position.z = -2
const protonSphere = new THREE.SphereGeometry(.3, 16, 16)
const protonMaterial = new THREE.MeshStandardMaterial({ emissive: "#080E54", roughness:0.4, metalness: 0.9 })
const proton = new THREE.Mesh(protonSphere, protonMaterial)
atom.add(proton)
const electronCloudSphere = new THREE.SphereGeometry(2, 32, 32)
const electronCloudMaterial = new THREE.MeshPhysicalMaterial({ opacity: .2, transparent: true, reflectivity:0, side:THREE.DoubleSide, roughness:0.5, metalness: 0, ior: 2, thickness: 2, transmission: 1 })
const electronCloud = new THREE.Mesh(electronCloudSphere, electronCloudMaterial)
atom.add(electronCloud)
const electronSphere = new THREE.SphereGeometry(.1, 16, 16)
const electronMaterial = new THREE.MeshStandardMaterial({ emissive: "#2FFFFF", roughness:0, metalness: 0.9 })
const electron = new THREE.Mesh(electronSphere, electronMaterial)
const radius = 1.5
let theta = Math.random() * Math.PI * 2
let phi = Math.random() * Math.PI
const x = radius * Math.sin(theta)
const z = radius * Math.cos(theta)
const y = radius * Math.cos(phi)
electron.position.set(x, y, z)
atom.add(electron)
const pointLight = new THREE.PointLight("#B7DBFF", 1, 5)
pointLight.position.z = 1
scene.add(pointLight)
/* Camera & Controls */
// Base camera
const camera = new THREE.PerspectiveCamera(75, sizes.width / sizes.height, 0.1, 100)
camera.position.z = 3
scene.add(camera)
/* Renderer */
const renderer = new THREE.WebGLRenderer({
canvas,
antialias: true,
})
renderer.setSize(sizes.width, sizes.height)
renderer.setPixelRatio(Math.min(pixelRatio, 2))
/* Animate */
const clock = new THREE.Clock()
let time = 0
const tick = () =>
{
const elapsedTime = clock.getElapsedTime()
const deltaTime = elapsedTime - time
time = elapsedTime
// Animations
let shake = (Math.random() * 0.015)
proton.rotateX(.5)
proton.position.set(shake, shake, 0)
atom.rotateX(.01 * Math.random())
atom.rotateY(.02 * Math.random())
atom.rotateZ(.04 * Math.random())
// Render
renderer.render(scene, camera)
// Call tick again on the next frame
window.requestAnimationFrame(tick)
}
tick()
})
</script>
<template>
<div class="canvas-wrapper">
<canvas id="atom" class="webgl"></canvas>
</div>
</template>
发生您的问题是因为您确实有一个 querySelector,而推荐的方法是使用模板引用: https://vuejs.org/guide/essentials/template-refs.html#template-refs
原因是当您处于 SPA 上下文中时,
document
保持不变,但堆叠 querySelector 并尝试使用它们将无法在组件生命周期中正常运行。
Nuxt 的情况更糟,因为您实际上没有服务器上的
document
,因此可能会导致更多错误甚至内存泄漏。
mounted
仅在客户端调用,所以这就是它在那里运行良好的原因。但推荐的方法仍然是使用模板引用,以避免任何问题(并且无需删除监听器)。
终于搞清楚了。
const webgl = ref(null)
onMounted(() => {
// Canvas
const canvas = webgl.value
使用模板引用似乎可以解决我的问题。
<template>
<div class="canvas-wrapper">
<canvas ref="webgl" id="atom" class="webgl"></canvas>
</div>
</template>
不确定这是否是最好的解决方案......但它似乎有效!