為讓女朋友看下雪,這樣做有點意思.

2021-02-23 程式設計師黑叔

女朋友常逛的設計網站這兩天頁面上多了下雪的效果,於是問我我的網站能下雪嗎,作為一個程式設計師我一般會說實現不了,但是作為男朋友,不能說不行。

雪我們可以使用span標籤和css的徑向漸變簡單意思一下:

.snow {
  display: block;
  width: 100px;
  height: 100px;
  background-image: radial-gradient(#fff 0%, rgba(255, 255, 255, 0) 60%);
  border-radius: 50%;
}
複製代碼

效果如下:

很多雪

一片雪是不夠的,成千上萬才浪漫,世界上沒有兩片相同的雪花,所以每片雪都有自己的大小位置速度等屬性,為此先創建一個雪花類:

class Snow {
  constructor (opt = {}) {
    // 元素
    this.el = null
    // 直徑
    this.width = 0
    // 最大直徑
    this.maxWidth = opt.maxWidth || 80
    // 最小直徑
    this.minWidth = opt.minWidth || 2
    // 透明度
    this.opacity = 0
    // 水平位置
    this.x = 0
    // 重置位置
    this.y = 0
    // 速度
    this.speed = 0
    // 最大速度
    this.maxSpeed = opt.maxSpeed || 4
    // 最小速度
    this.minSpeed = opt.minSpeed || 1
    // 瀏覽器窗口尺寸
    this.windowWidth = window.innerWidth
    this.windowHeight = window.innerHeight
    
    this.init()
  }

  // 初始化各種屬性
  init () {
    this.width = Math.floor(Math.random() * this.maxWidth + this.minWidth)
    this.opacity = Math.random()
    this.x = Math.floor(Math.random() * (this.windowWidth - this.width))
    this.y = Math.floor(Math.random() * (this.windowHeight - this.width))
    this.speed = Math.random() * this.maxSpeed + this.minSpeed
  }

  // 設置樣式
  setStyle () {
    this.el.style.cssText = `
      position: fixed;
      left: 0;
      top: 0;
      display: block;
      width: ${this.width}px;
      height: ${this.width}px;
      opacity: ${this.opacity};
      background-image: radial-gradient(#fff 0%, rgba(255, 255, 255, 0) 60%);
      border-radius: 50%;
      z-index: 9999999999999;
      pointer-events: none;
      transform: translate(${this.x}px, ${this.y}px);
    `
  }

  // 渲染
  render () {
    this.el = document.createElement('div')
    this.setStyle()
    document.body.appendChild(this.el)
  }
}
複製代碼

init方法用來生成隨機的初始大小、位置、速度等屬性,在瀏覽器窗口內new100片試試:

let snowList = []
for (let i = 0; i < 100; i++) {
    let snow = new Snow()
    snow.render()
    snowList.push(snow)
}
複製代碼

效果如下:

動起來

雪動起來才能叫下雪,動起來很簡單,不斷改變x和y坐標就可以了,給snow類加個運動的方法:

class snow {
    move () {
        this.x += this.speed
        this.y += this.speed
        this.el.style.left = this.x + 'px'
        this.el.style.top = this.y + 'px'
    }
}
複製代碼

接下來使用requestAnimationFrame不斷刷新:

moveSnow () {
    window.requestAnimationFrame(() => {
        snowList.forEach((item) => {
            item.move()
        })
        moveSnow()
    })
}
複製代碼

效果如下,因為速度是正數,所以整體是往右斜的:

可以看到動起來了,但是出屏幕就不見了,所以雪是會消失的對嗎?要讓雪不停很簡單,檢測雪的位置,如果超出屏幕了就讓它回到頂部,修改一下move方法:

move () {
    this.x += this.speed
    this.y += this.speed
    // 完全離開窗口就調一下初始化方法,另外還需要修改一下init方法,因為重新出現我們是希望它的y坐標為0或者小於0,這樣就不會又憑空出現的感覺,而是從天上下來的
    if (this.x < -this.width || this.x > this.windowWidth || this.y > this.windowHeight) {
      this.init(true)
      this.setStyle()
    }
    this.el.style.left = this.x + 'px'
    this.el.style.top = this.y + 'px'
  }
複製代碼

init (reset) {
    // ...
    this.width = Math.floor(Math.random() * this.maxWidth + this.minWidth)
    this.y = reset ? -this.width : Math.floor(Math.random() * this.windowHeight)
    // ...
  }
複製代碼

這樣就能源源不斷的下雪了:

優化1.水平速度

水平和垂直方向的速度是一樣的,但是看起來有點太斜了,所以調整一下,把水平速度和垂直速度區分開來:

class Snow {
  constructor (opt = {}) {
    // ...
    // 水平速度
    this.sx = 0
    // 垂直速度
    this.sy = 0
  // ...
  }
  
  init (reset) {
    // ...
    this.sy = Math.random() * this.maxSpeed + this.minSpeed
    this.sx = this.sy * Math.random()
  }
  
  move () {
    this.x += this.sx
    this.y += this.sy
    // ...
  }
}
複製代碼

2.左下角沒有雪

因為整體向右傾斜,所以左下角大概率沒有雪,這可以通過讓雪隨機出現在左側來解決:

init (reset) {
  // ...
  this.x = Math.floor(Math.random() * (this.windowWidth - this.width))
  this.y = Math.floor(Math.random() * (this.windowHeight - this.width))
  if (reset && Math.random() > 0.8) {// 讓一小部分的雪初始化在左側
    this.x = -this.width
  } else if (reset) {
    this.y = -this.width
  }
  // ...
}
複製代碼

3.眼前的雪

隨機性的選擇一點雪給它較大的體積、透明度和速度,然後再使用css3的3D透視效果,把它的z軸數值調大一點,這樣的感覺就好像是在眼前划過的一樣:

<body style="perspective: 500;-webkit-perspective: 500"></body>
複製代碼

class Snow {
  constructor (opt = {}) {
    // ...
    // z軸數值
    this.z = 0
    // 快速划過的最大速度
    this.quickMaxSpeed = opt.quickMaxSpeed || 10
    // 快速划過的最小速度
    this.quickMinSpeed = opt.quickMinSpeed || 8
    // 快速划過的寬度
    this.quickWidth = opt.quickWidth || 80
    // 快速划過的透明度
    this.quickOpacity = opt.quickOpacity || 0.2
    // ...
  }
  
  init (reset) {
    let isQuick = Math.random() > 0.8
    this.width = isQuick ? this.quickWidth : Math.floor(Math.random() * this.maxWidth + this.minWidth)
    this.z = isQuick ? Math.random() * 300 + 200 : 0
    this.opacity = isQuick ? this.quickOpacity : Math.random()
    // ...
    this.sy = isQuick ? Math.random() * this.quickMaxSpeed + this.quickMinSpeed : Math.random() * this.maxSpeed + this.minSpeed
    // ...
  }
  
  move () {
    // ...
    this.el.style.transform = `translate3d(${this.x}px, ${this.y}px, ${this.z}px)`
  }
}
複製代碼

4.鵝毛大雪

雪花嘛,輕如鵝毛,鵝毛是怎麼飄的?是不是左右擺動的飄?那我們也可以選擇一部分的雪花讓它跟鵝毛一樣飄,左右搖擺很簡單,速度一會加一會減就可以了:

class Snow {
  constructor (opt = {}) {
    // ...
    // 是否左右搖擺
    this.isSwing = false
    // 左右搖擺的步長
    this.stepSx = 0.03
    // ...
  }

  // 隨機初始化屬性
  init (reset) {
    // ...
    this.isSwing = Math.random() > 0.8
    // ...
  }

  move () {
    if (this.isSwing) {
      if (this.sx >= 1 || this.sx <= -1) {
        this.stepSx = -this.stepSx
      }
      this.sx += this.stepSx
    }
    // ...
  }
}
複製代碼

除了上述這種方法,左右搖擺還有一種方式,就是使用正弦或餘弦函數,因為它們的曲線翻轉90度就是左右搖擺:

img

我們使用正弦函數,公式為:y=sin(x),x的值是弧度表示,只要一直增加就可以了,y的值用來修改雪花的水平方向的速度變化步長:

class Snow {
  constructor (opt = {}) {
    // ...
    // 是否左右搖擺
    this.isSwing = false
    // 左右搖擺的正弦函數x變量
    this.swingRadian = 0
    // 左右搖擺的正弦x步長
    this.swingStep = 0.01
    // ...
  }

  init (reset) {
    // ...
    this.swingStep = 0.01 * Math.random()
  }

  move () {
    if (this.isSwing) {
      this.swingRadian += this.swingStep
      this.x += this.sx * Math.sin(this.swingRadian * Math.PI) * 0.2
    } else {
      this.x += this.sx
    }
    // ...
  }
}
複製代碼

因為正弦函數y的值是從1變化到-1,擺動幅度太了,所以乘了個小數0.2縮小一點,想要幅度小一點,還有一個方法是不要使用整個正弦曲線,可以從中截取一個適合的區間大小,比如就讓x的值在0.9π到1.1π之前變化:

class Snow {
  constructor (opt = {}) {
    // ...
    // 是否左右搖擺
    this.isSwing = false
    // 左右搖擺的正弦函數x變量
    this.swingRadian = 1// 需要改成一個中間值
    // 左右搖擺的正弦x步長
    this.swingStep = 0.01
    // ...
  }

  init (reset) {
    // ...
    this.swingStep = 0.01 * Math.random()
    this.swingRadian = Math.random() * (1.1 - 0.9) + 0.9// 也讓它隨機一下
  }

  move () {
    if (this.isSwing) {
      if (this.swingRadian > 1.1 || this.swingRadian < 0.9) {
        this.swingStep = -this.swingStep
      }
      this.swingRadian += this.swingStep
      this.x += this.sx * Math.sin(this.swingRadian * Math.PI)
    } else {
      this.x += this.sx
    }
    // ...
  }
}
複製代碼

5.下的慢一點

既然給水平加了曲線,垂直方向上是不是也可以改成非勻速呢?當然可以,區別是速度得一直是正的,不然就要出現反自然現象了,改變速度曲線同樣可以使用正餘弦,上面我們使用了0.9π到1.1π之間的正弦曲線,根據上圖可以發現對應的餘弦曲線都是負的,趨勢是先慢後快,所以可以利用這一段來改變垂直方向的速度:

move () {
  if (this.isSwing) {
    if (this.swingRadian > 1.1 || this.swingRadian < 0.9) {
      this.swingStep = -this.swingStep
    }
    this.swingRadian += this.swingStep
    this.x += this.sx * Math.sin(this.swingRadian * Math.PI)
    this.y -= this.sy * Math.cos(this.swingRadian * Math.PI)// 因為速度都是負的,所以改成-
  } else {
    this.x += this.sx
    this.y += this.sy
  }
  // ...
}
複製代碼

6.在最上面

為了防止為頁面上原本層級更高的元素遮擋,給雪花的樣式加一個很大的層級:

render () {
    this.el = document.createElement('div')
    this.el.style.cssText = `
        // ...
        z-index: 9999999999999;
    `
    document.body.appendChild(this.el)
}
複製代碼

7.看不見我

修改了層級,所以雪花會在頁面的最上層,那麼可能會擋住其他元素的滑鼠事件,需要禁止它響應滑鼠事件:

render () {
    this.el = document.createElement('div')
    this.el.style.cssText = `
      // ...
      pointer-events: none;
    `
    document.body.appendChild(this.el)
  }
複製代碼

8.更好一點

使用性能更好的transform屬性來做動畫:

render () {
    this.el = document.createElement('div')
    this.el.style.cssText = `
        left: 0;
        top: 0;
        transform: translate(${this.x}px, ${this.y}px);
    `
    document.body.appendChild(this.el)
}
複製代碼

move () {
    // ...
    // this.el.style.left = this.x + 'px'
    // this.el.style.top = this.y + 'px'
    this.el.style.transform = `translate(${this.x}px, ${this.y}px)`
}
複製代碼

當然,最好的方式是用canvas來畫。

最終效果:

下雨&雨夾雪

下完雪,接下來順便下個雨,雨和雪差不多,都是從天上掉下來,但是雨的速度更快,通常也不會左右搖擺什麼的,方向也基本是一致的,先來修改一下樣式:

setStyle () {
  this.el.style.cssText = `
    // ...
    width: 1px;
    // ...
  `
}
複製代碼

很簡單,只要把寬度寫死為1就行了:

接下來把搖擺去掉:

move () {
  this.x += this.sx
  this.y += this.sy
  // ...
}
複製代碼

效果如下:

可以發現雨是豎著在水平移動,顯然是不行的,需要讓它傾斜一定的角度,和運動方向保持一致,這個也很簡單,算一下斜率,水平速度除以垂直速度:

move () {
  // ...
  this.el.style.transform = `translate(${this.x}px, ${this.y}px) ${this.getRotate(this.sy, this.sx)}`
}
getRotate(sy, sx) {
  return `rotate(${sx === 0 ? 0 : (90 + Math.atan(sy / sx) * (180 / Math.PI))}deg)`
}
複製代碼

因為tan(θ)=sy/sx,θ=Math.atan(sy / sx),因為雨的線段默認是從上到下垂直的,θ是代表和水平方向上的夾角,所以需要先旋轉90度,再旋轉夾角的度數,最後弧度轉角度的公式為:角度=弧度*(180/π)。

雨和雪都實現了,讓它們一起出來,就是雨夾雪了:

根據天氣下雪

把上面的代碼放到網站上就有下雪的效果了,另外也可以使用天氣廠商的api,根據實時天氣來下雪或者下雨,再實現一下太陽、烏雲等效果,一個沉浸式天氣就完成了,有興趣的可自行實踐。

完整代碼

https://github.com/wanglin2/snow

相關焦點

  • 【實戰解析】女朋友非常喜歡別人網頁上的下雪效果,我立馬打開電腦擼代碼…
    女朋友常逛的設計網站這兩天頁面上多了下雪的效果,於是問我我的網站能下雪嗎,作為一個程式設計師我一般會說實現不了,但是作為男朋友,不能說不行。雪雪我們可以使用span標籤和css的徑向漸變簡單意思一下:.snow {  display: block;  width: 100px;  height: 100px;  background-image
  • 安溪下雪原來是這樣的!美爆了!!!
    前兩天各種圈都在瘋傳安溪下雪了而2016年安溪的第一場雪也在今天如期而至,瞬間刷爆了安溪人的朋友圈:看來安溪人都很期待雪啊,畢竟因為稀有才珍貴啊,心碎一地。。。5毛錢特效一次性讓你看個夠!▼不信你看媽蛋,看完前面幾幅圖我居然真的以為安溪的雪下的介麼大了……其實,安溪下雪還是有的,只是不大而已雪後的安溪,你喜歡嗎?
  • 冬天到了,女朋友想讓我的網站也下雪,我立馬打開電腦擼代碼…
    雪我們可以使用span標籤和css的徑向漸變簡單意思一下:
  • 下次下雪的時候,我們一起吃飯吧!
    每個季節有著不同的期待,去上班的路上,看到第一朵花開了,不緊不慢地拿出手機拍一張;上課的時候,看到外面下雪了,讓學生趴在窗前驚叫一聲:「哇—」。這些是時間帶給大自然的禮物,順帶也捎給了我們一份。碰到下雪天就格外激動是南方人的條件反射,哪怕是在應該睡懶覺的日子,天氣預報裡的雪花標誌也是最好的鬧鐘。
  • 這樣玩水族箱?有點意思!
    最近閒來無事,小編網羅了一些奇葩缸,16個「世界之最」,看完記得擦口水,話說不多,看圖體驗吧!
  • joe問jack:最近有沒有交女朋友呀 | 我們仨
    而他要修滿學分,估計還得多加一個學期,也就是說,他這個大學本科要讀5年半,出來工作都快25歲了,想想是有點吃虧。開學後他沒怎麼主動跟我們聯繫,有時問他在幹嘛,他也很久才回那麼一句兩句。那天,joe有點憋不住,主動去撩他——
  • 組局朋友變炮友,如何上位女朋友.
    本想十點多怎麼也回來了,結果那個地兒有點偏,打不到車,十一點門禁。他們大概玩通宵,一起玩耍好像也不是太壞。凌晨兩點多,過生日的小夥伴們卻準備撤了,尷尬了就。他舍友們都在,就說不然會住的地方鬥地主好了…恩…他們在外面租了房子。
  • 雲南多地下雪!文山天氣是?今年冬天會很冷嗎…
    文山天氣:立冬了,冷一哈意思意思!不過雲南部分地方就冷得有點認真了好多地方都下起了雪一起來感受一波濃濃的冬意↓↓昆明轎子雪山轎子雪山景區工作人員介紹,今天上午雪山上氣溫只有7度左右,清晨更是冷。進入冬天,轎子山每天早上可以看到壯觀的霧松。
  • 搞笑段子:「路上撿到的我有點怕,這是什麼意思啊?」哈哈哈不要給我...
    其實很多人在外面都撿到過東西,有的很正常,但是有的就很奇葩了,你們有沒有看到過什麼奇葩的東西,在民間有很多的習俗,做什麼事情首先要封個紅包,以此來表達事情做得順順利利的,不同的地方有不同的風俗習慣,也代表著不同的意思...話不多說來進入今日份快樂源泉!01「路上撿到的我有點怕,這是什麼意思啊?」哈哈哈不要給我...
  • 幽默 | 找女朋友的方法
    竺吉捷郭小明有天在家上網的時候看到一個網站閃著「視頻直播」的廣告,在好奇心驅使下點了進去,結果不看不知道,一看便一發不可收拾。他在家還嫌看得不夠,上班只要有點空閒就會塞著耳機偷偷地看。有天郭小明看得太入迷,就連老闆站在身後都沒察覺。
  • 沙漠下雪,驚呆駱駝
    但是,世界就是如此奇妙~沙漠下雪這樣的奇景也許是幾十年難以一遇,但它確確實實的發生了~ 近日,內蒙古包頭市一群戶外愛好者在庫布其沙漠進行越野車穿越沙漠時,偶遇了沙漠下雪的美景,這群經常組織參加沙漠穿越活動的戶外愛好者也是第一次見到沙漠下雪的奇景。
  • 【驚險】為看女朋友,20多歲小夥爬上10多米高的廣告架
    「你看,他還睡在上面呢,翻個身掉下來不就沒命了啊。」李女士一邊向民警說,一邊用手指向廣告架。順著李女士手指的方向望去,民警一眼就看見了有名男子橫睡在廣告架的外沿臺上,廣告架是搭建在小區大門的平臺上方,離地面距離至少十米,外沿臺是供拆裝廣告人員使用的,很窄,男子稍有不慎極有可能掉落。
  • 臺灣奇景,從不下雪的地區都下雪了~
    六福村樂園下雪了楊梅也下雪了
  • 【天天爆圖】這樣的女朋友,你敢和她結婚嗎?
    1央視再批畸形吃播:為博流量胡吃海塞,自傷又浪費!
  • 金寨下雪了,我又想你了……
    金寨論壇官方網站:www.lajzx.com  請關注金寨下雪了一下雪,就想你了
  • 田字格:放學時外面下雪了
    「放學了,外面下雪了。」
  • 有點意思…
    一號女主意嵐一臉興奮地來到森林中的大房子來當小助理,在森林中拍花拍鳥迷路了的意嵐,遇上了一臉陰鬱的房子主人二號寶嵐…看這樣子,要麼是被寵壞的小孩,要麼就是有什麼心理陰影…寶嵐的防備之心在意嵐體貼入微的關懷下漸漸瓦解…兩人開始有了一些
  • 誰讓你把雪人堆成這樣的....腦洞別這麼大,好麼?
    妹子趕緊看看天氣預報,果然這是要下雪的節奏啊!如果下雪了,你最想做的一件事是什麼呢?是窩在被窩裡看韓劇,外加一份炸雞啤酒↓走你!來來來一起做運動就不冷了了!看這銷魂的舞步!或者這樣的↓因為南方小夥伴們堆出來的雪人是這樣的↓
  • 2018年的第一場雪,下雪天聽薛之謙的歌更配哦
    首先,從今天起,開始一個人的獨居生活……因為他們退休之後終於可以好好嗨了,開啟了韓國五日遊……徒留本人對著家裡冰箱裡的菜望菜興嘆……盯著它看能把菜變熟就好了……然後,今早睜眼,發現外面鋪了薄薄的一層雪很興奮哎,8號那天,我還在感嘆
  • 宅男到底容不容易找女朋友,看日本網友是怎麼說的
    看過的人就知道這部動畫內容和標題不符,這表現的宅男戀愛也太容易了,女朋友也太好找了。所以整天愛投票的日本網友,就開始進行了一項名叫宅男找女朋友到底難不難的投票,共得到了574個御宅族的回覆。說法:1.看多了動畫裡的美好戀愛,但現實截然相反。完全不明白對方(非御宅族)的感受。