IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> JavaScript知识库 -> 【幻妙音符——HTML 实现(效果+代码)】 -> 正文阅读

[JavaScript知识库]【幻妙音符——HTML 实现(效果+代码)】

效果

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
鉴于篇幅以及图片大小,效果仅展示一部分,具体特效欢迎大家体验~

代码

这是HTML代码,主要是引用的JS部分代码。

index.html

<!DOCTYPE html>
<html >
<head>
<meta charset="UTF-8">
<title>主要引用JS代码~</title>
<link rel="stylesheet" href="css/style.css">
</head>
<body>
<div id="three-container"></div>
<script src='js/three.min.js'></script>
<script src='js/bas.min.1.1.3.js'></script>
<script src='js/three_post.js'></script>
<script src='js/OrbitControls.js'></script>
<script src='js/LuminosityHighPassShader.js'></script>
<script src='js/UnrealBloomPass.js'></script>
<script src='js/1b4e36af86e7ae4e4d4f8a3dc.js'></script>
<script src='js/TweenMax.min.js'></script>
<script src="js/index.js"></script>
</body>
</html>

JS部分代码示例:
index.js

const COLORS = {
  red: 0xf54843,
  green: 0x43f565,
  yellow: 0xeff543,
};

const SCENE_CONFIG = {
  pathRadius: 4,
  pathAnimationDuration: 20,
  cameraSpeed: 14.079549454417457,
};

const NEXT_PATH_MATRIX = new THREE.Matrix4().multiplyMatrices(
  new THREE.Matrix4().makeTranslation(0, 0, -8),
  new THREE.Matrix4().makeScale(-1, -1, 1)
);

let root;
let tubes = [];
let cameraTween;

window.onload = function() {
  root = new THREERoot({
    createCameraControls: false,
    antialias: true, //(window.devicePixelRatio === 1),
    fov: 80,
    zNear: 0.001,
    zFar: 2000,
  });
  
  root.renderer.setClearColor(new THREE.Color().setHSL(0, 0, 0.05));
  root.camera.position.set(0, 0.05, 1);
  
  createTubes();
  beginTubesSequence();
};

// METHODS

function createTubes() {
  const matrix = new THREE.Matrix4();
  
  matrix.makeRotationZ(Math.PI * 0.00);
  tubes[0] = createPathMesh(matrix);
  
  matrix.makeRotationZ(Math.PI * 0.66);
  tubes[1] = createPathMesh(matrix);
  
  matrix.makeRotationZ(Math.PI * 1.32);
  tubes[2] = createPathMesh(matrix);
}

function beginTubesSequence() {
  // BLOOM
  
  const strength = 1.25; // 0 - x
  const radius = 1.0; // 0 - 1
  const threshold = 0.5; // 0 - 1
  const bloomPass = new THREE.UnrealBloomPass(
    new THREE.Vector2(window.innerWidth, window.innerHeight),
    strength,
    radius,
    threshold
  );
  const copyPass = new THREE.ShaderPass(THREE.CopyShader);
  
  root.initPostProcessing([
    bloomPass,
    copyPass
  ]);
  
  // LIGHT
  
  let light;
  
  light = new THREE.DirectionalLight(COLORS.red, 1);
  light.position.set(1, 0, 0);
  root.add(light);
  
  light = new THREE.DirectionalLight(COLORS.green, 1);
  light.position.set(-1, 0, 0);
  root.add(light);
  
  light = new THREE.DirectionalLight(COLORS.red, 1);
  light.position.set(0, 0, 1);
  root.add(light);
  
  let cameraPanRange = 1.0, cameraYawRange = cameraPanRange * 1.125;
  
  window.addEventListener('mousemove', (e) => {
    const nx = e.clientX / window.innerWidth * 2 - 1;
    const ny = -e.clientY / window.innerHeight * 2 + 1;
    const ry = -THREE.Math.mapLinear(nx, -1, 1, cameraPanRange * -0.5, cameraPanRange * 0.5);
    const rx = THREE.Math.mapLinear(ny, -1, 1, cameraYawRange * -0.5, cameraYawRange * 0.5);

    TweenMax.to(root.camera.rotation, 1, {
      x: rx,
      y: ry,
      ease: Power2.easeOut,
    });
  });
  
  const tweenCamera = () => {
    cameraTween = TweenMax.to(root.camera.position, SCENE_CONFIG.cameraSpeed, {
      z: `-=${SCENE_CONFIG.pathRadius * 2}`,
      ease: Power0.easeIn,
      onComplete: tweenCamera
    });
  };
  
  tweenCamera();
  
  cameraTween.timeScale(0);
  
  const proxy = {
    rx: 0,
    ry: 0,
    rz: 0,
    cz: 0,
  };
  const camTL = new TimelineMax();
  
  camTL.to(proxy, 4, {rz: 1, cz: 1, ease: Power2.easeIn, onUpdate: () => {
    cameraTween.timeScale(proxy.cz);
  }}, 0);
  
  root.addUpdateCallback(() => {
    root.scene.rotation.z -= proxy.rz * 0.003;
  });
  
  tubes.forEach((tube) => {
    root.add(tube);
    advanceTube(tube);
  });    
}

