vue项目中实现3D万花筒和3D文字旋转效果

news2024/11/26 18:46:38

一、万花筒

 

        1、html部分 

//万花筒html
<div class="carousel" data-gap="368">
    <figure>
        <div class="carouselItem" v-for="(item,index) in exhibitionList" :key="index">
            <div class="itemContent" :class="`cardBg${index+1}`" @click="toExhibitionHall(item)" @mouseenter="showEnterprise(item)">
                <h6>{{item.name}}</h6>
                <p :title="item.des">{{item.des}}</p>
            </div>
        </div>
    </figure>
</div>

        2、js部分

mounted() {
    this.cardRotate();
},
methods: {
    // 卡片旋转
    cardRotate(){
        let that = this;
        carousel();
        var xdeg = 0;
        this.timer = setInterval(function() {
            xdeg = xdeg + 0.3;
            $('figure').css('transform', "rotateY(" + (-xdeg) + "deg)");
        }, 20);

        function carousel(){
            var figure = $("figure"),
            items = $(".carouselItem"),
            n = items.length,
            theta = 2 * Math.PI / n,
            currImage = 0;
            setUpCarousel(n, items.width());

            function setUpCarousel(n, s) {
                let padg = window.screen.width / 1920 * $(".carousel").attr("data-gap");
                var apothem = s / (2 * Math.tan(Math.PI / n));
                figure.css('transformOrigin', '50% 50% ' + (-apothem) + 'px');
                items.css("padding", "20px " + padg + "px 0");
                for (var i = 1; i < n; i++) {
                    items.eq(i).css({
                        'transformOrigin': '50% 50% ' + (-apothem) + 'px',
                        'transform': 'rotateY(' + i * theta + 'rad)',
                    });
                }
                rotateCarousel(currImage);
            }

            function rotateCarousel(index) {
                figure.css({
                    "transform": "rotateX(0deg) rotateY(" + index * -theta + "rad)"
                });
            }

            // 暂停启动定时器
                $(".carousel").mouseleave(function() {
                that.timer = setInterval(function() {
                    xdeg = xdeg + 0.3;
                    $('figure').css('transform', "rotateY(" + (-xdeg) + "deg)");
                }, 20);
            })
            $(".carousel").mouseenter(function() {
                clearInterval(that.timer);
            })
            // $(".prev").click(() => {
            //     currImage--;
            //     rotateCarousel(currImage);
            // })
            // $(".next").click(() => {
            //     currImage++;
            //     rotateCarousel(currImage);
            // })
        }
    },
},

        3、css部分

.carousel {
    width: 26.75rem;
    height: 13rem;
    // perspective: 700px;
    overflow: hidden;
    display: flex;
    flex-direction: column;
    align-items: center;
    position: absolute;
    bottom: 3.5rem;
    left:50%;
    transform: translateX(-50%);
    z-index: 4;
}
.carousel figure {
    margin: 0;
    // width: 1070px;
    transform-style: preserve-3d;
    transition: transform 0.5s;
}

.carouselItem {
    width: 100%;
    height: 11.55rem;
    box-sizing: border-box;
    cursor: pointer;
    z-index: 1;
    // -webkit-box-reflect: below 20px -webkit-linear-gradient(top, rgba(0, 0, 0, 0) 40%, rgba(0, 0, 0, .5) 100%);
}

.itemContent {
    width: 100%;
    height: 100%;
    padding: 5.8rem 1.7rem 0;
    box-sizing: border-box;
    color:#fff;
    h6{
        font-weight: bold;
        font-size: .65rem;
        line-height: .65rem;
        margin-bottom:.4rem;
        text-align: center;
    }
    p{
        font-size: .45rem;
        font-weight: 400;
        text-align: center;
        overflow: hidden;
        text-overflow: ellipsis;
        display: -webkit-box;
        -webkit-line-clamp: 2;
        -webkit-box-orient: vertical;
    }
}
.cardBg1{
    background: url(~img/cardBg1.png) no-repeat center / 100% 100%;
}
.cardBg2{
    background: url(~img/cardBg2.png) no-repeat center / 100% 100%;
}
.cardBg3{
    background: url(~img/cardBg3.png) no-repeat center / 100% 100%;
}
.cardBg4{
    background: url(~img/cardBg4.png) no-repeat center / 100% 100%;
}
.cardBg5{
    background: url(~img/cardBg5.png) no-repeat center / 100% 100%;
}
.cardBg6{
    background: url(~img/cardBg6.png) no-repeat center / 100% 100%;
}

