开发者问题收集

在页面之间导航时,三个 JS 对象无法在 Nuxt 3 中加载

2022-09-26
1179

首先我要说的是,我还有很多东西需要学习。 我讨厌寻求帮助,但我翻遍了好几页寻找答案却毫无收获。

我确信这个问题很简单,但有人能解释一下我做错了什么吗?我的 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>
2个回答

发生您的问题是因为您确实有一个 querySelector,而推荐的方法是使用模板引用: https://vuejs.org/guide/essentials/template-refs.html#template-refs

原因是当您处于 SPA 上下文中时, document 保持不变,但堆叠 querySelector 并尝试使用它们将无法在组件生命周期中正常运行。

Nuxt 的情况更糟,因为您实际上没有服务器上的 document ,因此可能会导致更多错误甚至内存泄漏。

mounted 仅在客户端调用,所以这就是它在那里运行良好的原因。但推荐的方法仍然是使用模板引用,以避免任何问题(并且无需删除监听器)。

kissu
2022-09-26

终于搞清楚了。

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>

不确定这是否是最好的解决方案......但它似乎有效!

Kyrony
2022-09-26