如何做活動頁面的滾動動畫?讓用戶體驗MAX的demo在這裡!

来源:https://www.cnblogs.com/qcloud1001/archive/2018/12/21/10157337.html
-Advertisement-
Play Games

本文由雲+社區發表 最近的一個活動頁面需要做一個可以左右滑動的抽簽效果,故通過用css的transform屬性和js結合來模擬可以無限滾動的效果。 先上效果: demo地址: 實現過程 1. 結構與樣式 結構:卡片分前後兩排,每列插入10個div結點,以便做左右位移效果。 樣式:設置每一列都恰好好在 ...


本文由雲+社區發表

最近的一個活動頁面需要做一個可以左右滑動的抽簽效果,故通過用css的transform屬性和js結合來模擬可以無限滾動的效果。

先上效果:

img

demo地址:https://kiroroyoyo.github.io/cardTransform/index.html

實現過程

1. 結構與樣式

結構:卡片分前後兩排,每列插入10個div結點,以便做左右位移效果。

樣式:設置每一列都恰好好在中間位置(或中間位置附近),如下所示。

a. 前排(cardFrond)相對於視口的初始位置(left:-255.5%;):

img

b. 後排(backFrond)相對於視口的初始位置(left:-228.3%;):

img

2. 無限滾動原理

由於這裡的停止位置是固定的,前排永遠是當前卡片相對於視口居中,後排永遠是兩個卡片相對於視口居中,且每個卡片是一樣的,所以當卡片列表向前或向右移動到一個目標位置時,都將列表重置為初始位置繼續滾動。如下圖以前排卡片為例:

img

所以當滾動停止後會統一將列表樣式設置為transform: translateX(0)。而對於用戶這一操作是無感知的,認為已經滑動到了新的位置。

3.滑動過程實現

a. 目標位移與幀位移

為了做出滑動後到停留位置的緩動效果,所以當用戶左右滑動屏幕時,會記錄滑動距離,計算出卡片該到的目標位移位置,目標位移位置是有規則的,因為這裡有10張卡片均分寬度,位置必須是(100%/10)的整數倍,例如-40%、-30%、……40%,這樣才能保證目標位置與初始位置相重合。

目標位移代碼片段

onDocumentMouseUp : function(e){
    //如果是點擊事件 不設置移動
    if (!this.fingerTouch)
      return;
    this.moveDirect = this.lon > 0 ? 1 : -1;
    this.transNum = this.lon/10 + this.moveDirect;
    this.lon = Math.round(this.transNum) * 10;
    this.fingerTouch = false;   
}

記錄了目標位移後,每一幀會以一定的幀位移不斷靠近目標位移,使其在手指離開屏幕時仍有慢慢滑動到目標位置的緩動效果。此時需要判斷當前位置是否大於40%或者小於-40%,若超過這個極限值需要重設目標位移及幀位移,使其在極限值內。

animate: function(){
    this.prePos += (this.lon - this.prePos) * 0.1;
    if (this.prePos > 40) {
      this.lon = this.lon - 40;
      this.prePos = this.prePos - 40;
    }else if (this.prePos < -40) {
      this.lon = this.lon + 40;
      this.prePos = this.prePos + 40;
    }
    //判斷是否到達了目標位置
    if (Math.abs(this.prePos - this.lon) < 0.01 && Math.abs(this.lon) > 0.01 && (!this.fingerTouch))
    {
        this.ani_move = false;
        this.prePos = 0;
        this.frondCard.style = "transform: translateX("+ this.prePos +"%)";
        this.backCard.style = "transform: translateX("+ this.prePos +"%)";
    }else{
        this.frondCard.style = "transform: translateX("+ this.prePos +"%)";
        this.backCard.style = "transform: translateX("+ (-this.prePos) +"%)";
        requestAnimationFrame(this.animate.bind(this));
    }
  },

b. 連續滑動判斷

當在上次滑動動畫還未播放結束時用戶又進行了第二次滑動時,需要執行一下操作:

​ 1). 判斷滑動時機處於上次滑動手指已離開屏幕但動畫還未結束,此時需要記錄兩個flag,一個是ani_move,記錄動畫是否仍在進行,fingerTouch記錄手指是否停留屏幕。

​ 2). 判斷第二次滑動是否與第一次不同方向,若不同向需重置上次幀位移為0。以免上次幀位移太大影響移動方向。

1)與2)代碼片段:

if( this.ani_move && this.fingerTouch == false) {
    // 判斷是否不同向
    if (((e.clientX - prex) > 0 ? 1: -1) == -this.moveDirect ) {
        this.lon = 0;
        this.prePos = 0;
        this.moveDirect = -this.moveDirect;
    }
}

3). 取消第二次滑動時的動畫播放和位移重置

// 若是上次動畫未結束不需要再次啟動動畫和重置目標位移
if( this.ani_move && this.fingerTouch == false) {
}
else {
    this.lon = 0;
     cardAnimate.animate();
}

寫在最後

目前這個滑動效果只能針對卡片相同,停留位置固定的情況,因為需要做到位置重合。使用css transform來做無限滾動的效果,可以避免改變dom結點帶來的頁面重新佈局。

下圖是chrome cpu6倍減速調試效果,沒有觸發layout,FPS基本維持在60左右。

img

代碼地址:

https://github.com/kiroroyoyo/cardTransform

此文已由作者授權騰訊雲+社區發佈



您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Editor.md是一款優秀的開源Markdown 編輯器,在使用中遇到的一些問題和功能改進分享給需要的伙伴。 ...
  • 一、組件渲染 當組件的props或者state發生改變時,組件會自動調用render方法重新渲染。當父組件被重新渲染時,子組件也會被遞歸渲染。那麼組件是如何渲染的呢? 二、組件的生命周期 組件的聲明周期可分成三個狀態:Mounting,已插入真實 DOM;Updating,正在被重新渲染;Unmou ...
  • <meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1.0"/> ...
  • redux數據流向 基本使用 " " ...
  • 學習時,為了搜集最全的中文資料,有時候不得不使用Baidu搜索引擎。在你還是個小菜雞的時候你可能會花費大量時間在百度上! 但是,時間久了你會發現,你總會被網路上一些奇奇怪怪或者有趣的事情吸引過去而逐漸忘記自己曾經打開百度是要乾什麼?時間就這樣被一些無關緊要的有趣的事情給浪費了。 對於廣告吧,還能使用 ...
  • 源地址:https://jingyan.baidu.com/article/546ae185fa4f721149f28cbf.htm 文件:index.htm 文件:calendar.js ...
  • 要解決跨域的問題,我們可以使用以下幾種方法: 1、通過jsonp跨域 2、通過修改document.domain來跨子域 3、使用window.name來進行跨域 4、使用HTML5中新引進的window.postMessage方法來跨域傳送數據 使用postMessage來跨域傳送數據還是比較直觀 ...
  • [TOC] 一. Decorator裝飾器 修飾器是 加入的新特性, 中進行了大量使用,有很多內置的修飾器,後端的同學一般稱之為 “註解” 。修飾器的作用,實際上就是設計模式中常說的 裝飾者模式 的一種實現,早在 開始,設計模式原生化就已經是非常明顯的趨勢了,無論是 和`Iterator Proxy ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...