threejs使用的一些分享

最近有朋友需要渲染模型,就帮忙写了个简单的案例,下面分享下代码


image.png

代码块分享

<template>
    <div class="threejs-container">
        <div class="title">
            <p>{{ findRes.title }}</p>
        </div>
        <div class="subTitle"><img
                src=""
                alt="">
            <p class="subTitle-val">{{ findRes.subTitle }}</p> <img
                src=""
                alt="">
        </div>
        <div ref="canvasContainer"></div>
        <div v-if="isLoading" class="loading-overlay">
            <div class="loading-bar"
                :style="{ background: `conic-gradient(#4caf50 ${loadingProgress}%, #eee  ${loadingProgress}%)` }">
                <span class="loading-text">{{ loadingProgress.toFixed(0) }}%</span>
            </div>
        </div>
    </div>
</template>

<script>
import * as THREE from 'three';
import { FBXLoader } from 'three/examples/jsm/loaders/FBXLoader.js';
import { OrbitControls } from 'three/examples/jsm/controls/OrbitControls.js';
import { GLTFLoader } from 'three/examples/jsm/loaders/GLTFLoader.js';
export default {
    name: 'MyView',
    data() {
        return {
            findRes: {},
            isLoading: true,
            loadingProgress: 0
        }
    },
    methods: {
        initSence() {
            const scene = new THREE.Scene();
            const camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);
            const renderer = new THREE.WebGLRenderer({ alpha: true, antialias: true });
            renderer.setSize(window.innerWidth, window.innerHeight);
            this.$refs.canvasContainer.appendChild(renderer.domElement);

            // 添加光源
            const lightPower = this.findRes.lightPower || 1
            const ambientLight = new THREE.AmbientLight(0xffffff, lightPower);
            scene.add(ambientLight);
            const directionalLight = new THREE.DirectionalLight(0xffffff, lightPower);
            directionalLight.position.set(1, 1, 1);
            scene.add(directionalLight);

            const pointLight = new THREE.PointLight(0xffffff, 2, 100);
            pointLight.position.set(5, 5, 5);
            scene.add(pointLight);

            const loadingManager = new THREE.LoadingManager();

            loadingManager.onStart = () => {
                this.isLoading = true;
            };

            loadingManager.onLoad = () => {
                //this.loadingProgress = 100;
                setTimeout(() => {
                   this.isLoading = false;
                }, 500)
            };

            loadingManager.onProgress = (url, itemsLoaded, itemsTotal) => {
                this.loadingProgress = Math.floor((itemsLoaded / itemsTotal) * 10) + 90; // 更新进度
            };


            const loader = this.findRes.modelPath.endsWith('.gltf') || this.findRes.modelPath.endsWith('.glb') ?
                new GLTFLoader(loadingManager) : new FBXLoader(loadingManager); // 根据文件类型选择加载器

            let object; // 声明一个变量来存储加载的模型
            let isMouseControlled = false; // 添加标志来跟踪鼠标控制状态

            loader.load(this.findRes.modelPath, (loadedObject) => {
                object = loadedObject.scene || loadedObject; // 将加载的模型赋值给变量
                scene.add(object);
            }, (xhr) => {
                // 使用 xhr 对象获取更精确的加载进度
                const percentComplete = Math.round((xhr.loaded / xhr.total) * 100);
                this.loadingProgress = percentComplete>90 ? 90 : percentComplete;
                //console.log('xhr',percentComplete);
            }, (error) => {
                console.error('加载模型时出错:', error);
            });
            const cameraRng = this.findRes.cameraRng ? this.findRes.cameraRng : [0, 20, 22]
            //console.log('cameraRng', cameraRng);
            camera.position.set(...cameraRng);
            const controls = new OrbitControls(camera, renderer.domElement);



            // 监听鼠标控制事件
            controls.addEventListener('start', () => {
                isMouseControlled = true; // 开始控制时设置标志
            });

            const animate = function () {
                requestAnimationFrame(animate);
                if (object && !isMouseControlled) {
                    object.rotation.y += 0.005; // 使模型自动旋转
                }
                renderer.render(scene, camera);
            };
            animate();
        }
    },
    mounted() {
        const routeParams = this.$route.query;
        //const findRes = window.config.lists.find(item => item.id == routeParams.id);
        //console.log('obj:', findRes);
        const findRes  = {
            id: 1,
            title: '解放战争时期粟裕大将使用的臂搁',
            subTitle: '新四军江南指挥部纪念馆',
            modelPath: "/model/1/model.fbx",
            cameraRng: [0, 3, 30], //相机配置
            lightPower:2 //光源
        }
        this.findRes = findRes || window.config.lists[0];
        this.initSence();
    }
};
</script>

<style scoped>
.loading-overlay {
    position: fixed;
    top: 0;
    left: 0;
    width: 100%;
    height: 100%;
    background: rgba(0, 0, 0, 0.5);
    display: flex;
    align-items: center;
    justify-content: center;
    z-index: 9999;
    /* 确保在最上层 */
}

.loading-bar {
    width: 100px;
    /* 圆形的直径 */
    height: 100px;
    /* 圆形的直径 */
    border-radius: 50%;
    /* 使其成为圆形 */
    position: relative;
    /* 使文本能够绝对定位 */
    transition: background 0.5s ease;
    /* 添加过渡效果 */
    background: conic-gradient(gray 0%, transparent 0%);
    /* 默认颜色为灰色 */
}



.loading-text {
    position: absolute;
    top: 50%;
    left: 50%;
    transform: translate(-50%, -50%);
    /* 居中对齐 */
    color: #000;
    /* 文本颜色为黑色 */
    font-size: 16px;
    /* 文本大小 */
    font-weight: bold;
    /* 文本加粗 */
    background: #fff;
    /* 白色背景 */
    border-radius: 50%;
    /* 使其成为圆形 */
    width: 80px;
    /* 圆形的直径 */
    height: 80px;
    /* 圆形的直径 */
    display: flex;
    align-items: center;
    justify-content: center;
    /* 居中对齐文本 */
}


.threejs-container {
    width: 100vw;
    height: 100vh;
    position: relative;
    overflow: hidden;
    background: url('/pc_bg.png') 50% / cover no-repeat;
    z-index: 1;
}

.threejs-container .title {
    position: absolute;
    top: 20px;
    left: 50%;
    z-index: 10;
    -webkit-transform: translateX(-50%);
    transform: translateX(-50%);
    font-size: 26px;
    font-weight: 700;
    color: #ffeac0;
    text-align: center;
    display: -webkit-box;
    display: -ms-flexbox;
    display: flex
}

.threejs-container .subTitle {
    position: absolute;
    top: 80px;
    left: 50%;
    z-index: 10;
    -webkit-transform: translateX(-50%);
    transform: translateX(-50%);
    font-size: 22px;
    color: #ffeac0;
    opacity: .5;
    text-align: center;
    display: -webkit-box;
    display: -ms-flexbox;
    display: flex;

}

.subTitle-val {
    margin: 0px 10px;
}

canvas {
    display: block;
    position: absolute;
    top: 0;
    left: 0;
    z-index: 2;
}
</style>
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容