.carouselItem:not(:first-of-type) {
    position: absolute;
    left: 0;
    top: 0;
}

二、文字旋转

         1、html部分  

<div class="textCricle1">
    <canvas id="canvas1" width="1310" height="1310" style="width: 100%;height: 100%;"></canvas>
</div>
<div class="textCricle2">
    <canvas id="canvas2" width="1550" height="1550" style="width: 100%;height: 100%;"></canvas>
</div>
<div class="textCricle3">
    <canvas id="canvas3" width="1790" height="1790" style="width: 100%;height: 100%;"></canvas>
</div>

         2、js部分

mounted() {
    this.textMove();
},
methods: {
    // 环形文字
    textMove(){
        let canvas1 = document.getElementById('canvas1'),
            fill1 = 'rgba(0, 168, 255, 0.8)',
            stroke1 = 'rgba(0, 168, 255, 0.8)';
            
        this.canvasDraw(canvas1, fill1, stroke1)

        let canvas2 = document.getElementById('canvas2'),
            fill2 = 'rgba(0, 168, 255, 0.2)',
            stroke2 = 'rgba(0, 168, 255, 0.2)';

        this.canvasDraw(canvas2,fill2,stroke2)

        let canvas3 = document.getElementById('canvas3'),
            fill3 = 'rgba(0, 168, 255, 0.1)',
            stroke3 = 'rgba(0, 168, 255, 0.1)';

        this.canvasDraw(canvas3,fill3,stroke3)
    },
    // canvas绘制文字
    canvasDraw(dom,fill,stroke){
        let context = dom.getContext('2d'),

            TEXT_FILL_STYLE = fill,
            TEXT_STROKE_STYLE = stroke,
            TEXT_SIZE = 30,

            circle = {
                x: dom.width / 2,
                y: dom.height / 2,
                radius: (dom.width / 2) - 20,
            },

            startAngle = Math.PI*2+Math.PI/2,
            endAngle = Math.PI/60+Math.PI/2,
            text = '0101010101010101010101010101010101010101010101010101010101010101010101010101010101010101010101010101010101010101010101010101010101010101010101010101';

        /**
         * 绘制环形文字
         * @param{string} string 传入的文字
         * @param startAngle 起始角
         * @param endAngle 末尾角
         */
        let drawCircularText = (string, startAngle, endAngle) => {
            let radius = circle.radius,
                angleDecrement = (startAngle - endAngle) / (string.length - 1),
                angle = parseFloat(startAngle),
                index = 0,
                character;

            context.save();
            context.fillStyle=TEXT_FILL_STYLE;
            context.strokeStyle=TEXT_STROKE_STYLE;
            context.font=TEXT_SIZE+'px Lucida Sans';

            while(index < string.length){
                //获取传入的字符串的每个字符
                character = string.charAt(index);

                context.save();
                context.beginPath();
                //位移到每个字符的指定位置
                context.translate(circle.x+Math.cos(angle)*radius,circle.y - Math.sin(angle)*radius);

                //旋转坐标系到每个字符应该达到到角度
                context.rotate(Math.PI/2-angle);
                context.fillText(character,0,0);
                context.strokeText(character,0,0);

                //角度递减
                angle -= angleDecrement;

                index++;

                context.restore();
            }
        };


        //Init
        let init=(startAngle,endAngle)=>{

            context.textAlign='center';
            context.textBaseline='middle';
            drawCircularText(text,startAngle,endAngle);
        };
        init(startAngle,endAngle);
    }
},

        3、css部分

.textCricle1{
    position: absolute;
    bottom:2rem;
    left:50%;
    width:32.75rem;
    height:32.75rem;
    transform: translate(-50%, 38%) rotateX(77deg);
    z-index: 1;
    #canvas1{
        animation: rotate 80s infinite linear;
    }
}

.textCricle2{
    position: absolute;
    bottom:2rem;
    left:50%;
    width:38.75rem;
    height:38.75rem;
    transform: translate(-50%, 40%) rotateX(78deg);
    z-index: 1;
    #canvas2{
        animation: rotate 80s infinite linear;
    }
}

