分享CodePen上6个酷炫demo特效

最近创作灵感匮乏, 来 CodePen 上找找灵感, 同时也给同样需要获取灵感的 coder (程序员们)带来一点点想象空间.

分享CodePen上6个酷炫demo特效

首先分享一下 CodePen demo 集合网址:

  • https://codepen.io/spark

1. 像素背景

分享CodePen上6个酷炫demo特效

这是 Wakana Y.K. 的一个巧妙的技巧:将图像的低分辨率版本放在顶部,并将其扩展以占据与原始图像相同的大小,使浏览器对其进行像素化。然后,悬停时的蒙版(使用 JS 更新位置)完成剩下的工作。巧妙而有效。

css代码如下:

@import url("https://fonts.googleapis.com/css2?family=Lexend:wght@400&display=swap");
:root {
  --x50%;
  --y50%;
  --radius30vmin;
  --blur3vmax;
}
* {
  margin0;
  padding0;
  box-sizing: border-box;
}
html,
body {
  background-color: black;
  overscroll-behavior-x: none;
  overscroll-behavior-y: none;
  overflow: hidden;
}
body {
  width100vw;
  height100vh;
  font-family"Lexend", serif;
  text-align: center;
  line-height1;
  display: flex;
  justify-content: center;
  align-items: center;
}
main {
  z-index1;
}
h1 {
  font-size7vw;
  color: white;
  text-shadow1px 1px 1vw rgba(0000.3);
}
#bg {
  position: absolute;
  top0;
  left0;
  width100vw;
  height100vh;
}
#bg img {
  position: absolute;
  top0;
  left0;
  width100%;
  height100%;
  object-fit: cover;
  object-position: center center;
  pointer-events: none;
  -webkit-user-select: none;
  user-select: none;
}
#bg_focus {
  mask-imageradial-gradient(
    circle 50vmin at var(--x) var(--y),
    black var(--radius),
    transparent calc(var(--radius) + var(--blur)),
    transparent
  );
}
#bg_pixelate {
  image-rendering: pixelated;
}
/*
#bg img:nth-of-type(2) {
  filter: hue-rotate(50deg);
  animation-name: animation;
  animation-duration: 0.5s;
  animation-iteration-count: infinite;
  animation-timing-function: linear;
  animation-direction: alternate;
}
@keyframes animation {
  0% {
    opacity: 0;
  }
  100% {
    opacity: 1;
  }
}
*/

2. 烟火特效

分享CodePen上6个酷炫demo特效

用技术复原童年的烟火味, css代码:

<css-doodle click-to-update><style>
  @grid30 / 60vmin noclip;
  border-radius: 50%;
  backgroundhsl(@y(* -12), 80%60%);
  backgroundlch(90, 120, @y(*10deg));
  animationf @once.p(5s, 7.5s) ease infinite;
  animation-delay: $s(-90/@I*@i(-1)*@x/@y);
  @keyframes f {
    0%, 25%, 80% {opacity1scale: .2}
    25%, 75%, 100% {opacity0scale: @r(23)}
  }
 </style></css-doodle>

3. 动态模糊


