您好,登錄后才能下訂單哦!
這篇文章主要講解了“javascript如何操作單個dom元素添加動畫”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“javascript如何操作單個dom元素添加動畫”吧!
讓一個元素從左至右進行運動
<p id="box"></p>
var box = document.getElementById("box"); var t = null; t = setInterval(function(){ })
t = setInterval(function(){終止條件})
// 元素的屬性值 === 目標點 if(dom.attr === target){ clearInterval(t); }
起始點
一個運動的起始點其實就是當前元素的位置,我們通過API獲取當前元素的位置,讓這個位置作為運動的起始。
目標
速度
讓元素通過定時器在很短的間隔內進行CSS屬性值的改變
這樣連續的運動在用戶看來就是動畫效果
封裝的主要作用就是讓元素可以在短時間間隔內不斷改變屬性實現動畫效果
單屬性運動框架:
<button id="btn">開始運動</button><p id="box"></p><p id="line"></p><script> var box = document.getElementById("box"); var btn = document.getElementById("btn"); var target = 500; // 速度可以根據 起始點和目標點進行判斷,從而決定正負; var speed = 17; // - 方向; btn.onclick = function(){ // 1. 獲取元素初始位置; var _left = box.offsetLeft; speed = target - _left >= 0 ? speed : -speed ; var interval = setInterval( function(){ // 4. 判定運動的終止條件; if(Math.abs(target - _left) <= Math.abs(speed) ){ // 因為終止時有可能沒有到達目標點,因此我們把元素賦值到目標點位置; box.style.left = target + "px"; clearInterval( interval ); }else{ // 2. 元素根據初始位置進行改變; _left += speed; // 3. DOM操作,根據已有數據讓元素屬性發生改變; box.style.left = _left + "px"; } } , 30) } </script>
勻速運動封裝
<script>// - 提取屬性名作為參數; btn.onclick = function(){ animate( "left", 500 )}function animate( attr , target , speed = 10 ){ // 1. 獲取元素初始樣式 var _style = getComputedStyle( box ); // 2. 根據屬性要求取出當前的屬性的屬性值; var _css_style = parseInt(_style[attr]); speed = target - _css_style >= 0 ? speed : -speed ; var interval = setInterval( function(){ // 4. 判定運動的終止條件; if(Math.abs(target - _css_style) <= Math.abs(speed) ){ // 因為終止時有可能沒有到達目標點,因此我們把元素賦值到目標點位置; box.style[attr] = target + "px"; clearInterval( interval ); }else{ // 2. 元素根據初始位置進行改變; _css_style += speed; // 3. DOM操作,根據已有數據讓元素屬性發生改變; box.style[attr] = _css_style + "px"; } } , 30)}</script>
兼容透明度
<script>function animate( dom , attr , target , speed = 10 ){ // 1. 獲取元素初始樣式 var _style = getComputedStyle( dom ); // 2. 根據屬性要求取出當前的屬性的屬性值; if( attr === "opacity"){ var _css_style = parseInt(_style[attr] * 100 ); target *= 100; }else{ var _css_style = parseInt(_style[attr]); } speed = target - _css_style >= 0 ? speed : -speed ; var interval = setInterval( function(){ // 4. 判定運動的終止條件; if(Math.abs(target - _css_style) <= Math.abs(speed) ){ // 因為終止時有可能沒有到達目標點,因此我們把元素賦值到目標點位置; if( attr === "opacity"){ dom.style[attr] = target / 100; }else{ dom.style[attr] = target + "px"; } clearInterval( interval ); }else{ // 2. 元素根據初始位置進行改變; _css_style += speed; // 3. DOM操作,根據已有數據讓元素屬性發生改變; if( attr === "opacity"){ dom.style[attr] = _css_style / 100 ; }else{ dom.style[attr] = _css_style + "px"; } } } , 30)}</script>
緩沖運動
緩沖運動是一種運動方式
這種運動方式是速度在運動過程中會有改變的運動
距離越小 速度越小
<button id="btn">開始運動</button><p id="box"></p><script>var box = document.getElementById("box");var btn = document.getElementById("btn");btn.onclick = function(){ animate( box , "left" , 500 )}function animate( dom , attr , target , transition = "buffer", speed = 10 ){ var _style = getComputedStyle( dom ); if( attr === "opacity"){ var _css_style = parseInt(_style[attr] * 100 ); target *= 100; }else{ var _css_style = parseInt(_style[attr]); } if( transition === "liner"){ speed = target - _css_style >= 0 ? speed : -speed ; } var interval = setInterval( function(){ if( transition === "buffer"){ // 計算速度; speed = (target - _css_style) / 10; //速度不取整在小數部分會做很多無意義的計算; speed = speed > 0 ? Math.ceil(speed) :Math.floor( speed ) } if(Math.abs(target - _css_style) <= Math.abs(speed) ){ // 因為終止時有可能沒有到達目標點,因此我們把元素賦值到目標點位置; if( attr === "opacity"){ dom.style[attr] = target / 100; }else{ dom.style[attr] = target + "px"; } clearInterval( interval ); }else{ // 2. 元素根據初始位置進行改變; _css_style += speed; // 3. DOM操作,根據已有數據讓元素屬性發生改變; if( attr === "opacity"){ dom.style[attr] = _css_style / 100 ; }else{ dom.style[attr] = _css_style + "px"; } } } , 30)} </script>//只需要改變里面transition的值就可以調整運動模式//buffer為緩沖運動//liner為勻速運動
多屬性運動框架
我們在多次調用animate的時候會開啟多個定時器
因為定時器之中的數據都一樣,我們看不出在效果上的差異
但是多次開啟定時器會極其嚴重的消耗計算機性能
開啟當前定時器之前關閉上一個定時器
<script>// 在一個定時器之中,用for循環同時執行多次dom樣式操作; // 1. 需要優化的部分:參數,要把樣式部分的參數優化成一個對象; function animate( dom , attrs , transition = "buffer", speed = 10 ){ var _style = getComputedStyle( dom ); // 獲取元素當前的屬性 : for(var attr in attrs ){ // attr ? 要過渡的css屬性名; // attrs[attr] ? 要過渡的當前屬性; attrs[attr] = { target : attrs[attr], // 元素當前的屬性放入到這個對象之中; now : parseInt(_style[attr]) } } // 因為直接關閉interval是沒有作用的,此時的inteval是一個局部變量,每次animate被調用的時候都會直接重置; // 我們應該吧定時器的id放在當前正在執行過渡效果的dom對象上; clearInterval(dom.interval); dom.interval = setInterval( function(){ for(var attr in attrs){ // 取出 attrs 之中的目標點和當前值; speed = (attrs[attr].target - attrs[attr].now) / 10 ; // 根據速度正負,進行速度取整; speed = speed > 0 ? Math.ceil( speed ) : Math.floor( speed ); // 判定終止條件; if( attrs[attr].target === attrs[attr].now){ // 刪除已經到達目標點的屬性; delete attrs[attr] // 判定attrs里面已經沒有屬性了; for(var a in attrs){ return false; } clearInterval(dom.interval); }else{ attrs[attr].now += speed; dom.style[ attr ] = attrs[attr].now + "px"; } } } , 30)}// 優化參數之后,key值是等待運動的css屬性,value值是元素的目標; btn.onclick = function(){ animate( box , { width : 500 , height : 400 } )}</script>
<style> *{ margin: 0; padding: 0; } .container{ width: 1130px; height: 286px; margin: 0 auto; position: relative; overflow: hidden; } .wrapper{ width: 6780px; position: absolute; left: 0; } .slide{ float: left; } .slide , .slide img{ width: 1130px; height: 286px; } .button-prev{ left: 0; background-position: 30px center; background-image: url(https://static.zcool.cn/git_z/z/widget/slider/images/svg/left_arrow.svg?v=2); top: 0; width: 13px; height: 100%; padding: 0 80px; border-radius: 2px; position: absolute; background-repeat: no-repeat; } .button-prev:hover{ background-image: url(https://static.zcool.cn/git_z/z/widget/slider/images/svg/left_arrowhover.svg?v=2); } .button-next{ right: 0; background-position: 91px center; background-image: url(https://static.zcool.cn/git_z/z/widget/slider/images/svg/right_arrow.svg?v=2); top: 0; width: 13px; height: 100%; padding: 0 80px; border-radius: 2px; position: absolute; background-repeat: no-repeat; } .button-next:hover{ background-image: url(https://static.zcool.cn/git_z/z/widget/slider/images/svg/right_arrowhover.svg?v=2); } .pagination{ position: absolute; bottom: 10%; width: 100%; height: 10px; left: 30px; } .pagination span{ display: inline-block; width: 10px; height: 10px; margin-left: 10px; border-radius: 50%; background-color: cornflowerblue; border: 2px solid transparent; background-clip: content-box; } .pagination span.active{ border: 2px solid skyblue; box-shadow: 0 0 5px skyblue; background-color: #fff; } </style>
<!-- 類名請使用和我一樣的類名 --><p class="container"> <p class="wrapper"> <!-- 第0張圖片 --> <p class="slide"> <img src="https://img.zcool.cn/ad_manager/location/f35d611484931101c43350bbdbd5.jpg" alt=""> </p> <p class="slide"> <img src="https://img.zcool.cn/ad_manager/location/3dcb6113a3471101c433505bbd72.jpg" alt=""> </p> <p class="slide"> <img src="https://img.zcool.cn/ad_manager/location/65f36113a3341101c4335014b174.jpg" alt=""> </p> <p class="slide"> <img src="https://img.zcool.cn/ad_manager/location/7f97611481181101c43350225b33.jpg" alt=""> </p> <p class="slide"> <img src="https://img.zcool.cn/ad_manager/location/afcf6114801e1101c433507f9e28.jpg" alt=""> </p> <!-- 最后一張圖片 --> <!-- 把第0張圖片放在整個圖片結構的最后 --> <p class="slide"> <img src="https://img.zcool.cn/ad_manager/location/f35d611484931101c43350bbdbd5.jpg" alt=""> </p> </p> <p class="button-next"></p> <p class="button-prev"></p> <p class="pagination"> <span class="active"></span> <span></span> <span></span> <span></span> <span></span> </p></p><script src="./utils.js"></script><script> // 輪播圖的核心就是左右切換按鈕,實現顯示元素下標的改變; var index = 0 ; var prev = 0; var next_btn = document.querySelector(".button-next"); var prev_btn = document.querySelector(".button-prev"); var slides = document.querySelectorAll(".slide"); var wrapper = document.querySelector(".wrapper"); // 自動播放的阻止功能是在鼠標移入container容器之中就觸發的; var container = document.querySelector(".container"); var bullets = document.querySelectorAll(".pagination span"); // 綁定事件 // - 輪播圖改變下標功能必須設置邊界; function bindEvent(){ next_btn.onclick = function(){ add(); bannerAnimate(); } prev_btn.onclick = function(){ reduce() bannerAnimate(); } container.onmouseover = function(){ stop(); } container.onmouseout = function(){ autoPlay(); } bullets.forEach( function( ele , i ){ ele.onmouseover = function(){ prev = index; // 防止穿幫邏輯; // - 如果在假的第0張圖片上(在最后一張圖片上) // - 我們先讓真假圖片呼喊然后在進行元素的動畫效果; if( index === 5 ){ wrapper.style.left = 0; } index = i; bannerAnimate(); } }) } // 下標增加; function add(){ prev = index; if( index === slides.length - 1 ){ // 這個邏輯會在最后一張圖片進行切換時進入; // 我們讓wrapper直接位移到開頭,改變元素位置的同時讓用戶無法感知; wrapper.style.left = 0; // 我們需要從第0個圖片切換到第一個圖片; // -因為我們最后一張圖片的顯示和開頭圖片的顯示是一樣的 index = 1; }else{ index ++; } } // 下標減少; function reduce(){ prev = index; if( index === 0 ){ wrapper.style.left = -(slides.length - 1) * 1130 + "px"; index = slides.length - 2; }else{ index --; } } // 根據我們的算法去實現動畫效果; function bannerAnimate(){ animate( wrapper , { left : - index * 1130 }); // 給對應的分頁器按鈕添加active; // 先去清空所有的類名; bullets.forEach( function( ele ){ ele.classList.remove("active") }) // 下標需要進行特殊處理,在顯示最后一張圖片的時候,給第0個按鈕添加active; bullets[ index === 5 ? 0 : index ].classList.add("active"); } bindEvent(); var interval = null; function autoPlay(){ // 間隔3s,讓js點擊一下下一頁按鈕; interval = setInterval( function(){ // 虛擬點擊 : next_btn.dispatchEvent( new Event("click")); } , 3000 ) } function stop(){ clearInterval( interval ) } autoPlay();</script>
示例如下
項目背景:uniapp h6應用,為了提示用戶下載,這里有個 提示下載的dom, 本想來想在 每個 tabbar 中添加(一共添加四個);但是想 嘗試換種玩法 如下: 效果如下:
直接上代碼:
function showDownloadTisp() { console.log('--------------------------->showDownloadTisp') // #ifdef H5 const parent = document.querySelector('.uni-tabbar') console.log('parent:', parent) const tips = document.createElement('p') tips.id = 'downloadTisp' tips.setAttribute('style', 'background: rgba(51,51,51,0.75);' ) tips.setAttribute('style', `background: rgba(51,51,51,0.75); position: fixed;width: 100%;height: ${uni.upx2px(120)}px; bottom:${uni.upx2px(-140)}px; display: flex;flex-direction: row;justify-content: space-between;align-items:center;transition:0.5s;` ) const desParent = document.createElement('p') const des = document.createTextNode('Download our App, you will get a better experience.') desParent.appendChild(des) desParent.setAttribute('style', `padding: 0;color: #FFFFFF;font-size:${uni.upx2px(24)}px;margin-left: ${uni.upx2px(27)}px;padding-right: ${uni.upx2px(25)}px;` ) const iosImage = document.createElement('img') iosImage.src = '../static/guide/download ios.png' iosImage.setAttribute('style', `width: ${uni.upx2px(154)}px;height: ${uni.upx2px(54)}px;margin-right: ${uni.upx2px(25)}px;`) const anroidImage = document.createElement('img') anroidImage.src = '../static/guide/download android.png' anroidImage.setAttribute('style', `width: ${uni.upx2px(154)}px;height: ${uni.upx2px(54)}px;margin-right: ${uni.upx2px(50)}px;`) const closedImage = document.createElement('img') closedImage.src = '../static/guide/download closd.png' closedImage.setAttribute('style', `width: ${uni.upx2px(30)}px;height: ${uni.upx2px(30)}px;position: absolute;right:${uni.upx2px(15)}px;top: ${uni.upx2px(15)}px;padding: ${uni.upx2px(5)};` ) tips.appendChild(desParent) tips.appendChild(iosImage) tips.appendChild(anroidImage) tips.appendChild(closedImage) iosImage.onclick = () => { //console.log("iosImage.onclick") window.location.href = 'https://apps.apple.com/cn/app/gbm001/id1574324240' // window.open('https://www.baidu.com/') } anroidImage.onclick = () => { //console.log("anroidImage.onclick") window.location.href = 'https://play.google.com/store/apps/details?id=com.vandream.gbmpro' // window.open('https://www.sina.com.cn/') } closedImage.onclick = () => { console.log("closedImage.onclick") tips.remove() } parent.parentNode.appendChild(tips) //parent.appendChild(tips) setTimeout(() => { tips.style.transform = `translateY(${uni.upx2px(-140) - 50}px);` //console.log(" tips.style.transform done") }, 2500) // #endif }
通過代碼創建節點 并且代碼這是 style; 以及動畫;在應用啟動的時候 調用就可以了;
關于 js 設置變換動畫;這邊改成了 3D 的形式
function showDownloadTisp() { // #ifdef H5 const parent = document.querySelector('.uni-tabbar') // console.log('parent:', parent) const tips = document.createElement('p') tips.id = 'downloadTisp' // tips.setAttribute('style', // 'background: rgba(51,51,51,0.75);' // ) // tips.setAttribute('style', // `background: rgba(51,51,51,0.75); position: fixed;width: 100%;height: ${uni.upx2px(120)}px; bottom:${uni.upx2px(-140)}px; display: flex;flex-direction: row;justify-content: space-between;align-items:center;transition:0.5s;` // ) tips.setAttribute('style', `background: rgba(51,51,51,0.75); position: fixed;width: 100%;height: ${uni.upx2px(120)}px; bottom:50px; display: flex;flex-direction: row;justify-content: space-between;align-items:center;transition:0.5s; transform-origin:center bottom; transform:perspective(900px) rotateX(90deg);` ) const desParent = document.createElement('p') const des = document.createTextNode('Download our App, you will get a better experience.') desParent.appendChild(des) desParent.setAttribute('style', `padding: 0;color: #FFFFFF;font-size:${uni.upx2px(24)}px;margin-left: ${uni.upx2px(27)}px;padding-right: ${uni.upx2px(10)}px;` ) const iosImage = document.createElement('img') // iosImage.src = '../static/guide/download ios.png' iosImage.src = 'https://img.vandream.com/54/0330f6211506cc.png' iosImage.setAttribute('style', `width: ${uni.upx2px(154)}px;height: ${uni.upx2px(54)}px;margin-right: ${uni.upx2px(25)}px;`) const anroidImage = document.createElement('img') // anroidImage.src = '../static/guide/download android.png' anroidImage.src = 'https://img.vandream.com/53/0330f45097465d.png' anroidImage.setAttribute('style', `width: ${uni.upx2px(154)}px;height: ${uni.upx2px(54)}px;margin-right: ${uni.upx2px(50)}px;`) const closedImage = document.createElement('img') // closedImage.src = '../static/guide/download closd.png' closedImage.src = 'https://img.vandream.com/52/0330f523d7709d.png' closedImage.setAttribute('style', `width: ${uni.upx2px(30)}px;height: ${uni.upx2px(30)}px;position: absolute;right:${uni.upx2px(15)}px;top: ${uni.upx2px(15)}px;padding: ${uni.upx2px(5)};` ) tips.appendChild(desParent) tips.appendChild(iosImage) tips.appendChild(anroidImage) tips.appendChild(closedImage) iosImage.onclick = () => { console.log('iosImage.onclick') window.location.href = 'https://apps.apple.com/cn/app/gbm001/id1574324240' // window.open('https://www.baidu.com/') } anroidImage.onclick = () => { console.log('anroidImage.onclick') window.location.href = 'https://play.google.com/store/apps/details?id=com.vandream.gbmpro' // window.open('https://www.sina.com.cn/') } closedImage.onclick = () => { console.log('closedImage.onclick') tips.remove() } parent.parentNode.appendChild(tips) // parent.appendChild(tips) setTimeout(() => { // tips.style.transform = `translateY(${uni.upx2px(-140) - 50}px);` tips.style.transform = 'rotateX(0deg)' // console.log(" tips.style.transform done"):rotateX(90deg); }, 2500) // #endif }
感謝各位的閱讀,以上就是“javascript如何操作單個dom元素添加動畫”的內容了,經過本文的學習后,相信大家對javascript如何操作單個dom元素添加動畫這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。