javascript dom編程藝術學習筆記之實現動畫效果

javascript dom編程藝術學習筆記之實現動畫效果

@(JavaScript)[學習筆記]javascript


[toc]css


1.動畫基礎知識

動畫是讓元素的位置隨時間的變化不斷變化html

1.1 位置

假定有個id爲message的元素,能夠用javascript來設置這個屬性的位置positionMessage() 也能夠用javascript來改變元素的位置"moveMessage()" 頁面加載時,兩個函數幾乎同時運行,肉眼看不到變化,位置就改變了,沒法實現動畫效果java

1.2 時間

能夠用setTimeout("function", interbal)函數 第一個參數是函數名 第二個參數是延時時間,單位毫秒 取消函數clearTimeout(variable)web

function positionMessage() {
  if (!document.getElementById) return false;
  if (!document.getElementById("message")) return false;
  var elem = document.getElementById("message");
  elem.style.position = "absolute";
  elem.style.left = "50px";
  elem.style.top = "100px";
  movement = setTimeout("moveMessage()" , 5000)
  }

movement是一個全局變量。能夠在該函數之外任意位置取消跳躍編程

1.3 時間遞增量

更新moveMessage函數,讓元素以漸變方式移動瀏覽器

  • 獲取元素的位置
  • 若是元素到達目的地,則退出這個函數
  • 若是還沒有到達,向目的地移動
  • 通過一段時間間隔,重複步驟一

JavaScript函數parseInt能夠把字符串數值提取出來 用parseFloaat函數能夠提取浮點數安全

1.4 抽象

編寫moveElement函數app

function moveElement(elementID,final_x,final_y,interval) {
  if (!document.getElementById) return false;
  if (!document.getElementById(elementID)) return false;
  var elem = document.getElementById(elementID);
  var xpos = parseInt(elem.style.left);
  var ypos = parseInt(elem.style.top);
  if (xpos == final_x && ypos == final_y) {
    return true;
  }
  if (xpos < final_x) {
    xpos++;
  }
  if (xpos > final_x) {
    xpos--;
  }
  if (ypos < final_y) {
    ypos++;
  }
  if (ypos > final_y) {
    ypos--;
  }
  elem.style.left = xpos + "px";
  elem.style.top = ypos + "px";
  var repeat = "moveElement('"+elementID+"',"+final_x+","+final_y+","+interval+")";
  movement = setTimeout(repeat,interval);
}
1.5 使用moveElement函數
moveElement("mwssage", 200, 100, 10);

2. 實用的動畫

動畫元素易引發用戶反感,除非瀏覽器容許用戶「凍結」移動的內容,不然就應該避免內容在頁面中移動dom

2.1 提出問題

有一個包含一些列連接的網頁,用戶鼠標懸停某個連接,用先睹爲快的方式告訴用戶連接去往何處,能夠展現一張預覽圖片 標記語言以下

<!DOCTYPE html>
<html lang="en">
<head>
  <meta http-equiv="content-type" content="text/html; charset=utf-8" />
  <title>Web Design</title>
  <style type="text/css" media="screen">
    @import url(styles/layout.css);
  </style>
  <script type="text/javascript" src="scripts/addLoadEvent.js">
  </script>
  <script type="text/javascript" src="scripts/moveElement.js">
  </script>
  <script type="text/javascript" src="scripts/prepareSlideshow.js">
  </script>
</head>
<body>
  <h1>Web Design</h1>
  <p>These are the things you should know.</p>
  <ol id="linklist">
    <li>
      <a href="structure.html">Structure</a>
    </li>
    <li>
      <a href="presentation.html">Presentation</a>
    </li>
    <li>
      <a href="behavior.html">Behavior</a>
    </li>
  </ol>
  <div id="slideshow">
    <img src="topics.gif" alt="building blocks of web design" id="preview" />
  </div>
</body>
2.3 CSS

overflow屬性處理元素尺寸超過容器的狀況。 overflow可取屬性有四種:visible,hidden,scroll,auto

  • visible:不裁減溢出內容
  • hidden:隱藏溢出內容
  • scroll:隱藏溢出內容,但有一個滾動條
  • auto:發生溢出時纔有滾動條,無溢出不滾動
2.4 JavaScript
function prepareSlideshow() {
// Make sure the browser understands the DOM methods
  if (!document.getElementsByTagName) return false;
  if (!document.getElementById) return false;
// Make sure the elements exist
  if (!document.getElementById("linklist")) return false;
  var slideshow = document.createElement("div");
  slideshow.setAttribute("id","slideshow");
  var preview = document.createElement("img");
  preview.setAttribute("src","topics.gif");
  preview.setAttribute("alt","building blocks of web design");
  preview.setAttribute("id","preview");
  slideshow.appendChild(preview);
  var list = document.getElementById("linklist");
  insertAfter(slideshow,list);
// Get all the links in the list
  var links = list.getElementsByTagName("a");
// Attach the animation behavior to the mouseover event
  links[0].onmouseover = function() {
    moveElement("preview",-100,0,10);
  }
  links[1].onmouseover = function() {
    moveElement("preview",-200,0,10);
  }
  links[2].onmouseover = function() {
    moveElement("preview",-300,0,10);
  }
}
addLoadEvent(prepareSlideshow);
2.5 變量做用域問題

若是把鼠標在連接間快速移動,動畫效果會變得很混亂 這是由全局變量引發的,在抽象化過程,未對變量movement作任何修改 存在與特定元素有關的變量,就是屬性

function moveElement(elementID,final_x,final_y,interval) {
  if (!document.getElementById) return false;
  if (!document.getElementById(elementID)) return false;
  var elem = document.getElementById(elementID);
  if (elem.movement){
      clearTimeout(elem.movement);
  }
  var xpos = parseInt(elem.style.left);
  var ypos = parseInt(elem.style.top);
  if (xpos == final_x && ypos == final_y) {
    return true;
  }
  if (xpos < final_x) {
    xpos++;
  }
  if (xpos > final_x) {
    xpos--;
  }
  if (ypos < final_y) {
    ypos++;
  }
  if (ypos > final_y) {
    ypos--;
  }
  elem.style.left = xpos + "px";
  elem.style.top = ypos + "px";
  var repeat = "moveElement('"+elementID+"',"+final_x+","+final_y+","+interval+")";
  elem.movement = setTimeout(repeat,interval);
}

若是函數執行時已經有movement屬性,就用clearTimeout對其復位,這樣就只執行一條函數調用語句

2.6 改進動畫效果

讓過分更平滑一些,每次移動十分之一

dist = (final_x - dlist)/10;
xpos = xpos + dlist;

Math對象的ceil方法:返回不小於結果值的一個整數 Math對象的floor方法:返回大於結果值的一個整數 Math對象的round方法:返回與之最接近的一個整數

2.7 添加安全檢查
var xpos = parseInt(elem.style.left);
  var ypos = parseInt(elem.style.top);

這裏有一個假設,必須有left和top屬性 方法一:

if (!elem.stytle.left || !elem.stytle.top) return false;

方法二:

if (!elem.stytle.left){
	elem.stytle.left = "0px";
}
if (!elem.stytle.top){
	elem.stytle.top = "0px";
}
2.8 生成HTML標記
  • 建立div元素
  • 建立img元素
  • 把img元素加入div子元素
相關文章
相關標籤/搜索