搜索

查看: 3044|回复: 11

[CSS/HTML] 通过CSS数学函数实现动画特效

[复制链接]
发表于 2023-5-4 16:46:59 | 显示全部楼层 |阅读模式
Editor 2023-5-4 16:46:59 3044 11 看全部
目录
  • 前言
  • CSS 数学函数
  • 绝对值
  • 中位数
  • 数轴上两点距离
  • 三角函数
  • 例子
  • 一维交错动画
  • 初始状态
  • 应用动画
  • 交错动画
  • 二维交错动画
  • 初始状态
  • 应用动画
  • 交错动画
  • 另一种动画
  • 余弦波动动画
  • 初始状态
  • 余弦排列
  • 波动动画
  • 交错动画

    前言

    大家好,这里是 CSS 兼 WebGL 魔法使——alphardex。
    之前一直在玩 three.js ,接触了很多数学函数,用它们创造过很多特效。于是我思考:能否在 CSS 中也用上这些数学函数,但发现 CSS 目前还没有,据说以后的新规范会纳入,估计也要等很久。
    然而,我们可以通过一些小技巧,来创作出一些属于自己的 CSS 数学函数,从而实现一些有趣的动画效果。
    让我们开始吧!

    CSS 数学函数

    注意:以下的函数用原生 CSS 也都能实现,这里用 SCSS 函数只是为了方便封装,封装起来的话更方便调用


    绝对值

    绝对值就是正的还是正的,负的变为正的
    可以创造 2 个数,其中一个数是另一个数的相反数,比较它们的最大值,即可获得这个数的绝对值


    @function abs($v) {
      @return max(#{$v}, calc(-1 * #{$v}));
    }

    中位数
    原数减 1 并乘以一半即可


    @function middle($v) {
      @return calc(0.5 * (#{$v} - 1));
    }

    数轴上两点距离

    数轴上两点距离就是两点所表示数字之差的绝对值,有了上面的绝对值公式就可以直接写出来


    @function dist-1d($v1, $v2) {
      $v-delta: calc(#{$v1} - #{$v2});
      @return #{abs($v-delta)};
    }

    三角函数

    其实这个笔者也不会实现~不过之前看到过好友 chokcoco 的一篇文章写到了如何在 CSS 中实现三角函数,在此表示感谢


    @function fact($number) {
      $value: 1;
      @if $number>0 {
        @for $i from 1 through $number {
          $value: $value * $i;
        }
      }
      @return $value;
    }
    @function pow($number, $exp) {
      $value: 1;
      @if $exp>0 {
        @for $i from 1 through $exp {
          $value: $value * $number;
        }
      } @else if $exp

    例子

    以下的几个动画特效演示了上面数学函数的作用


    一维交错动画


    初始状态

    创建一排元素,用内部阴影填充,准备好我们的数学函数



      
      ...(此处省略14个 list-item)
      


    body {
      display: flex;
      justify-content: center;
      align-items: center;
      min-height: 100vh;
      margin: 0;
      background: #222;
    }
    :root {
      --blue-color-1: #6ee1f5;
    }
    (这里复制粘贴上文所有的数学公式)
    .list {
      --n: 16;
      display: flex;
      flex-wrap: wrap;
      justify-content: space-evenly;
      &-item {
        --p: 2vw;
        --gap: 1vw;
        --bg: var(--blue-color-1);
        @for $i from 1 through 16 {
          &:nth-child(#{$i}) {
            --i: #{$i};
          }
        }
        padding: var(--p);
        margin: var(--gap);
        box-shadow: inset 0 0 0 var(--p) var(--bg);
      }
    }

    2021823144956278.png

    2021823144956278.png



    应用动画

    这里用了 2 个动画:grow 负责将元素缩放出来;melt 负责“融化”元素(即消除阴影的扩散半径)



      
      ...(此处省略14个 list-item)
      


    .list {
      &.grow-melt {
        .list-item {
          --t: 2s;
          animation-name: grow, melt;
          animation-duration: var(--t);
          animation-iteration-count: infinite;
        }
      }
    }
    @keyframes grow {
      0% {
        transform: scale(0);
      }
      50%,
      100% {
        transform: scale(1);
      }
    }
    @keyframes melt {
      0%,
      50% {
        box-shadow: inset 0 0 0 var(--p) var(--bg);
      }
      100% {
        box-shadow: inset 0 0 0 0 var(--bg);
      }
    }

    2021823145433307.gif

    2021823145433307.gif



    交错动画
    [ol]
       
  • 计算出元素下标的中位数
       
  • 计算每个元素 id 到这个中位数的距离
       
  • 根据距离算出比例
       
  • 根据比例算出 delay
    [/ol]

      
      ...(此处省略14个 list-item)
      


    .list {
      &.middle-stagger {
        .list-item {
          --m: #{middle(var(--n))}; // 中位数,这里是7.5
          --i-m-dist: #{dist-1d(var(--i), var(--m))}; // 计算每个id到中位数之间的距离
          --ratio: calc(var(--i-m-dist) / var(--m)); // 根据距离算出比例
          --delay: calc(var(--ratio) * var(--t)); // 根据比例算出delay
          --n-delay: calc((var(--ratio) - 2) * var(--t)); // 负delay表示动画提前开始
          animation-delay: var(--n-delay);
        }
      }
    }

    2021823145248772.gif

    2021823145248772.gif


    地址:Symmetric Line Animation


    二维交错动画

    初始状态

    如何将一维的升成二维?应用网格系统即可



      
      ...(此处省略62个 grid-item)
      


    .grid {
      $row: 8;
      $col: 8;
      --row: #{$row};
      --col: #{$col};
      --gap: 0.25vw;
      display: grid;
      gap: var(--gap);
      grid-template-rows: repeat(var(--row), 1fr);
      grid-template-columns: repeat(var(--col), 1fr);
      &-item {
        --p: 2vw;
        --bg: var(--blue-color-1);
        @for $y from 1 through $row {
          @for $x from 1 through $col {
            $k: $col * ($y - 1) + $x;
            &:nth-child(#{$k}) {
              --x: #{$x};
              --y: #{$y};
            }
          }
        }
        padding: var(--p);
        box-shadow: inset 0 0 0 var(--p) var(--bg);
      }
    }

    2021823145602666.png

    2021823145602666.png


    应用动画

    跟上面的动画一模一样



      
      ...(此处省略62个 grid-item)
      


    .grid {
      &.grow-melt {
        .grid-item {
          --t: 2s;
          animation-name: grow, melt;
          animation-duration: var(--t);
          animation-iteration-count: infinite;
        }
      }
    }

    2021823145657893.gif

    2021823145657893.gif


    交错动画
    [ol]
       
  • 计算出网格行列的中位数
       
  • 计算网格 xy 坐标到中位数的距离并求和
       
  • 根据距离算出比例
       
  • 根据比例算出 delay
    [/ol]

      
      ...(此处省略62个 grid-item)
      


    .grid {
      &.middle-stagger {
        .grid-item {
          --m: #{middle(var(--col))}; // 中位数,这里是7.5
          --x-m-dist: #{dist-1d(var(--x), var(--m))}; // 计算x坐标到中位数之间的距离
          --y-m-dist: #{dist-1d(var(--y), var(--m))}; // 计算y坐标到中位数之间的距离
          --dist-sum: calc(var(--x-m-dist) + var(--y-m-dist)); // 距离之和
          --ratio: calc(var(--dist-sum) / var(--m)); // 根据距离和计算比例
          --delay: calc(var(--ratio) * var(--t) * 0.5); // 根据比例算出delay
          --n-delay: calc(
            (var(--ratio) - 2) * var(--t) * 0.5
          ); // 负delay表示动画提前开始
          animation-delay: var(--n-delay);
        }
      }
    }

    2021823145809665.gif

    2021823145809665.gif


    地址:Symmetric Grid Animation


    另一种动画

    可以换一种动画 shuffle(穿梭),会产生另一种奇特的效果



      
      ...(此处省略254个 grid-item )
      


    .grid {
      $row: 16;
      $col: 16;
      --row: #{$row};
      --col: #{$col};
      --gap: 0.25vw;
      &-item {
        --p: 1vw;
        transform-origin: bottom;
        transform: scaleY(0.1);
      }
      &.shuffle {
        .grid-item {
          --t: 2s;
          animation: shuffle var(--t) infinite ease-in-out alternate;
        }
      }
    }
    @keyframes shuffle {
      0% {
        transform: scaleY(0.1);
      }
      50% {
        transform: scaleY(1);
        transform-origin: bottom;
      }
      50.01% {
        transform-origin: top;
      }
      100% {
        transform-origin: top;
        transform: scaleY(0.1);
      }
    }

    2021823145855492.gif

    2021823145855492.gif


    地址:Shuffle Grid Animation


    余弦波动动画

    初始状态

    创建 7 个不同颜色的(这里直接选了彩虹色)列表,每个列表有 40 个子元素,每个子元素是一个小圆点
    让这 7 个列表排列在一条线上,且 z 轴上距离错开,设置好基本的 delay



      
       
        ...(此处省略39个 list-item)
      
      ...(此处省略6个 list)


    .lists {
      $list-count: 7;
      $colors: red, orange, yellow, green, cyan, blue, purple;
      position: relative;
      width: 34vw;
      height: 2vw;
      transform-style: preserve-3d;
      perspective: 800px;
      .list {
        position: absolute;
        top: 0;
        left: 0;
        display: flex;
        transform: translateZ(var(--z));
        @for $i from 1 through $list-count {
          &:nth-child(#{$i}) {
            --bg: #{nth($colors, $i)};
            --z: #{$i * -1vw};
            --basic-delay-ratio: #{$i / $list-count};
          }
        }
        &-item {
          --w: 0.6vw;
          --gap: 0.15vw;
          width: var(--w);
          height: var(--w);
          margin: var(--gap);
          background: var(--bg);
          border-radius: 50%;
        }
      }
    }

    2021823150011901.png

    2021823150011901.png


    余弦排列

    运用上文的三角函数公式,让这些小圆点以余弦的一部分形状进行排列


    .lists {
      .list {
        &-item {
          $item-count: 40;
          $offset: pi() * 0.5;
          --wave-length: 21vw;
          @for $i from 1 through $item-count {
            &:nth-child(#{$i}) {
              --i: #{$i};
              $ratio: ($i - 1) / ($item-count - 1);
              $angle-unit: pi() * $ratio;
              $wave: cos($angle-unit + $offset);
              --single-wave-length: calc(#{$wave} * var(--wave-length));
              --n-single-wave-length: calc(var(--single-wave-length) * -1);
            }
          }
          transform: translateY(var(--n-single-wave-length));
        }
      }
    }

    2021823150049651.png

    2021823150049651.png


    波动动画

    对每个小圆点应用上下平移动画,平移的距离就是余弦的波动距离


    .lists {
      .list {
        &-item {
          --t: 2s;
          animation: wave var(--t) infinite ease-in-out alternate;
        }
      }
    }
    @keyframes wave {
      from {
        transform: translateY(var(--n-single-wave-length));
      }
      to {
        transform: translateY(var(--single-wave-length));
      }
    }

    2021823150142668.gif

    2021823150142668.gif


    交错动画

    跟上面一个套路,计算从中间开始的 delay,再应用到动画上即可


    .lists {
      .list {
        &-item {
          --n: #{$item-count + 1};
          --m: #{middle(var(--n))};
          --i-m-dist: #{dist-1d(var(--i), var(--m))};
          --ratio: calc(var(--i-m-dist) / var(--m));
          --square: calc(var(--ratio) * var(--ratio));
          --delay: calc(
            calc(var(--square) + var(--basic-delay-ratio) + 1) * var(--t)
          );
          --n-delay: calc(var(--delay) * -1);
          animation-delay: var(--n-delay);
        }
      }
    }

    2021823150228880.gif

    2021823150228880.gif


    地址:Rainbow Sine

    到此这篇关于通过CSS数学函数实现动画特效的文章就介绍到这了,更多相关CSS实现动画特效内容请搜索知鸟论坛以前的文章或继续浏览下面的相关文章希望大家以后多多支持知鸟论坛
  • 知鸟论坛永久地址发布页:www.zn60.me
    回复

    使用道具 举报

    发表于 2023-6-29 00:20:32 | 显示全部楼层
    井底燕雀傥 2023-6-29 00:20:32 看全部
    这个帖子不回对不起自己!我想我是一天也不能离开知鸟论坛
    知鸟论坛永久地址发布页:www.zn60.me
    回复

    使用道具 举报

    发表于 2023-6-29 00:38:09 | 显示全部楼层
    462710480 2023-6-29 00:38:09 看全部
    楼主太厉害了!楼主,I*老*虎*U!我觉得知鸟论坛真是个好地方!
    知鸟论坛永久地址发布页:www.zn60.me
    回复

    使用道具 举报

    发表于 2023-6-29 16:03:32 | 显示全部楼层
    啤酒瓶空了缓 2023-6-29 16:03:32 看全部
    楼主发贴辛苦了,谢谢楼主分享!我觉得知鸟论坛是注册对了!
    知鸟论坛永久地址发布页:www.zn60.me
    回复

    使用道具 举报

    发表于 2023-6-29 16:09:32 | 显示全部楼层
    贺老师 2023-6-29 16:09:32 看全部
    其实我一直觉得楼主的品味不错!呵呵!知鸟论坛太棒了!
    知鸟论坛永久地址发布页:www.zn60.me
    回复

    使用道具 举报

    发表于 2023-6-29 16:55:33 | 显示全部楼层
    小妖花满楼满fx 2023-6-29 16:55:33 看全部
    这个帖子不回对不起自己!我想我是一天也不能离开知鸟论坛
    知鸟论坛永久地址发布页:www.zn60.me
    回复

    使用道具 举报

    发表于 2023-6-29 17:17:55 | 显示全部楼层
    ffycxyw2274436 2023-6-29 17:17:55 看全部
    这东西我收了!谢谢楼主!知鸟论坛真好!
    知鸟论坛永久地址发布页:www.zn60.me
    回复

    使用道具 举报

    发表于 2023-6-29 19:03:41 | 显示全部楼层
    知足常乐77 2023-6-29 19:03:41 看全部
    论坛不能没有像楼主这样的人才啊!我会一直支持知鸟论坛
    知鸟论坛永久地址发布页:www.zn60.me
    回复

    使用道具 举报

    发表于 2023-6-29 19:13:22 | 显示全部楼层
    风来时狂放 2023-6-29 19:13:22 看全部
    楼主,我太崇拜你了!我想我是一天也不能离开知鸟论坛
    知鸟论坛永久地址发布页:www.zn60.me
    回复

    使用道具 举报

    发表于 2023-6-30 01:18:47 | 显示全部楼层
    心随674 2023-6-30 01:18:47 看全部
    既然你诚信诚意的推荐了,那我就勉为其难的看看吧!知鸟论坛不走平凡路。
    知鸟论坛永久地址发布页:www.zn60.me
    回复

    使用道具 举报

    • 您可能感兴趣
    点击右侧快捷回复 【请勿灌水】
    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则 返回列表

    RSS订阅| SiteMap| 小黑屋| 赞兔论坛
    联系邮箱E-mail:zniao@foxmail.com
    快速回复 返回顶部 返回列表