function advanceTube(tube) {
  const tl = new TimelineMax();
  const firstCompleteTime = tube.geometry.firstCompleteTime * tube.__config.duration;
  
  tl.add(tube.animate(tube.__config.duration, {ease: Power0.easeInOut}));
  tl.add(() => {
    const transformMatrix = new THREE.Matrix4().multiplyMatrices(
      tube.__pathMatrix,
      NEXT_PATH_MATRIX
    );
    
    const nextTube = createPathMesh(transformMatrix, {
      tubeCount: tube.__config.tubeCount,
      tubeArcLength: tube.__config.tubeArcLength,
      tubeStagger: tube.__config.tubeStagger
    });
    
    root.add(nextTube);
    advanceTube(nextTube);
    
  }, firstCompleteTime);
  
  tl.add(() => {
    root.remove(tube);
    tube.geometry.dispose();
    tube.material.dispose();
  });
}

function createPath() {
  let length = 16;
  let path = [];
  let point = new THREE.Vector3();
  
  for (let i = 0; i < length; i++) {
    let angle = i / (length - 1) * Math.PI - Math.PI * 1.5;
    let radius = SCENE_CONFIG.pathRadius;
    
    let scaleX = THREE.Math.mapLinear(i, 0, length - 1, 0.75, 0.25) * THREE.Math.randFloat(0.6, 1.0);
    
    point.x = Math.cos(angle) * radius * scaleX;
    point.z = Math.sin(angle) * radius - radius;
    point.y = (i === 0 || i === length - 1) ? 0 : THREE.Math.randFloatSpread(2) * (i / length);
    // point.y = 0;
    
    let twistOffset = (i === 0 || i === length - 1) ? 0 : THREE.Math.randFloatSpread(2);
    // let twistOffset = 0;
    
    path.push(new THREE.Vector4(point.x, point.y, point.z, twistOffset));
  }
  
  return path;
}

function createPathMesh(matrix, cfg) {
  const config = Object.assign(
    {},
    cfg,
    {
      tubeSegments: 128,
      // tubeCount: THREE.Math.randInt(32, 48),
      // tubeArcLength: THREE.Math.randFloat(0.15, 0.3),
      // tubeStagger: THREE.Math.randFloat(0.0002, 0.002),
      
      tubeCount: 32,
      tubeArcLength: 0.25,
      tubeStagger: 0.001,
      
      tubeRadius: THREE.Math.randFloat(0.005, 0.01),
      twistDistance: THREE.Math.randFloat(0.1, 1.5),
      twistAngle: Math.PI * THREE.Math.randFloat(2, 16),
      duration: SCENE_CONFIG.pathAnimationDuration,
      
      path: createPath()
    }
  );
  
  const mesh = new Tubes(config);
  
  mesh.applyMatrix(matrix);
  mesh.__pathMatrix = matrix.clone();
  mesh.__config = config;
  
  return mesh;
}

// CLASSES

