css3制作3d盒子

本文首发于我的博客这是我的github,欢迎star。

一直想做一个立体的盒子,前段时间刚好看见掘金上有位朋友发了篇关于3d盒子的文章,看了决定自己做一下,再写一些和盒子互动的操作。这里是要做的效果,应该要翻过墙才可以访问,这里是github地址,可以直接查看代码,上边还有别的有趣的demo,欢迎来访。

首先看下怎么做一个静止的盒子,用到了css3transform。将盒子六个面放在一个div里,将这个div定位好,用transform属性改变不同的面的朝向,再将其向前移动盒子宽度的一半,盒子就做好了。html结构如下:

    <div class="wraper">
        <div class="cube">
            <div class="front">Front</div>
            <div class="end">End</div>
            <div class="left">Left</div>
            <div class="right">Right</div>
            <div class="top">Top</div>
            <div class="bottom">Bottom</div>
        </div>
    </div>

设置css,不熟悉transform可以看下下边这张图,需要注意的是元素的轴是跟着元素转动而转动的,所以几个面的translateZ属性的值都是一样的,旋转到正确的朝向然后向前平移盒子一半的宽度。

3d_axes.png
body,
html {
    height: 100%;
    padding: 0;
    margin: 0;
    /*将border和padding绘制在设置宽高之内*/
    box-sizing: border-box;      
    overflow: hidden;
}
.wraper {
    width: 260px;
    height: 260px;
    margin: 128px auto;
    /*景深,可以理解为视角到3D物体的距离,设置于舞台元素上*/
    perspective: 1000px;
}
.cube {
    height: 100%;
    width: 100%;
    position: relative;
    /*子元素继承3D效果*/
    transform-style: preserve-3d;
    /*加上下边这句可以体现出立体感*/
    /*transform: rotateX(-30deg) rotateY(-45deg);*/
}
.cube>div {
    width: 100%;
    height: 100%;
    position: absolute;
    top: 0;
    left: 0;
    background-color: rgba(0, 0, 0, .8);
    text-align: center;
    line-height: 260px;
    color: #fff;
    font-size: 48px;
    border: 2px solid #fff;
    /*设置文本内容不可选*/
    user-select: none;
}
.front {
    /*正面不用旋转,直接向前平移半个盒子宽*/
    transform: translateZ(130px);
}
.end {
    /*向后旋转,以确保盒子上的文字朝向盒子外*/
    transform: rotateY(180deg) translateZ(130px);
}
.top {
    transform: rotateX(90deg) translateZ(130px);
}
.bottom {
    transform: rotateX(-90deg) translateZ(130px);
}
.left {
    transform: rotateY(-90deg) translateZ(130px);
}
.right {
    transform: rotateY(90deg) translateZ(130px);
}

下来就是鼠标拖动盒子的核心代码了,先理一下思路,鼠标拖动盒子,就是记录下盒子的角度,然后点击鼠标并拖动的时候根据鼠标移动方向和距离重新计算出盒子的旋转角度,鼠标松开时取消监听,记录盒子角度。这和我之前写的一个例子鼠标拖动div 有点相似:

var cube = document.querySelector(".cube"),
    downX, downY, moveX, moveY, tempX, tempY, degX = 0, degY = 0;

window.onmousedown = function (e) {
    e = e || event;
    downX = e.clientX;          //获取鼠标点下去时的坐标
    downY = e.clientY;

    window.onmousemove = function (e) {
        e = e || event;
        moveX = e.clientX - downX;          //算出鼠标移动的距离
        moveY = e.clientY - downY;
        //根据一定比例将变化反应在盒子上,改变比例5可以调节拖动的速度
        tempX = degX + moveX / 5;           
        tempY = degY - moveY / 5;
        cube.style.transform = "rotatex(" + tempY + "deg) rotatey(" + tempX + "deg)";
    };

};

window.onmouseup = function (e) {
    e = e || event;
    degX += moveX / 5;          //鼠标松开时将拖动期间改变的最终结果保存
    degY += - moveY / 5;
    window.onmousemove = null;          //取消监听
};

再加一个使用滚轮改变景深的函数,因为景深太小视角会到盒子里边,并不是很好看,所以限制了一下,在景深小于300px时将不会减小。

!function () {
    var n = 1000;
    var wraper = document.querySelector('.wraper');
    wraper.style.perspective = n + 'px';
    window.onmousewheel = function (e) {
        e = e || event;
        if (e.wheelDelta) {  //判断浏览器IE,谷歌滑轮事件
            if (e.wheelDelta > 0) { //当滑轮向上滚动时减小景深
                wraper.style.perspective = n - 50 + 'px';
                if (n > 350) {
                    n = n - 50;
                }
            }
            if (e.wheelDelta < 0) { //当滑轮向下滚动时增加景深
                wraper.style.perspective = n + 50 + 'px';
                n += 50;
            }
        } else if (e.detail) {  //Firefox滑轮事件
            if (e.detail > 0) {
                wraper.style.perspective = n - 50 + 'px';
                if (n > 350) {
                    n = n - 50;
                }
            }
            if (e.detail < 0) {
                wraper.style.perspective = n + 50 + 'px';
                n += 50;
            }
        }
    };
}();

好了,到这里这个盒子看起来已经很3D了,你可以直接在github上复制代码查看效果,我多加了一个入场动画,喜欢可以顺手点个star。有什么建议或问题可以在博客下方留言,欢迎探讨,共同学习。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352

推荐阅读更多精彩内容