.textCricle3{
    position: absolute;
    bottom:2rem;
    left:50%;
    width:44.75rem;
    height:44.75rem;
    transform: translate(-50%, 41%) rotateX(79deg);
    z-index: 1;
    #canvas3{
        animation: rotate 80s infinite linear;
    }
}
@keyframes rotate {
    0% { transform: rotate(0deg); }
    50% { transform: rotate(-180deg); }
    100% { transform: rotate(-360deg); }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/764555.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

动态内存函数详解-【malloc,calloc,realloc,free】

动态内存函数详解 malloc一、malloc的简介1.malloc函数的定义&#xff1a;2.参数&#xff1a;3.返回值&#xff1a;4.功能&#xff1a; 二、malloc函数的使用 free函数free函数的简介free函数使用注意事项 calloccalloc函数的简介二、calloc函数的使用calloc函数的注意事项 rea…

Ubuntu环境搭建

本文以Ubuntu 18.04为例 安装repo mkdir ~/bin export PATH~/bin:$PATH如果可以访问 google 的地址&#xff0c;下载 Repo 工具&#xff0c;并确保它可执行&#xff1a; curl https://storage.googleapis.com/git-repo-downloads/repo > ~/bin/repo chmod ax ~/bin/rep…

memset的坑

前言 memset 作为对内存初始化的函数&#xff0c;还是有不少坑和误区的&#xff0c;今天就来对这个函数作一个总结。 一、函数作用 memset 函数在 C 中被广泛应用于内存的初始化和设置。它可以将一段连续的内存空间快速设置为指定的值。这个函数主要作用于数组、结构体等数据类…

python 安装、配置、使用 xlrd模块

xlrd模块的分为python安装 和pycharm配置两个步骤 1. 安装xlrd模块 打开cmd&#xff0c;输入 pip install xlrd 按 enter键 安装完成即可&#xff0c;如果想升级&#xff0c;根据提示完成即可 2. pycharm 配置xlrd pycharm模块导入xlrd模块时&#xff0c;import xlrd #导入模…

【前端知识】JavaScript——var 与 let 的区别

【前端知识】JavaScript——var 与 let 的区别 var声明的变量会自动提升到函数作用域顶部&#xff0c;而let不会。 在解析代码时&#xff0c;JavaScript 引擎会注意出现在块后面的 let 声明&#xff0c;只不过在此之前不能以任何方式来引用未声明的变量。在 let 声明之前的执行…

基于simulink的DPLL仿真笔记

该笔记主要用于本人思路整理与记录 本设计运用的是电荷泵一阶环路滤波器&#xff0c;二阶三阶则在此基础上举一反三&#xff0c;以后如有机会会慢慢补全 文章目录 一.仿真模型PS&#xff08;题外话&#xff09; 二.仿真结果三.环路滤波器分析1. 环路滤波器对比LPF2. 环路滤波器…

(阿里云)STM32L+BC20+MQTT协议传输温湿度,ADC,电压,GPS数据到阿里云物联网平台

1、材料准备 准备以下材料 2、设备连接 2.1 插入物联网卡 首先把BC20核心板从开发板上拆下来 然后将物联卡放置在BC20核心板内 物联卡放置完成将BC20核心板重新插入到开发板内&#xff08;注意不要弄错方向&#xff09; 同时接入天线 2.2 连接ST-Link仿真器 用3条杜邦线接…

mzjh 项目鉴权

获取后端的接口后 将后端的数据转成数组 并报错保存 业务逻辑 function getPoints(menus) {let Points []for (let menu of menus) {if (menu.useType 2) {if (menu.grantName ! undefined && menu.grantName ! null) {Points.push(menu.grantName)}}if (menu.child…

TCP的窗口控制和重发控制【TCP原理(笔记三)】

文章目录 利用窗口控制提高速度窗口控制与重发控制确认应答未能返回的情况某个报文段丢失的情况 控制流 利用窗口控制提高速度 TCP以1个段为单位&#xff0c;每发一个段进行一次确认应答的处理&#xff0c;如图。这样的传输方式有一个缺点。那就是&#xff0c;包的往返时间越长…

YOLO-V5分类实战系列 —— 调优自己的数据集+RK1808部署

YOLO-V5分类实战系列 —— 调优自己的数据集 1、保存训练和测试图片2、数据归一化3、数据增强3.1、数据增强库&#xff1a;albumentations3.2、数据增强库&#xff1a;torchvision 4、ONNX CPU 推理4.1、Pt 模型转为 ONNX4.2、ONNX 推理验证4.3、 ONNX CPU推理&#xff08;C&am…

理解Deformable Convolution网络

1.简介 偶然了解到了可形变卷积这篇文章&#xff0c;看了几篇博文后大致了解的差不多了&#xff0c;但是有些细节还是看了代码之后才理解。这里想自己写一下关于这篇论文的了解&#xff0c;希望自己能够讲清楚。这里放一篇写的很好地博文链接&#xff0c;想更深入了解代码的可…

Gateway自定义过滤器——全局过滤器

一、什么是全局过滤器&#x1f349; 首先&#xff0c;我们要知道全局过滤器其实是特殊路由过滤器(特殊的GatewayFilter)&#xff0c;会有条件地作用于所有路由。 为什么要自定义全局过滤器&#xff1f;就好比是看大门的保安大叔&#xff0c;平时主要是做好进出大门外来人员登记…

Chatglm实现agent控制

背景&#xff1a; 这个系列文章&#xff0c;会从LLM搭建应用生态角度来写。从0到1训练一个大的通用的模型对于大部分人和团队来讲是不现实的。重资金&#xff0c;重技术含量、重投入这几个门槛可以把很多团队直接劝退。那么在LLM蓬勃发展的时候我们可以做些什么呢&#xff0c;…

C语言程序设计——字符、字符串、内存函数

一、长度不受限的字符串函数 1. strlen size_t strlen(const char* str); 功能&#xff1a;求字符串长度 &#xff08;1&#xff09;字符串以\0作为结束标志&#xff0c;strlen函数返回的是在字符串中\0之前出现的字符个数&#xff08;不包含\0&#xff09;。 &#xff08…

【每日运维】大文件的分割与合并

产生背景 特殊单位需要将文件刻盘带入&#xff0c;并且刻盘有大小限制&#xff0c;所以有了这个需求 推荐方法 个人电脑上使用 split 命令指定大小分割Linux 服务上使用 cat 命令进行合并使用 md5sum 命令校验包的完整性 方法演示 需要将一个按照100M分割后刻盘导入 在个…

数据结构 ~ 树

什么是树 - tree 一种分层数据的抽象模型&#xff1b; 如&#xff1a;DOM、级联选择、树形控件&#xff0c;js 中没有树 可以用 Object 构建树&#xff1a; const tree {val: a,children: [{val: a-1,children: [{val: a-1-1,children: []}]},{val: a-2,children: [{val: a…

mapbox绘制多边形

1、实现效果 请忽略马赛克 2、实现思路 绘制一个填充的多边形&#xff0c;再描个边框。 3、实现代码 绘制多边形函数 drawPolygon() {map.addSource(maine, {type: geojson,data: https://asc-test1.oss-cn-beijing.aliyuncs.com/2023/07/05/45f6bd80f2f34d79b3e457b31ec5d…

云原生网关如何实现安全防护能力

作者&#xff1a;刘晓瑞(钰诚) 云原生网关&#xff1a;将安全、流量和微服务三合一 作为面向南北向的公网网关&#xff0c;使用 Waf 防护异常流量是很常规的需求&#xff0c;而且随着互联网环境变得越来越复杂&#xff0c;用户对防护的诉求是持续增强的&#xff0c;常规做法是…

需要我怎么帮你?关于维护Nutsdb开源社区的思考

背景 近来有人问我打算怎么继续维护Nutsdb社区&#xff0c;他们当中不乏有开源项目的狂热爱好者&#xff0c;发起了好几个几千star的开源项目&#xff0c;也有对Nutsdb感兴趣的&#xff0c;想来问问后续的计划。这不禁让我回想到从刚开始参与这个项目到最近这段时间一个人维护…

三维重建的工作

文章目录 一、北京&#xff1a;二、广州&#xff1a;三、深圳&#xff1a; 一、北京&#xff1a; 链接 三维重建技术是自动驾驶领域4D真值数据生成的核心基础能力。融合LiDAR、Camera、IMU、轮速计等传感器数据像BlockNeRF一样重建城市级别逼真精细的三维场景&#xff0c;将…