function Tubes(config) {
  const geometry = new TubesGeometry(config);
  
  const material = new THREE.BAS.StandardAnimationMaterial({
    shading: THREE.FlatShading,
    defines: {
      ROBUST: false,
      TUBE_LENGTH_SEGMENTS: config.tubeSegments.toFixed(1),
      PATH_LENGTH: config.path.length,
      PATH_MAX: (config.path.length - 1).toFixed(1)
    },
    
    uniforms: {
      thickness: {value: config.tubeRadius},
      uTwist: {
        value: new THREE.Vector2(
          config.twistDistance,
          config.twistAngle
        )
      },
      time: {value: 0.0},
      uPath: {value: config.path}
    },
    
    uniformValues: {
      diffuse: new THREE.Color(COLORS.yellow),
      roughness: .75,
      metalness: .0
    },
    
    vertexParameters: [
      THREE.BAS.ShaderChunk['catmull_rom_spline'],
      
      `
      attribute vec2 aAngle;
      attribute float aTwistOffset;

      uniform float thickness;
      uniform float time;
      uniform vec2 uTwist;
      uniform vec4 uPath[PATH_LENGTH];
      
      varying float vProgress;
      #ifdef ROBUST
        const float MAX_NUMBER = 1.79769313e+32;
      #endif
      
      vec3 sample(float t) {
      
        float pathProgress = t * PATH_MAX;
      
        ivec4 indices = getCatmullRomSplineIndices(PATH_MAX, pathProgress);

        vec4 p0 = uPath[indices[0]];
        vec4 p1 = uPath[indices[1]];
        vec4 p2 = uPath[indices[2]];
        vec4 p3 = uPath[indices[3]];
        
        float angle = t * uTwist.y;
        float ca = cos(angle);
        float sa = sin(angle);
        vec3 offset = vec3(ca, sa * ca, sa) * aTwistOffset * uTwist.x;
      
        return catmullRomSpline(
          p0.xyz + offset * p0.w,
          p1.xyz + offset * p1.w,
          p2.xyz + offset * p2.w,
          p3.xyz + offset * p3.w,
          fract(pathProgress)
        );
      }
      vec3 getTangent (vec3 a, vec3 b) {
        return normalize(b - a);
      }
      
      void rotateByAxisAngle (inout vec3 normal, vec3 axis, float angle) {
        float halfAngle = angle / 2.0;
        float s = sin(halfAngle);
        vec4 quat = vec4(axis * s, cos(halfAngle));
        normal = normal + 2.0 * cross(quat.xyz, cross(quat.xyz, normal) + quat.w * normal);
      }   
      void createTube (float t, vec2 volume, out vec3 outPosition) {
        vec3 point0 = sample(0.0);
        vec3 point1 = sample(1.0 / TUBE_LENGTH_SEGMENTS);
      
        vec3 lastTangent = getTangent(point0, point1);
        vec3 absTangent = abs(lastTangent);
        vec3 tmpNormal = vec3(1.0, 0.0, 0.0);
        vec3 tmpVec = normalize(cross(lastTangent, tmpNormal));
        vec3 lastNormal = cross(lastTangent, tmpVec);
        vec3 lastBinormal = cross(lastTangent, lastNormal);
        vec3 lastPoint = point0;
        vec3 normal;
        vec3 tangent;
        vec3 binormal;
        vec3 point;
        float maxLen = (TUBE_LENGTH_SEGMENTS - 1.0);
        float epSq = EPSILON * EPSILON;
        for (float i = 1.0; i < TUBE_LENGTH_SEGMENTS; i += 1.0) {
          float u = i / maxLen;
          point = sample(u);
          tangent = getTangent(lastPoint, point);
          normal = lastNormal;
          binormal = lastBinormal;
      
          tmpVec = cross(lastTangent, tangent);
          if ((tmpVec.x * tmpVec.x + tmpVec.y * tmpVec.y + tmpVec.z * tmpVec.z) > epSq) {
            tmpVec = normalize(tmpVec);
            float tangentDot = dot(lastTangent, tangent);
            float theta = acos(clamp(tangentDot, -1.0, 1.0)); // clamp for floating pt errors
            rotateByAxisAngle(normal, tmpVec, theta);
          }
      
          binormal = cross(tangent, normal);
          if (u >= t) break;
      
          lastPoint = point;
          lastTangent = tangent;
          lastNormal = normal;
          lastBinormal = binormal;
        }
        float circX = aAngle.x;
        float circY = aAngle.y;
        vec3 T = tangent;
        vec3 B = binormal;
        vec3 N = -normal;
        outPosition.xyz = point + B * volume.x * circX + N * volume.y * circY;
      }
      #else
      void createTube (float t, vec2 volume, out vec3 offset) {
        // find next sample along curve
        // float nextT = t + (1.0 / TUBE_LENGTH_SEGMENTS) * fract(time * TUBE_LENGTH_SEGMENTS);
        float nextT = t + (1.0 / TUBE_LENGTH_SEGMENTS);
        vec3 current = sample(t);
        vec3 next = sample(nextT);
        vec3 T = normalize(next - current);
        vec3 B = normalize(cross(T, next + current));
        vec3 N = -normalize(cross(B, T));
        float circX = aAngle.x;
        float circY = aAngle.y;
        float a = length(cross(next, current));      
        volume *= 0.5 + a * a * 0.5;
        offset.xyz = current + B * volume.x * circX + N * volume.y * circY;
      }
      #endif      `
    ],
    fragmentParameters: [
      `varying float vProgress;`
    ],
    vertexPosition: [
      `
      float t = position.x;
      
      t = clamp(t + time, 0.0, 1.0);
      
      vec2 volume = vec2(thickness);
      vec3 tTransformed;
      
      createTube(t, volume, tTransformed);
      transformed = tTransformed;
      
      vProgress = t;
      `
    ], 
   fragmentInit: [
      `if (vProgress == 0.0 || vProgress == 1.0) discard;`
    ]
  }); 
  THREE.Mesh.call(this, geometry, material);
  this.frustumCulled = false;
}
Tubes.prototype = Object.create(THREE.Mesh.prototype);
Tubes.prototype.constructor = Tubes;