Tom Hinton 的这个演示吸引我的是它看起来多么随机和奇怪(以一种好的方式)(这在 Tom 的艺术作品中相对常见。ThreeJS 中着色器和网格的组合。

js 代码如下:

let camera, scene, renderer, clock;
let uniforms;

function init({
  const container = document.getElementById("shader");

  clock = new THREE.Clock();
  camera = new THREE.Camera();
  camera.position.z = 1;

  scene = new THREE.Scene();

  const geometry = new THREE.PlaneBufferGeometry(22);
  
    const texture = new THREE.TextureLoader().load( 'https://upload.wikimedia.org/wikipedia/commons/8/84/Male_and_female_chicken_sitting_together.jpg' );

  uniforms = {
    u_time: { type"f"value1.0 },
    u_resolution: { type"v2"valuenew THREE.Vector2() },
    uTexture: {
      value: texture
    }
  };

  const material = new THREE.ShaderMaterial({
    uniforms,
    vertexShaderdocument.getElementById("vertex").textContent,
    fragmentShaderdocument.getElementById("fragment").textContent
  });
  
  material.transparent = true

  const mesh = new THREE.Mesh(geometry, material);
  scene.add(mesh);

  renderer = new THREE.WebGLRenderer({alphatrue});
  renderer.setPixelRatio(window.devicePixelRatio);

  container.appendChild(renderer.domElement);

  onWindowResize();
  window.addEventListener("resize", onWindowResize);
}

function onWindowResize({
  renderer.setSize(window.innerWidth, window.innerHeight);
  uniforms.u_resolution.value.x = renderer.domElement.width;
  uniforms.u_resolution.value.y = renderer.domElement.height;
}

function render({
  uniforms.u_time.value = clock.getElapsedTime();
  renderer.render(scene, camera);
}

function animate({
  render();
  requestAnimationFrame(animate);
}

init();
animate();

4. 雪球圣诞树

Amit Sheen 在 YouTube 上直播了整个过程(您仍然可以在他的频道上观看),让每个人都了解如何使用 HTML 和 CSS 创建这棵 3D 外观的圣诞树。

css代码:

*, *::before, *::after {
    padding0;
    margin0 auto;
    box-sizing: border-box;
}

body {
  background-color#001;
  color#fff;
  min-height100vh;
  display: grid;
  place-items: center;
  perspective1000px;
  overflow: hidden;
  
  * {
    transform-style: preserve-3d;
  }
}

$srDuration: 120s;
$flDuration: 12s;

.scene {
  position: relative;
  animation: sceneRotate $srDuration infinite linear;
  
  @keyframes sceneRotate {
    to { rotate: y 1turn; }
  }
  
  * { position: absolute; }
}

.floor {
  inset: -100em;
  background-color: green;
  transformrotateX(-90degtranslateZ(15em);
  background-image:
    radial-gradient(closest-side, transparent, #001),
    radial-gradient(closest-side, #000, transparent 15em),
    repeating-linear-gradient(#fff2 0, transparent, #fff2 2em),
    repeating-linear-gradient(90deg, #fff2 0, transparent, #fff2 2em)
    ;
}

.snowglobe {
  cursor: pointer;
  animationvar(--animation, snowglobe) $flDuration;
  
  @keyframes snowglobe {
    0%, 100% { pointer-events: none; }
  }

  &:active {
    --animation: none;
  }
}

.text {
  top: -20em;
  animation: sceneRotate $srDuration infinite linear reverse;
  
  i {
    font-size3em;  
    width: max-content; 
    translate: -50%;
    animationvar(--animation, text) $flDuration ease-in-out;
    
    @keyframes text {
      0%, 95% { opacity0; } 
      100% { opacity1; } 
    }  
  }
}

.base {
  transformtranslateY(11em);
  
  i {
    inset: -10em -2.7em;
    background-color: maroon;
    background-imageradial-gradient(circle, transparent, 90%, #0007);
    transformrotateX(-90degrotate(var(--angle, 0));
    
    @for $i from 0 to 6 {
      &:nth-child(#{$i + 1}) {
        --angle: #{$i * 30deg};
      }      
    }
    
    &::before, &::after {
      content'';
      position: absolute;
      width100%height4em;
      box-shadow0 0 1em #0007 inset;
      background-color: inherit;
      transform-origin: top;
      transformrotateX(90deg);
    }
    
    &::after { top100%; }
  }
}

.glass {
  inset: -15em;
  background-imageradial-gradient(farthest-side at 50% 60%, transparent, 95%, #fff5);
  border-radius50%;
  animation: sceneRotate $srDuration infinite linear reverse;
  
  &::after {
    content'';
    position: absolute;
    inset2em;
    border-radius50%;
    background-imageradial-gradient(circle at 50% 60%, transparent 65%, #ff7a 85%);
    filterblur(1em);
    animation: lightblur $srDuration infinite ease-in-out;
    
    @keyframes lightblur {
      0%, 100% { rotate: -45deg; }
      50% { rotate45deg; }
    }
  }
}

.tree {
  .trunk {
    inset: -7em -1em;
    transformtranslateY(4em);
    background-color: brown;
    border-radius50% / 90% 90% 3% 3%;
    animation: sceneRotate $srDuration infinite linear reverse;
  }

  .leafs i {
    topvar(--top, 0);
    width8em;
    aspect-ratio1;
    background-imageradial-gradient(circle at top left, #000, green);
    border-radius0 0 100% 0;
    transform-origin: top left;
    transformrotateY(var(--ry)) rotateX(45degrotateZ(45degscale(var(--scale, 1));
    
    $angle0;
    @for $i from 0 to 48 {
      &:nth-child(#{$i + 1}) {
        $angle: $angle + 60 + random(60);
        --top: #{-5 + $i * 0.2}em;
        --ry: #{$angle}deg;
        --scale: #{0.25 + $i / 75};
      }      
    }
  }

  .lights i {
    inset: -0.4em;
    background-colorhsl(var(--hue) 100% 50%);
    background-imageradial-gradient(circle at top, transparent, #0007);
    box-shadow0 0 0.5em #fff7;
    transformtranslatey(var(--ty)) rotateY(var(--ry, 0)) translateZ(var(--tz, 5em));
    border-radius50%;
    animation: lightRotate $srDuration infinite linear reverse;
    
    @keyframes lightRotate {
      from { transformtranslatey(var(--ty)) rotateY(var(--ry, 0)) translateZ(var(--tz, 5em)) rotateY(calc(0deg - var(--ry))); }
      to { transformtranslatey(var(--ty)) rotateY(var(--ry, 0)) translateZ(var(--tz, 5em)) rotateY(calc(360deg - var(--ry))); }
    }

    $angle: 0;
    @for $i from 0 to 48 {
      &:nth-child(#{$i + 1}) {
        $angle: $angle + 60 + random(60);
        --ty: #{-4 + $i * 0.25}em;
        --tz: #{1 + $i * 0.09}em;
        --ry: #{$angle}deg;
        --hue: #{random(360)};
      }      
    }
  }
}

.flakes {
  transformtranslateY(11emrotateX(-90deg);
  animationvar(--animation, flakes) $flDuration ease-in-out;
  
  @keyframes flakes {
    0%, 100% { transformtranslateY(11emrotateX(-90deg); }
    25%, 50% { transformtranslateY(0emrotateX(-90deg); }
  }
  
  i {
    inset: -0.35em;
    background-colorhsl(55 100% var(--light, 100%));
    rotatevar(--rotate1);
    transformtranslateX(var(--tx1));
    animation:
      var(--animation, flakeRotate) $flDuration ease-in-out,
      var(--animation, flakeTransform) $flDuration ease-in-out;
    
    @keyframes flakeTransform {
      0% { transformrotateY(0degtranslateX(var(--tx1)) rotate3d(1110deg); } 
      25% { transformrotateY(var(--ry)) translateX(var(--tx2)) rotate3d(111, calc(var(--r3d) * 0.25)); animation-timing-function: linear; } 
      50% { transformrotateY(var(--ry)) translateX(var(--tx2)) rotate3d(111, calc(var(--r3d) * 0.5)); } 
      100% { transformrotateY(0degtranslateX(var(--tx1)) rotate3d(111, var(--r3d)); } 
    }
    
    @keyframes flakeRotate {
      from { rotatevar(--rotate1); }
      to { rotatevar(--rotate2); }
    }

    @for $i from 0 to 256 {
      &:nth-child(#{$i + 1}) {
        $rrandom(360);
        --rotate1: #{$r}deg;
        --rotate2: #{$r + 5 * 360}deg;
        --tx1: #{(random(100) / 100) * 5 + 3}em;
        --tx2: #{(random(100) / 100) * 6 + 7}em;
        --light: #{70 + random(30)}%;
        --ry: #{random(120) - 45}deg;
        --r3d: #{random(8) + 8}turn;
      }      
    }
  }
}

.star {
  position: absolute;
  top: -5.5em;

  i {
    position: absolute;
    left: -1.25em;
    width2.5emheight0.5em;
    background-color: gold;
    background-imagelinear-gradient(90deg, transparent, #000a);
    clip-pathpolygon(0 0100% 050% 100%);
    transform-origin: top;
    transform
      rotate(var(--angle))
      translateY(-0.41em)
      rotateX(var(--rx, 35deg));

    &:nth-child(5n + 1) { --angle0deg; }
    &:nth-child(5n + 2) { --angle72deg; }
    &:nth-child(5n + 3) { --angle144deg; }
    &:nth-child(5n + 4) { --angle216deg; }
    &:nth-child(5n + 5) { --angle288deg; }

    &:nth-child(n + 6) { --rx: -35deg; }
  }
}

5. 点状旋转加载动画

分享CodePen上6个酷炫demo特效

这是 Josetxu 用 HTML 和 CSS 创建的催眠加载器。一个有趣的动画组合创造了这种效果,点在改变大小(和 z 索引)的同时移动。

css代码:

:root {
  --w#fafafa
  --b#141414
  --s1s;
  --dcalc(var(--s) / 6);
}

$dvar(--d);

* {
  box-sizing: border-box;
}

body {
  margin0;
  padding0;
  width100vw;
  height100vh;
  overflow: hidden;
}

input {
  width100vw;
  height100vh;
  position: absolute;
  z-index4;
  opacity0;
  cursor: pointer;
  &:checked ~ div {
    filterinvert(1);
  }
  &:checked + .bg:before {
    content"CLICK TO DARK";   
  }
}

.bg:before {
  content"CLICK TO LIGHT";
  position: absolute;
  colorvar(--w);
  width100%;
  text-align: center;
  bottom10vh;
  font-family: Arial, Helvetica, serif;
  font-size12px;
  text-shadow0 0 1px var(--w);
  opacity0.25;
}

body.dots {
  display: flex;
  align-items: center;
  justify-content: center;
}

.bg {
  width100vw;
  height100vh;
  position: absolute;
  backgroundvar(--b);
  z-index: -2;
}

.dots {
  width50vmin;
  height50vmin;
  position: relative;
}

.ring {
  border1.5vmin solid var(--w);
  width64%;
  height64%;
  border-radius100%;
  z-index0;
  box-shadow0 0 0 1vmin var(--b), 0 0 0 1vmin var(--b) inset;
}

.dot {
  width50%;
  position: absolute;
  height7vmin;
  left0;
  transform-origin100% 50%;
  z-index: -1;
  animation
    over-ring calc(var(--s) * 2) linear 0s infinite, 
    spin calc(var(--s) * 8) linear 0s infinite;
  &:before {
    content"";
    width5.5vmin;
    height5.5vmin;
    left0;
    box-sizing: border-box;
    border1vmin solid var(--b);
    position: absolute;
    backgroundvar(--w);
    border-radius100%;
    animation: ball var(--s) ease-in-out 0s infinite alternate;
  }
  @for $i from 1 through 12 { 
    &:nth-child(#{$i}) {
      $n: (($i - 1) * -1);
      $n4: ($n * 4);
      animation-delaycalc(#{$d} * #{$n}), calc(#{$d} * #{$n4});
      &:before {
        animation-delaycalc(#{$d} * #{$n});
      }
    }
  }
}

@keyframes spin {
  100% { transformrotate(-360deg); }  
}

@keyframes ball {
  100% { left12vminwidth4vminheight4vmin; } 
}

@keyframes over-ring {
  0%, 50% { z-index: -1; }  
  51%, 100% { z-index1; } 
}

6. Wow 冬季毯子

分享CodePen上6个酷炫demo特效

另一个带有 ThreeJS 和着色器的演示。这次是安娜·禅恩·清道夫(Anna Zenn Scavenger)的作品。这是 CodePen 每周关于对立面挑战的一部分。将鼠标移到毯子上以查看其平稳移动。

js代码:

import * as THREE from 'https://unpkg.com/three@0.118.3/build/three.module.js';

let container, scene, camera, renderer;
let blanket;

// LANDSCAPE / PORTRAIT

let isMobile = /(Android|iPhone|iOS|iPod|iPad)/i.test(navigator.userAgent);
let windowRatio = window.innerWidth / window.innerHeight;
let isLandscape = (windowRatio > 1) ? true : false;

// MOUSE

let isMouseMove = false;
let mouseX = 0;

const clock = new THREE.Clock();

init();
render();

function init({
  
  container = document.querySelector("#scene-container");

  scene = new THREE.Scene();

  initCamera();
  initLights();
  initRenderer();
  
  initBlanket();
  
  window.addEventListener('resize', onWindowResize, false);
  document.addEventListener('mousemove', onMouseMove);
  document.addEventListener('touchmove', onTouchMove);
  window.addEventListener('mouseout', onMouseLeave);
  
}

function initCamera({
  
  camera = new THREE.PerspectiveCamera(40window.innerWidth / window.innerHeight, 11000);
  camera.position.set(02.010.0);
  camera.position.z = (windowRatio > 2) ? ((5 / windowRatio) + 9) : (15 / windowRatio);
  
}

function initLights({

  const dirLight = new THREE.DirectionalLight(0xffffff0.75);
  dirLight.position.set(-0.510-10);
  scene.add(dirLight);

  const ambientLight = new THREE.AmbientLight(0xffffff0.6);
  //ambientLight.position.set(0.5, 10, -5);
  scene.add(ambientLight);

}

function initRenderer({
    
  renderer = new THREE.WebGLRenderer({alphatrueantialiastrue });
  // renderer.setPixelRatio(window.devicePixelRatio > 1.4 ? Math.min(window.devicePixelRatio, 1.25) : Math.min(window.devicePixelRatio, 1.25));
  renderer.setSize(window.innerWidth, window.innerHeight);
  //renderer.outputColorSpace = THREE.SRGBColorSpace;
  container.appendChild(renderer.domElement);
    
}

function initBlanket({
  
  const SIZE = 6.5;
  const RESOLUTION = 75;

  const geometry = new THREE.PlaneBufferGeometry(SIZE, SIZE, RESOLUTION, RESOLUTION);
  geometry.rotateX(-0.5 * Math.PI);

  const tartanMaterial = new THREE.ShaderMaterial({
  
    lightstrue,
    side: THREE.DoubleSide,
  
    extensions: {
      derivativestrue,
    },

    defines: {
      STANDARD'',
      PHYSICAL'',
    },

    uniforms: {
    
      ...THREE.ShaderLib.physical.uniforms,
      roughness: { value0.0 },
      diffuse: {valuenew THREE.Color(0xffffff)},
      time: { value0.0 },
      amplitude: { value0.4 },
      frequency: { value0.4 },
      speed: { value0.3 },
      u_time: { value0.0 },
      u_resolution: { valuenew THREE.Vector2(window.innerWidth, window.innerHeight) }
    
    },

    vertexShader: monkeyPatch(THREE.ShaderChunk.meshphysical_vert, {
    
      header`
        uniform float time;
        uniform float amplitude;
        uniform float speed;
        uniform float frequency;
      
        varying vec2 vUv;

        ${noise()}
      
        float displace(vec3 point) {
          return noise(vec3(point.x * frequency, point.z * frequency, time * speed)) * amplitude;
        }
      
        // http://lolengine.net/blog/2013/09/21/picking-orthogonal-vector-combing-coconuts
        vec3 orthogonal(vec3 v) {
          return normalize(abs(v.x) > abs(v.z) ? vec3(-v.y, v.x, 0.0)
        : vec3(0.0, -v.z, v.y));
        }
      `
,
      // adapted from http://tonfilm.blogspot.com/2007/01/calculate-normals-in-shader.html
      main`
        vec3 displacedPosition = position + normal * displace(position);

        float offset = ${SIZE / RESOLUTION};
        vec3 tangent = orthogonal(normal);
        vec3 bitangent = normalize(cross(normal, tangent));
        vec3 neighbour1 = position + tangent * offset;
        vec3 neighbour2 = position + bitangent * offset;
        vec3 displacedNeighbour1 = neighbour1 + normal * displace(neighbour1);
        vec3 displacedNeighbour2 = neighbour2 + normal * displace(neighbour2);

        /
/ https://i.ya-webdesign.com/images/vector-normals-tangent-16.png
        vec3 displacedTangent = displacedNeighbour1 - displacedPosition;
        vec3 displacedBitangent = displacedNeighbour2 - displacedPosition;

        /
/ https://upload.wikimedia.org/wikipedia/commons/d/d2/Right_hand_rule_cross_product.svg
        vec3 displacedNormal = normalize(cross(displacedTangent, displacedBitangent));
      `,

      '#include <defaultnormal_vertex>': THREE.ShaderChunk.defaultnormal_vertex.replace(
      // transformedNormal will be used in the lighting calculations
      'vec3 transformedNormal = objectNormal;',
      `
vec3 transformedNormal = displacedNormal;`
      ),

      // transformed is the output position
      '#include <morphtarget_vertex>': `
vUv = uv;`,
      '#include <displacementmap_vertex>': `

        transformed = displacedPosition;
      `,
    }),
  
    fragmentShader: monkeyPatch(THREE.ShaderChunk.meshphysical_frag, {
    
      header: `

      
        #define FREQUENCY 40
        #define TILT -60
        #define PATTERN 0.7
      
        varying vec2 vUv;
        uniform vec2 u_resolution;
      
        float coordinateGrid(vec2 r, float lineWidth, float offset, bool doubleLine) {

          float pixel = 0.0;
  
  for(float i = 0.0; i < 2.0; i += PATTERN) {
               
        float x = mod(i, PATTERN * 2.0);
        
        if (doubleLine) {
            
            if (x == 0.0) {
                pixel += 1.0 - step(lineWidth, abs(r.x - i - offset)); //first x line
                pixel += 1.0 - step(lineWidth, abs(r.y - i + offset)); //first y line
            }
 else {
                pixel += 1.0 - step(lineWidth, abs(r.x - i + offset)); //second x line
                pixel += 1.0 - step(lineWidth, abs(r.y - i - offset)); //second y line
            }
            
        } else {
            pixel += 1.0 - step(lineWidth, abs(r.x - i*2.0 - offset)); //first x line
            pixel += 1.0 - step(lineWidth, abs(r.y - i*2.0 + offset)); //first y line
        }
  }

  return pixel;
}
    `
,
    main``,
    '#include <logdepthbuf_fragment>'`
    
    vec2 r = vUv;

    vec4 lightred = vec4(220.0/255.0, 23.0/255.0, 10.0/255.0, 1.0);
    vec4 darkRed = vec4(120.0/255.0, 12.0/255.0, 30.0/255.0, 1.0);
    vec4 yellow = vec4(190.0/255.0, 170.0/255.0, 59.0/255.0, 1.0);
    vec4 white = vec4(242.0/255.0, 242.0/255.0, 203.0/255.0, 0.1);
    vec4 blue = vec4(5.0/255.0, 10.0/255.0, 0.0/255.0, 0.8); 
    vec4 purp = vec4(0.0/255.0, 0.0/255.0, 0.0/255.0, 0.9); 

    vec4 pixel = lightred; // bg color
    
    pixel = mix(pixel, darkRed, coordinateGrid(r, 0.15, 0.0, true)); //paired line
    pixel = mix(pixel, white, coordinateGrid(r, 0.01, 0.005, true)); //paired line
    pixel = mix(pixel, white, coordinateGrid(r, 0.01, -0.35, false)); //paired line
    pixel = mix(pixel, purp, coordinateGrid(r, 0.01, -0.4, false)); //single line
    pixel = mix(pixel, purp, coordinateGrid(r, 0.01, -0.3, false)); //single line
    pixel = mix(pixel, blue, coordinateGrid(r, 0.02, 0.15, true)); //paired line
    pixel = mix(pixel, yellow, coordinateGrid(r, 0.01, 0.05, true)); //paired line

    float stripe = fract( dot(r, vec2(FREQUENCY,TILT)));
    pixel = mix(pixel, darkRed, stripe);
  
          diffuseColor = pixel;
    `
,
    
  })
});

  blanket = new THREE.Mesh(geometry, tartanMaterial);
  blanket.position.set(02.0-0.5);
  blanket.rotation.set(Math.PI * 0.1Math.PI * 0.250);
  scene.add(blanket);

}

function render({
  
  update();
  renderer.render(scene, camera);
  requestAnimationFrame(render);
  
}

function update({
  
  let t = clock.getDelta();
  
  if (isMouseMove) {
    
    blanket.material.uniforms.time.value = 0.1 + 3.0 * mouseX;
    
  } else {
    
    blanket.material.uniforms.time.value += 3.0 * t;
  
  }

}

// * UTILS * 

function monkeyPatch(shader, { defines = '', header = '', main = '', ...replaces }{
  
  let patchedShader = shader;

  const replaceAll = (str, find, rep) => str.split(find).join(rep);
  
  Object.keys(replaces).forEach((key) => {
    
    patchedShader = replaceAll(patchedShader, key, replaces[key])
    
  });

  patchedShader = patchedShader.replace(
    'void main() {',
    `
    ${header}
    void main() {
      ${main}
    `

  );

  return `
    ${defines}
    ${patchedShader}
  `


}

// * EVENTS *

function onWindowResize({
  
  camera.aspect = window.innerWidth / window.innerHeight;
  camera.updateProjectionMatrix();
  renderer.setSize(window.innerWidth, window.innerHeight);
  
}

function onMouseMove(event{
  
  mouseX = (event.clientX / window.innerWidth) * 2 - 1;
  isMouseMove = true;
  console.log(mouseX);
    
}

function onTouchMove(event{
    
  let x = event.changedTouches[0].clientX;
  mouseX = (x / window.innerWidth) * 2 - 1;
  isMouseMove = true;
        
}

function onMouseLeave(event{
  
  console.log('mouseleft');
  isMouseMove = false;
        
}


最后

下周我会继续对话一些优秀的开源和独立开发者, 如果你恰好也是, 欢迎和我们分享你的故事, 如果你是一名程序员, 对现在工作比较迷茫, 也欢迎和我交个朋友, 聊聊你的问题, 让优秀的人祝你一臂之力.


同时如果大家觉得不错, 欢迎点赞反馈, 不辜负每一位开源贡献者的努力, 让技术更美好~

如果大家有好的开源项目或者产品, 欢迎在趣谈前端公众号回复“自荐”.

更多推荐


原文始发于微信公众号(趣谈前端):分享CodePen上6个酷炫demo特效

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/202038.html

(0)
小半的头像小半

相关推荐

发表回复

登录后才能评论
极客之音——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!