位置: IT常识 - 正文

分享一个CSS的垂帘效果(css垂直导航栏)

编辑:rootadmin
分享一个CSS的垂帘效果

推荐整理分享分享一个CSS的垂帘效果(css垂直导航栏),希望有所帮助,仅作参考,欢迎阅读内容。

分享一个CSS的垂帘效果(css垂直导航栏)

文章相关热门搜索词:css实现垂直居中的方法,css垂直定位,css垂直导航栏,css div垂直居中的几种方法,css设置垂直位移一半,css垂直定位,css垂直对齐方式怎么设置,css 垂直,内容如对您有帮助,希望把文章链接给更多的朋友!

先上效果图: 再上代码:

<!DOCTYPE html><html lang="en"><head> <meta charset="UTF-8"> <title>Title</title> <style> html, body, canvas { -webkit-user-select: none; -moz-user-select: none; -ms-user-select: none; user-select: none; } html, body { width: 100%; height: 100%; overflow: hidden; margin: 0; display: flex; align-items: center; justify-content: center; background: #191919; } .asset-img { display: none; } </style></head><body><canvas></canvas><img class="asset-img" id="light-img" src="" alt="base64"></body><script> class Mouse { constructor(canvas) { this.pos = new Vector(-1000, -1000) this.radius = 40 canvas.onmousemove = e => this.pos.setXY(e.clientX, e.clientY) canvas.ontouchmove = e => this.pos.setXY(e.touches[0].clientX, e.touches[0].clientY) canvas.ontouchcancel = () => this.pos.setXY(-1000, -1000) canvas.ontouchend = () => this.pos.setXY(-1000, -1000) } } class Dot { constructor(x, y) { this.pos = new Vector(x, y) this.oldPos = new Vector(x, y) this.friction = 0.97 this.gravity = new Vector(0, 0.6) this.mass = 1 this.pinned = false this.lightImg = document.querySelector('#light-img') this.lightSize = 15 } update(mouse) { if (this.pinned) return let vel = Vector.sub(this.pos, this.oldPos) this.oldPos.setXY(this.pos.x, this.pos.y) vel.mult(this.friction) vel.add(this.gravity) let { x: dx, y: dy } = Vector.sub(mouse.pos, this.pos) const dist = Math.sqrt(dx * dx + dy * dy) const direction = new Vector(dx / dist, dy / dist) const force = Math.max((mouse.radius - dist) / mouse.radius, 0) if (force > 0.6) this.pos.setXY(mouse.pos.x, mouse.pos.y) else { this.pos.add(vel) this.pos.add(direction.mult(force)) } } drawLight(ctx) { ctx.drawImage( this.lightImg, this.pos.x - this.lightSize / 2, this.pos.y - this.lightSize / 2, this.lightSize, this.lightSize ) } draw(ctx) { ctx.fillStyle = '#aaa' ctx.fillRect(this.pos.x - this.mass, this.pos.y - this.mass, this.mass * 2, this.mass * 2) } } class Stick { constructor(p1, p2) { this.startPoint = p1 this.endPoint = p2 this.length = this.startPoint.pos.dist(this.endPoint.pos) this.tension = 0.3 } update() { const dx = this.endPoint.pos.x - this.startPoint.pos.x const dy = this.endPoint.pos.y - this.startPoint.pos.y const dist = Math.sqrt(dx * dx + dy * dy) const diff = (dist - this.length) / dist const offsetX = diff * dx * this.tension const offsetY = diff * dy * this.tension const m = this.startPoint.mass + this.endPoint.mass const m1 = this.endPoint.mass / m const m2 = this.startPoint.mass / m if (!this.startPoint.pinned) { this.startPoint.pos.x += offsetX * m1 this.startPoint.pos.y += offsetY * m1 } if (!this.endPoint.pinned) { this.endPoint.pos.x -= offsetX * m2 this.endPoint.pos.y -= offsetY * m2 } } draw(ctx) { ctx.beginPath() ctx.strokeStyle = '#999' ctx.moveTo(this.startPoint.pos.x, this.startPoint.pos.y) ctx.lineTo(this.endPoint.pos.x, this.endPoint.pos.y) ctx.stroke() ctx.closePath() } } class Rope { constructor(config) { this.x = config.x this.y = config.y this.segments = config.segments || 10 this.gap = config.gap || 15 this.color = config.color || 'gray' this.dots = [] this.sticks = [] this.iterations = 10 this.create() } pin(index) { this.dots[index].pinned = true } create() { for (let i = 0; i < this.segments; i++) { this.dots.push(new Dot(this.x, this.y + i * this.gap)) } for (let i = 0; i < this.segments - 1; i++) { this.sticks.push(new Stick(this.dots[i], this.dots[i + 1])) } } update(mouse) { this.dots.forEach(dot => { dot.update(mouse) }) for (let i = 0; i < this.iterations; i++) { this.sticks.forEach(stick => { stick.update() }) } } draw(ctx) { this.dots.forEach(dot => { dot.draw(ctx) }) this.sticks.forEach(stick => { stick.draw(ctx) }) this.dots[this.dots.length - 1].drawLight(ctx) } } class App { static width = innerWidth static height = innerHeight static dpr = devicePixelRatio > 1 ? 2 : 1 static interval = 1000 / 60 constructor() { this.canvas = document.querySelector('canvas') this.ctx = this.canvas.getContext('2d') this.mouse = new Mouse(this.canvas) this.resize() window.addEventListener('resize', this.resize.bind(this)) this.createRopes() } createRopes() { this.ropes = [] const TOTAL = App.width * 0.06 for (let i = 0; i < TOTAL + 1; i++) { const x = randomNumBetween(App.width * 0.3, App.width * 0.7) const y = 0 const gap = randomNumBetween(App.height * 0.05, App.height * 0.08) const segments = 10 const rope = new Rope({ x, y, gap, segments }) rope.pin(0) this.ropes.push(rope) } } resize() { App.width = innerWidth App.height = innerHeight this.canvas.style.width = '100%' this.canvas.style.height = '100%' this.canvas.width = App.width * App.dpr this.canvas.height = App.height * App.dpr this.ctx.scale(App.dpr, App.dpr) this.createRopes() } render() { let now, delta let then = Date.now() const frame = () => { requestAnimationFrame(frame) now = Date.now() delta = now - then if (delta < App.interval) return then = now - (delta % App.interval) this.ctx.clearRect(0, 0, App.width, App.height) // draw here this.ropes.forEach(rope => { rope.update(this.mouse) rope.draw(this.ctx) }) } requestAnimationFrame(frame) } } function randomNumBetween(min, max) { return Math.random() * (max - min) + min } window.addEventListener('load', () => { const app = new App() app.render() }) class Vector { constructor(x, y) { this.x = x || 0 this.y = y || 0 } static add(v1, v2) { return new Vector(v1.x + v2.x, v1.y + v2.y) } static sub(v1, v2) { return new Vector(v1.x - v2.x, v1.y - v2.y) } add(x, y) { if (arguments.length === 1) { this.x += x.x this.y += x.y } else if (arguments.length === 2) { this.x += x this.y += y } return this } sub(x, y) { if (arguments.length === 1) { this.x -= x.x this.y -= x.y } else if (arguments.length === 2) { this.x -= x this.y -= y } return this } mult(v) { if (typeof v === 'number') { this.x *= v this.y *= v } else { this.x *= v.x this.y *= v.y } return this } setXY(x, y) { this.x = x this.y = y return this } dist(v) { const dx = this.x - v.x const dy = this.y - v.y return Math.sqrt(dx * dx + dy * dy) } }</script></html>

代码直接粘贴到html页面就能使用,顺滑的不可言说

本文链接地址:https://www.jiuchutong.com/zhishi/296029.html 转载请保留说明!

上一篇:情感计算——多模态情感识别(情感计算的应用)

下一篇:element - - - - - 你不知道的loading使用方式

  • 分公司需要独立法人吗
  • 纳税人规模证明是什么
  • 文化事业建设费减免政策
  • 贴现手续费计入投资收益
  • 资产负债表固定资产清理怎么填
  • 如何分清进口农产品增值税抵扣率
  • 个人动产租赁所得税税率表最新
  • 自行研发的无形资产会计处理
  • 会计录入凭证怎么录入
  • 怎么核算关联企业的借款费用?
  • 冲销成本结转需要什么凭证
  • 提前预支了应得的工资犯法吗
  • 所得税汇算期间费用填列说明
  • 小规模纳税人第四季度怎么报税
  • 电子税务局报表打印在哪里
  • 弥补以前年度亏损从哪里取数
  • 已确定收入后补缴增值税
  • 一般纳税人简易征收3%可以开专票吗
  • win11玩游戏怎样
  • 电脑搜索功能不准确
  • 物权请求权是什么意思
  • 缴纳的权利许可有哪些
  • 成本核算的意义是什么
  • php生成guid
  • win11dev预览修复工具
  • 损益明细表计提企业所得税
  • hbuilder怎么下载
  • 增值税专票跨月但未认证,怎么作废
  • 冰河湖怎么去
  • 财务费用明细科目怎么写
  • 扣缴个人所得税报告表在哪里打印
  • php实现简单的登录验证
  • 民办非盈利组织税收政策
  • 外汇申报中付款什么意思
  • 可解释深度学习:从感受野到深度学习的三大基本任务:图像分类,语义分割,目标检测,让你真正理解深度学习
  • 创造未来这首歌是谁
  • 一文看懂华为新品发布会
  • ssh-keygen -m pem
  • webserviceclient
  • 固定资产清理属于非流动资产吗
  • 中小企业的资产负债率均值是多少
  • mongodb运行
  • 跨年取得的发票金额大于暂估金额
  • 小微企业认定需要哪些资料
  • 土地使用权被政府收回要交个税吗
  • 年终奖可以税前扣除吗
  • 现金日记账每月都要写期初余额吗
  • 房地产行业扣税标准
  • 只知道主营业务怎么办
  • 公司购买银行理财产品收益交什么税
  • 扶贫小额信贷分贷统还违规吗
  • 银行对账单冲正的单据在财务软件哪里找
  • 专票不报销公司查得到吗
  • 同比增长率环比增长率怎么算
  • 劳务外包账务
  • 扣发当月工资是什么意思
  • 印花税怎么新增税源
  • 数据库语言主要有哪几种
  • friend怎么用
  • ubuntuone
  • Linux JDK,TOMCAT安装及环境设置
  • win10开始菜单样式
  • linux openfoam
  • cocos2dx官方教程
  • 在bootstrap中,有哪几种导航
  • android toast位置
  • android DefaultHttpClient设置setCookieStore
  • 跟踪子弹
  • android推送权限
  • Developing for Android, III: The Rules: Performance
  • android之startActivityForResult的使用
  • unity3d 赛车游戏
  • js操作网页
  • jquery 画图
  • js和jquery可以混用吗
  • 进项税额转出应交税费吗
  • 电子税务局辽宁省
  • 注册税务师考试时间2024
  • 船舶吨税的税目
  • 白酒消费税应纳税额
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设