位置: 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使用方式

  • 苹果11黑屏强制重启(苹果11黑屏强制也开不了没有充电显示了)

    苹果11黑屏强制重启(苹果11黑屏强制也开不了没有充电显示了)

  • 小米画报不更换壁纸(小米画报为什么就那么几张轮换)

    小米画报不更换壁纸(小米画报为什么就那么几张轮换)

  • 华为手机的通话记录在哪里能查到(华为手机的通话录音储存在哪里)

    华为手机的通话记录在哪里能查到(华为手机的通话录音储存在哪里)

  • 苹果13.3系统设备管理在哪(ios13设置)

    苹果13.3系统设备管理在哪(ios13设置)

  • 苹果x屏幕摔出一条光(苹果x摔了屏幕)

    苹果x屏幕摔出一条光(苹果x摔了屏幕)

  • 对方加你微信过期了怎么办(对方加微信过来还没认证不小心删了丛那里找回来)

    对方加你微信过期了怎么办(对方加微信过来还没认证不小心删了丛那里找回来)

  • ipad短信有个感叹号(ipad2020短信有个感叹号)

    ipad短信有个感叹号(ipad2020短信有个感叹号)

  • 抖音音浪比例(抖音音浪是怎么算的抖音收多少钱比例)

    抖音音浪比例(抖音音浪是怎么算的抖音收多少钱比例)

  • 荣耀x10是曲面屏吗(荣耀50换个曲面屏多少钱)

    荣耀x10是曲面屏吗(荣耀50换个曲面屏多少钱)

  • 云备份失败是什么意思(云备份失败是什么意思华为)

    云备份失败是什么意思(云备份失败是什么意思华为)

  • 网关工作在osi哪一层(网关工作在传输层还是应用层)

    网关工作在osi哪一层(网关工作在传输层还是应用层)

  • 在ie8.0的地址栏中应当输入(在ie5.0的地址栏中,应当输入)

    在ie8.0的地址栏中应当输入(在ie5.0的地址栏中,应当输入)

  • 微信提示音怎么调大小(微信提示音怎么设置自定义铃声)

    微信提示音怎么调大小(微信提示音怎么设置自定义铃声)

  • 美拍怎么保存视频(美拍如何保存到相册)

    美拍怎么保存视频(美拍如何保存到相册)

  • 魅族怎么录屏幕视频(魅族手机录屏怎么录制声音)

    魅族怎么录屏幕视频(魅族手机录屏怎么录制声音)

  • 苹果x无面容意味着什么(无面容苹果x怎么解锁)

    苹果x无面容意味着什么(无面容苹果x怎么解锁)

  • oppo手机怎么调出剪切板(oppo手机怎么调返回键出来)

    oppo手机怎么调出剪切板(oppo手机怎么调返回键出来)

  • wps文字查找功能在哪里(wps文字查找功能在哪里手机)

    wps文字查找功能在哪里(wps文字查找功能在哪里手机)

  • 苹果x诊断与用量在哪(iphonexr诊断与用量在哪)

    苹果x诊断与用量在哪(iphonexr诊断与用量在哪)

  • word数字自动编号(word数字自动编号 顺序不对)

    word数字自动编号(word数字自动编号 顺序不对)

  • 智能机器人怎么联网(智能机器人怎么连接不上WiFi)

    智能机器人怎么联网(智能机器人怎么连接不上WiFi)

  • 苹果手机的信号弱的解决方式(苹果手机的信号不好怎么解决)

    苹果手机的信号弱的解决方式(苹果手机的信号不好怎么解决)

  • 电脑开机蓝屏0x0000007f(电脑开机蓝屏0x0000006b的解决方法)

    电脑开机蓝屏0x0000007f(电脑开机蓝屏0x0000006b的解决方法)

  • 京东小程序怎么退货(京东小程序怎么扫码)

    京东小程序怎么退货(京东小程序怎么扫码)

  • PHPCMS 如何取消水印?(php怎么关闭)

    PHPCMS 如何取消水印?(php怎么关闭)

  • 免税收入不征税收入有哪些
  • 高新技术企业产品是什么意思
  • 什么是虚开增票
  • 如何申请成为一名党员
  • 员工交个人所得税对公司有什么影响
  • 清卡重用
  • 工资走公账和私账哪个好
  • 外购货物用于什么不得抵扣进项
  • 技术开发收入免征所得税吗
  • 一般纳税人工会经费可以零申报吗
  • 企业所得税的常设机构
  • 税前可以扣除的有哪些费用
  • 现金收货款要填什么单子
  • 商业承兑汇票风险大吗
  • 发行长期债券会计分录例题
  • 先付一半款财务应该怎么写
  • 期末留抵税额大于销项税额本月还需要计提吗
  • 母公司不经营分公司能报税吗?
  • 工程施工企业人工成本最高多少
  • 社会保险的登记和申报程序
  • 劳务公司差额征税怎么计算
  • 水电费的发票要交税吗
  • 涉农贷款损失
  • 税控机减免会计分录
  • 福利费专票进项转出怎么做账
  • 增加采购发票的方法
  • 补缴城镇土地使用税会计分录怎么做
  • 预付货款样品费怎么入账
  • 其他应付款下的固定资产怎么报废
  • 汇算清缴时所得税费用
  • 溢价发行债券利息
  • 怎么把公司账户的钱转到个人账号
  • edge浏览器设置主页网址
  • 承兑找零怎么做凭证
  • 安装win7前需要手动格式化c盘吗
  • 发票认证了,但是没有入账
  • 应收账款债权融资计划业务操作指引
  • 企业预付工程车租赁费
  • 华沙的教堂
  • 小企业会计准则2023电子版
  • 银行对账单附在记账凭证中吗
  • php加密zend
  • ps快速选择工具抠图后怎么拉出来
  • 小规模纳税人可以开增值税专用发票吗
  • Python怎么转化为中文
  • 织梦专题页模板
  • sql server打开方式
  • 收到汇算清缴的退税需要交税吗
  • 差额增值税发票和全额增值税发票
  • 财务软件是有哪些软件
  • 预计负债计提时调减还是调增
  • 如何计算政府补贴应摊销
  • 人力成本费用率和人工成本利润率
  • 固定资产折旧从什么时候
  • 小规模纳税人建筑服务预缴增值税
  • 建筑业确认收入的条件
  • sqlserver数据库和mysql区别
  • ubuntu系统安装无线网卡驱动
  • win10系统详情
  • window10桌面有白色框
  • hyper-v是啥
  • mac装双系统后无法开机
  • 电脑图片密码是什么意思
  • win10系统应用和浏览器控制需要启用吗
  • linux系统要求配置
  • 开机提示按CTRL+ALT+DEL,无限重启怎么办
  • 怎么改电脑windows7
  • 游戏开发那些事
  • 浅析javascript中function 的 length 属性
  • js函数调用函数
  • jquery选择指定标签
  • shell随机
  • mac怎么编写python
  • js原生dialog
  • python爬虫有道翻译
  • 计征土地增值税时需要以评估价格来确定
  • 江苏优抚对象
  • 发票代码如何查真伪
  • 新版电子发票怎么看发票代码图片
  • 河北税务怎么打不开网页
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

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

    友情链接: 武汉网站建设