Object.defineProperty(Tubes.prototype, 'time', {
  get: function () {
    return this.material.uniforms['time'].value;
  },
  set: function (v) {
    this.material.uniforms['time'].value = v;
  }
});
Tubes.prototype.animate = function(duration, options) {
  options = options || {};
  options.time = this.geometry.totalDuration;  
  return TweenMax.fromTo(this, duration, {time: 0.0}, options);
};
function TubesGeometry(config) {
  const radius = 1;
  const length = config.tubeArcLength;
  const sides = 6;
  const segments = config.tubeSegments;
  const openEnded = false;
  const prefab = new THREE.CylinderGeometry(radius, radius, length, sides, segments, openEnded);
  prefab.rotateZ(Math.PI / 2);
  this.tubeLength = length;
  this.tubeStagger = config.tubeStagger;
  THREE.BAS.PrefabBufferGeometry.call(this, prefab, config.tubeCount);
  let aAngle = this.createAttribute('aAngle', 2);
  let tmp = new THREE.Vector2();
  
  for (let i = 0, offset = 0; i < config.tubeCount; i++) {
    for (let j = 0; j < prefab.vertices.length; j++) {
      let v = prefab.vertices[j];
      
      tmp.set(v.y, v.z).normalize();
      
      let angle = Math.atan2(tmp.y, tmp.x);
      
      aAngle.array[offset++] = Math.cos(angle); // angle x
      aAngle.array[offset++] = Math.sin(angle); // angle y
    }
  }
  const offset = 0;

  this.createAttribute('aTwistOffset', 1, (data, i, count) => {
    data[0] = THREE.Math.mapLinear(i, 0, count - 1, -1, 1) + offset;
  });
}
TubesGeometry.prototype = Object.create(THREE.BAS.PrefabBufferGeometry.prototype);
TubesGeometry.prototype.constructor = TubesGeometry;
TubesGeometry.prototype.bufferPositions = function() {
  let positionBuffer = this.createAttribute('position', 3).array;
  
  let matrix = new THREE.Matrix4();
  let p = new THREE.Vector3();
  
  let tubeLength, tubeTimeOffset;
  
  for (let i = 0, offset = 0; i < this.prefabCount; i++) {
    tubeLength = this.tubeLength + i * this.tubeStagger;
    tubeTimeOffset = i * this.tubeStagger;
    
    matrix.identity();
    matrix.makeTranslation(tubeLength * -0.5 - tubeTimeOffset, 0.0, 0.0);

    for (let j = 0; j < this.prefabVertexCount; j++, offset += 3) {
      let prefabVertex = this.prefabGeometry.vertices[j];
      
      p.copy(prefabVertex);
      p.applyMatrix4(matrix);
      
      positionBuffer[offset    ] = p.x;
      positionBuffer[offset + 1] = p.y;
      positionBuffer[offset + 2] = p.z;
    }
  }
  
  // todo simplify this..
  this.totalDuration = 1.0 - (tubeLength * -0.5 - tubeTimeOffset) + this.tubeLength - this.tubeStagger;
  this.firstCompleteTime = 1.0 / this.totalDuration;
};
function LineHelper(points, params) {
  let g = new THREE.Geometry();
  let m = new THREE.LineBasicMaterial(params);  
  g.vertices = points;
  THREE.Line.call(this, g, m);
}
LineHelper.prototype = Object.create(THREE.Line.prototype);
LineHelper.prototype.constructor = LineHelper;

说明

由于JavaScript部分代码过多,不便于全部展示,所以我将所有资源文件已经上传到“资源”中了,可以直接运行,大家可以直接去下载~

  JavaScript知识库 最新文章
ES6的相关知识点
react 函数式组件 & react其他一些总结
Vue基础超详细
前端JS也可以连点成线(Vue中运用 AntVG6)
Vue事件处理的基本使用
Vue后台项目的记录 (一)
前后端分离vue跨域,devServer配置proxy代理
TypeScript
初识vuex
vue项目安装包指令收集
上一篇文章      下一篇文章      查看所有文章
加:2022-01-25 10:29:45  更:2022-01-25 10:29:47 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/9 16:14:17-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码