零JS筆記 | javascript運動 | 时间版

jq的animate是典型的时间版框架 默認400ms


如中间放大,改width,height,left,top值

sMove2(this,{width:200,height:200,left:150,top:150})

出现的bug,快结束时会抖动,速度没变,但移动距离变,所以时间上先后到达存在问题;


时间版算法根据Tween算法而来;


参考:http://www.cnblogs.com/bluedream2009/archive/2010/06/19/1760909.html


初始化: t = 時間,【可變】,b,初始值,c改變量,d:總時間 return目標點


如:div初始100,2s内变化到200,则,t = 0~2; b:100,c:200-100,d:2s;

var Tween = {
    linear: function (t, b, c, d){  //匀速  
      return c*t/d + b;
    },
    ....

因为可能多个值,所以设置iCur变量为{},然后再for循环,初始化一0,判断是否opacity,初始化二:获取传值数据;

function tMove(obj, json,times,fx,fn) {
    var iCur = {};//不變,不能寫在定時器裏  b
    for(var attr in json){
        iCur[attr] = 0;
        if(attr=='opacity'){
            iCur[attr] = Math.round(getStyle(obj.attr)*100)
        }
        else{
            iCur[attr]=parseInt(getStyle(obj.attr));
        }
    }
    clearInterval(obj._t)
    obj._t = setInterval(function () {
        for(var attr in json){
            var value = Tween[fx](t,b,c,d) //目的,通過js獲取tbcd值
            if(attr == 'opacity'){
                obj.style.opacity = value/100;
                obj.style.filter = 'alpha(opacity='+value+')';

            }
            else{
                obj.style[attr] = value +'px'  //value是不断变化的值,即Tween return回来的结果;
            }
        }
    },13)
}

以上说明即目的:获取在Tween内,tbce参数的值,之后再返回结果给value;


注意:设置定时器时,还需要一个循环;

function tMove(obj, json, times, fx, fn) {
    if (typeof times == 'undefined') {
        times = 400;
        fx = 'linear';
    }
    if (typeof times == 'string') {
        if (typeof fx == 'function') {
            fn = fx;
        }
        fx = times;
        times = 400;
    }
    else if (typeof times == 'function') {
        fn = times;
        times = 400;
        fx = 'linear';
    }
    else if (typeof times == 'number') {
        if (typeof fx == 'function') {
            fn = fx;
            fx = 'linear';
        }
        else if (typeof fx == 'undefined') {
            fx = 'linear';
        }
    }
    var iCur = {
    }; //不變,不能寫在定時器裏  b
    for (var attr in json) {
        iCur[attr] = 0;
        if (attr == 'opacity') {
            iCur[attr] = Math.round(css(obj, attr) * 100);
        }
        else {
            iCur[attr] = parseInt(css(obj, attr));
        }
    }
    var startTime = now();
    clearInterval(obj._t);
    obj._t = setInterval(function () {
        var changeTime = now();
        var t = times - Math.max(0, startTime - changeTime + times);
        //在time的範圍內越來越小  2000-0;
        for (var attr in json) {
            var value = Tween[fx](t, iCur[attr], json[attr] - iCur[attr], times); //目的,通過js獲取tbcd值
            if (attr == 'opacity') {
                obj.style.opacity = value / 100;
                obj.style.filter = 'alpha(opacity=' + value + ')';
            }
            else {
                obj.style[attr] = value + 'px';
            }
        }
        if (t == times) {
            clearInterval(obj._t);
            fn && fn.call(obj);
        }
    }, 13)
}
function now() {
    return (new Date()).getTime();
}
var Tween = {
    linear: function (t, b, c, d) { //匀速
        return c * t / d + b;
    },
    easeIn: function (t, b, c, d) { //加速曲线
        return c * (t /= d) * t + b;
    },
    easeOut: function (t, b, c, d) { //减速曲线
        return - c * (t /= d) * (t - 2) + b;
    },
    easeBoth: function (t, b, c, d) { //加速减速曲线 缓冲
        if ((t /= d / 2) < 1) {
            return c / 2 * t * t + b;
        }
        return - c / 2 * ((--t) * (t - 2) - 1) + b;
    },
    easeInStrong: function (t, b, c, d) { //加加速曲线
        return c * (t /= d) * t * t * t + b;
    },
    easeOutStrong: function (t, b, c, d) { //减减速曲线
        return - c * ((t = t / d - 1) * t * t * t - 1) + b;
    },
    easeBothStrong: function (t, b, c, d) { //加加速减减速曲线
        if ((t /= d / 2) < 1) {
            return c / 2 * t * t * t * t + b;
        }
        return - c / 2 * ((t -= 2) * t * t * t - 2) + b;
    },
    elasticIn: function (t, b, c, d, a, p) { //正弦衰减曲线(弹动渐入)
        if (t === 0) {
            return b;
        }
        if ((t /= d) == 1) {
            return b + c;
        }
        if (!p) {
            p = d * 0.3;
        }
        if (!a || a < Math.abs(c)) {
            a = c;
            var s = p / 4;
        } else {
            var s = p / (2 * Math.PI) * Math.asin(c / a);
        }
        return - (a * Math.pow(2, 10 * (t -= 1)) * Math.sin((t * d - s) * (2 * Math.PI) / p)) + b;
    },
    elasticOut: function (t, b, c, d, a, p) { //正弦增强曲线(弹动渐出)
        if (t === 0) {
            return b;
        }
        if ((t /= d) == 1) {
            return b + c;
        }
        if (!p) {
            p = d * 0.3;
        }
        if (!a || a < Math.abs(c)) {
            a = c;
            var s = p / 4;
        } else {
            var s = p / (2 * Math.PI) * Math.asin(c / a);
        }
        return a * Math.pow(2, - 10 * t) * Math.sin((t * d - s) * (2 * Math.PI) / p) + c + b;
    },
    elasticBoth: function (t, b, c, d, a, p) {
        if (t === 0) {
            return b;
        }
        if ((t /= d / 2) == 2) {
            return b + c;
        }
        if (!p) {
            p = d * (0.3 * 1.5);
        }
        if (!a || a < Math.abs(c)) {
            a = c;
            var s = p / 4;
        }
        else {
            var s = p / (2 * Math.PI) * Math.asin(c / a);
        }
        if (t < 1) {
            return - 0.5 * (a * Math.pow(2, 10 * (t -= 1)) *
                    Math.sin((t * d - s) * (2 * Math.PI) / p)) + b;
        }
        return a * Math.pow(2, - 10 * (t -= 1)) *
                Math.sin((t * d - s) * (2 * Math.PI) / p) * 0.5 + c + b;
    },
    backIn: function (t, b, c, d, s) { //回退加速(回退渐入)
        if (typeof s == 'undefined') {
            s = 1.70158;
        }
        return c * (t /= d) * t * ((s + 1) * t - s) + b;
    },
    backOut: function (t, b, c, d, s) {
        if (typeof s == 'undefined') {
            s = 3.70158; //回缩的距离
        }
        return c * ((t = t / d - 1) * t * ((s + 1) * t + s) + 1) + b;
    },
    backBoth: function (t, b, c, d, s) {
        if (typeof s == 'undefined') {
            s = 1.70158;
        }
        if ((t /= d / 2) < 1) {
            return c / 2 * (t * t * (((s *= (1.525)) + 1) * t - s)) + b;
        }
        return c / 2 * ((t -= 2) * t * (((s *= (1.525)) + 1) * t + s) + 2) + b;
    },
    bounceIn: function (t, b, c, d) { //弹球减振(弹球渐出)
        return c - Tween['bounceOut'](d - t, 0, c, d) + b;
    },
    bounceOut: function (t, b, c, d) {
        if ((t /= d) < (1 / 2.75)) {
            return c * (7.5625 * t * t) + b;
        } else if (t < (2 / 2.75)) {
            return c * (7.5625 * (t -= (1.5 / 2.75)) * t + 0.75) + b;
        } else if (t < (2.5 / 2.75)) {
            return c * (7.5625 * (t -= (2.25 / 2.75)) * t + 0.9375) + b;
        }
        return c * (7.5625 * (t -= (2.625 / 2.75)) * t + 0.984375) + b;
    },
    bounceBoth: function (t, b, c, d) {
        if (t < d / 2) {
            return Tween['bounceIn'](t * 2, 0, c, d) * 0.5 + b;
        }
        return Tween['bounceOut'](t * 2 - d, 0, c, d) * 0.5 + c * 0.5 + b;
    }
}
function css(obj, attr) {
    return obj.currentStyle ? obj.currentStyle[attr] : getComputedStyle(obj) [attr]
}

See the Pen vNWQej by elmok (@elmok) on CodePen.


<script async src="//assets.codepen.io/assets/embed/ei.js"></script>



在瀏覽器中,若定時器處於運動狀態時切換到其他頁面,如最小化,切換到其他標籤等操作,瀏覽器會默認將定時器走得很慢,可能出現bug,可使用window.onfocus && window.onblur解決;按實際情獎況而訂;




不可避免的JQ animate()擴展

$(function () {
$('#div1').click(function () {
    $(this).animate({width:200,height:200},2000,'bounceIn')
})
$.extend(jQuery.easing,{
    easeIn: function(x,t, b, c, d){  //加速曲线
        return c*(t/=d)*t + b;
    },
    easeOut: function(x,t, b, c, d){  //减速曲线
        return -c *(t/=d)*(t-2) + b;
    },
    easeBoth: function(x,t, b, c, d){  //加速减速曲线
        if ((t/=d/2) < 1) {
            return c/2*t*t + b;
        }
        return -c/2 * ((--t)*(t-2) - 1) + b;
    },
    easeInStrong: function(x,t, b, c, d){  //加加速曲线
        return c*(t/=d)*t*t*t + b;
    },
    easeOutStrong: function(x,t, b, c, d){  //减减速曲线
        return -c * ((t=t/d-1)*t*t*t - 1) + b;
    },
    easeBothStrong: function(x,t, b, c, d){  //加加速减减速曲线
        if ((t/=d/2) < 1) {
            return c/2*t*t*t*t + b;
        }
        return -c/2 * ((t-=2)*t*t*t - 2) + b;
    },
    elasticIn: function(x,t, b, c, d, a, p){  //正弦衰减曲线(弹动渐入)
        if (t === 0) {
            return b;
        }
        if ( (t /= d) == 1 ) {
            return b+c;
        }
        if (!p) {
            p=d*0.3;
        }
        if (!a || a < Math.abs(c)) {
            a = c;
            var s = p/4;
        } else {
            var s = p/(2*Math.PI) * Math.asin (c/a);
        }
        return -(a*Math.pow(2,10*(t-=1)) * Math.sin( (t*d-s)*(2*Math.PI)/p )) + b;
    },
    elasticOut: function(x,t, b, c, d, a, p){    //正弦增强曲线(弹动渐出)
        if (t === 0) {
            return b;
        }
        if ( (t /= d) == 1 ) {
            return b+c;
        }
        if (!p) {
            p=d*0.3;
        }
        if (!a || a < Math.abs(c)) {
            a = c;
            var s = p / 4;
        } else {
            var s = p/(2*Math.PI) * Math.asin (c/a);
        }
        return a*Math.pow(2,-10*t) * Math.sin( (t*d-s)*(2*Math.PI)/p ) + c + b;
    },
    elasticBoth: function(x,t, b, c, d, a, p){
        if (t === 0) {
            return b;
        }
        if ( (t /= d/2) == 2 ) {
            return b+c;
        }
        if (!p) {
            p = d*(0.3*1.5);
        }
        if ( !a || a < Math.abs(c) ) {
            a = c;
            var s = p/4;
        }
        else {
            var s = p/(2*Math.PI) * Math.asin (c/a);
        }
        if (t < 1) {
            return - 0.5*(a*Math.pow(2,10*(t-=1)) *
                    Math.sin( (t*d-s)*(2*Math.PI)/p )) + b;
        }
        return a*Math.pow(2,-10*(t-=1)) *
                Math.sin( (t*d-s)*(2*Math.PI)/p )*0.5 + c + b;
    },
    backIn: function(x,t, b, c, d, s){     //回退加速(回退渐入)
        if (typeof s == 'undefined') {
            s = 1.70158;
        }
        return c*(t/=d)*t*((s+1)*t - s) + b;
    },
    backOut: function(x,t, b, c, d, s){
        if (typeof s == 'undefined') {
            s = 3.70158;  //回缩的距离
        }
        return c*((t=t/d-1)*t*((s+1)*t + s) + 1) + b;
    },
    backBoth: function(x,t, b, c, d, s){
        if (typeof s == 'undefined') {
            s = 1.70158;
        }
        if ((t /= d/2 ) < 1) {
            return c/2*(t*t*(((s*=(1.525))+1)*t - s)) + b;
        }
        return c/2*((t-=2)*t*(((s*=(1.525))+1)*t + s) + 2) + b;
    },
    bounceIn: function(x,t, b, c, d){    //弹球减振(弹球渐出)
        return c - this['bounceOut'](x,d-t, 0, c, d) + b;
    },
    bounceOut: function(x,t, b, c, d){
        if ((t/=d) < (1/2.75)) {
            return c*(7.5625*t*t) + b;
        } else if (t < (2/2.75)) {
            return c*(7.5625*(t-=(1.5/2.75))*t + 0.75) + b;
        } else if (t < (2.5/2.75)) {
            return c*(7.5625*(t-=(2.25/2.75))*t + 0.9375) + b;
        }
        return c*(7.5625*(t-=(2.625/2.75))*t + 0.984375) + b;
    },
    bounceBoth: function(x,t, b, c, d){
        if (t < d/2) {
            return this['bounceIn'](x,t*2, 0, c, d) * 0.5 + b;
        }
        return this['bounceOut'](x,t*2-d, 0, c, d) * 0.5 + c*0.5 + b;
    }

})
})

零JQ筆記 | ajax

AJAX

callback返回三个参数,在load()方法中,只要请求完成,无法是否成功,callback即会被触发

$('#resText').load('test.html .para', function (responseText, textStatus, XMLHttpRequest) {
  console.log(responseText); //请求返回的内容
  console.log(textStatus); //请求状态:success,error
  console.log(XMLHttpRequest); //XMLHttpRequest对象
});

$.get()/$.post()
$.get(url ,data[,type])

$.get(url, 发送到服务器的key/v, 回调,服务器返回格式xml/html/script/json/text/_default)

发送到get1.php,发送的内容为:username/content/,要求服务器返回html格式;

$.get('get1.php', {
  username: $('#username').val(),
  content: $('#content').val()
}, function (data, textStatus) {
  $('#resText').html(data); // 把返回的数据添加到页面上
}
);

function (data, textStatus): data:返回的内容:xml/json/html等; testStatus: 请求状态:success/error/notmodified/timeout; 而且只有当数据成功返回success反才调用,注意与load()不一样;

data返回的数据内容如

返回内容 处理
html 可直接输出$(obj).html()
XML XML文档可使用常用的attr()/find()/filter()等处理
function (data, textStatus) {
  var username = $(data).find('comment').attr('username');
  var content = $(data).find('comment content').text();
  var txtHtml = '<div class=\'comment\'><h6>' + username + ':</h6><p class=\'para\'>' + content + '</p></div>';
  $('#resText').html(txtHtml); // 把返回的数据添加到页面上
}

</td>

</tr> 
<tr> 
 <td>json</td> 
 <td>json格式简洁,易读,对返回处理后输出html
function (data, textStatus) {
  var username = data.username;
  var content = data.content;
  var txtHtml = '<div class=\'comment\'><h6>' + username + ':</h6><p class=\'para\'>' + content + '</p></div>';
  $('#resText').html(txtHtml); // 把返回的数据添加到页面上
},
'json'); //要求服务器返回json格式

</td>

</tr> 

</tbody>
</table>


好处与利弊:html片段最简单;json数据重用、性能、大小、较好; xml远程应用程序未知时较好,适用于任何语言;



$.getScript('test.js')
当然,也有回调:

$.getScript('test.js',function(){
    $('#go').click(function(){
    .....
})
})



$.getJSON('test.json')
当然,也有回调:

$.getJSON('test.json',function(data){
   //data返回数据
})
})



通过$.getJSON得到的数据data,可使用$.each()来遍历[对象和数组]

$.each(data, function (i, elem) {
console.log(i); //i表示索引,elem表示值
console.log(elem['username']);
})
通过JSONP加载其它网站数据;跨域首选

$.getJSON('http://api...../',function(data){
   $.each(data[obj],function(i,elem){
        .....
        //i可作为判断条件,退出each循环,return false;
    })
})

底层$.ajax(options)方法

<style>

.ajax-tb-1 { border-collapse: collapse }
.ajax-tb-2{border-collapse: collapse}
.ajax-tb-2 td{border:1px solid #ddd;padding:5px 10px;line-height: 2;}
.ajax-tb-2 tr>td:nth-of-type(2){white-space: nowrap; }
.ajax-tb-1 td, .ajax-tb-1 th { border: 1px solid #ddd; padding: 5px 10px; line-height: 2; }

</style>

該方法只有一個參數,所有包含在$.ajav()裏,參數以key/value形式,所有參數除url外都是可選

參數名 類型 說明
url string 默認,發送,請求地址
type string post/get,默認get,注意put/delete部分瀏覽器不支持
data obj or string 發送到服務器的數據,如非字符串,則自動轉字符串形式 <br/>請求獎附加在url後,防止自動轉可查看processData選項,對象必須為key/value格式;<br/>例如 {fool:'bar1',foo2:'bar2'} ==>>&foo1=bar1&foo2=bar2 <br/>如果是數組,轉為不同值對應同一個名稱。 {foo:['bar1','bar2']}==>>&foo=bar1&foo=bar2
dataType string 預期服務器返回類型,不指定:自動按http包mime信息返回responseXML或responseText,並作為callback參數傳遞;可用類型:<br/>xml:返xml文檔,可用jq處理 <br/>html:純html信息,包含的script標籤會在插入dom時執行<br/>script:純js代碼,不自動緩存結果,除非設置cache,注意:遠程請求時所有跨域請求都轉為get請求 <br/>json:返回JSON數據;<br/>jsonp:使用JSONP形式調用函數時,如:myurl?callback=?,jq將自動替換後一個?為正確函數名 <br/>text:純txt字符串
beforeSend Function 發送前可修改XMLHttpRequst對象函數,如:添加算定義HTTP頭,在beforeSend中返回false可取消本次ajax請求,XMLHttpRequest對象是惟一請求;<br/> function(XMLHttpRequest){ this //調用本次ajax請求傳傳遞的options參數 }
complete Function 請求完成後回調,成功失敗都可調用;<br/>參數:XMLHttpRequest對像和一個描述成功請求類型的string, function(XMLHttpRequest,textStatus){ this //調用本次ajax請求傳傳遞的options參數 }
success Function 請求成功後callback <br/> 1).由服務器返回,按給的dataType參數進行處理後數據 ; <br/>2).描述狀態string; <br/> function(data,textStatus){ data; //可能是xmlDoc,jsonObj,html,text等等, this; //調用本次ajax請求傳傳遞的options參數 }
error Function 請求失敗時調用,3參數: function(XMLHttpRequest ,textStatus,errorThrown){ //通常情況下textStatus和errorThown只有其中一個包含信息 this //調用本次ajax請求傳傳遞的options參數 }
global Boolean 默認true,表示是否觸發全局ajax事件,設置false將不觸發,AjaxStart或AjaxStop可用於控制各種Ajax事件;

代替$.getScript()

$(function () {
    $('send').click(function () {
        $.ajax({
            type:'GET',
            url:'test.js',
            dataType:'script'
        })
    })
})

代替$.getJSON()

$.ajax({
    type:'GET',
    url:'test.json',
    dataType:'json',
    success: function (data) {
        var html=''
        $.each(data, function (index,elems) {
            html +=index 
            +':' 
            +elems['username'] 
            +':' 
            +elems['content']
        })
        $('body').html(html)
    }
})

test.json

[
  {
    "username": "张三",
    "content": "沙发."
  },
  {
    "username": "李四",
    "content": "板凳."
  },
  {
    "username": "王五",
    "content": "地板."
  }
]



1、serialize()方法,表單序列化,所有元素必須設置name,使用name來獲取;

<form action="" id="f2">
    <p>user:<input type="text" name="user" id=""/></p>
    <p>password:<input type="password" name="pws" id=""/></p>
    <input type="checkbox" name="ck" id=""/>
    <input type="checkbox" name="ck2" id=""/>
    <input type="radio" name="rrr" id=""/>
    <input type="radio" name="rrr" id=""/>
    <input type="radio" name="rrr" id=""/>
    <p><input type="button" id="send2" value="提交"/></p>
</form>
$('#send2').click(function () {
    console.log($('#f2').serialize());
    console.log($('#f2').serializeArray()); //若使用serializeArray()輸出對像如下圖
})

輸出:user=elmok&pws=elmokpws&ck2=on&rrr=on

輸出對象:

所以可使用$.each()

var fdata = $('#f2').serializeArray(),
$.each(fdata ,function(i,elems){
    $('body').append(fdata.value + '')
})

例子:

<form id="form1" action="#">
    <p>评论:</p>
    <p>姓名: <input type="text" name="username" id="username" /></p>
    <p>内容: <textarea name="content" id="content"  rows="2" cols="20"></textarea></p>
    <p><input type="button" id="send" value="提交"/></p>
</form>
<div id="resText"></div>

為取得姓名和內容,必須將字段一個個添加到data參數中

$('#send').click(function () {
    $.get('get1.php',{
        username:$('#username').val(),
        content:$('#content').val()
    }, function (data, textStatus) {
        $('#resText').html(data)  //返回頁面
    })
})

少量可以,多則使用serialize()方法

$('#send').click(function () {
    $.get('get1.php',$('#form1').serialize(), function (data, textStatus) {
        $('#resText').html(data)
    })
})

需要注意:$.get()方法中,data參數不僅可使用映射方式:

{
username:$('#username').val(),
content:$('#content').val()
}

也可使用字符串[中文問題]方式:注意,字符串無{}號

$('#send').click(function () {
    $.get('get1.php',
 "username="+encodeURIComponent($('#username').val())+"&content="+encodeURIComponent($('#content').val())     
    , function (data, textStatus) {
        $('#resText').html(data)
    })
})



3、$.param()方法
serialize()方法核心,對一個arr或obj按照key/value進行序列化

var obj = {a:1,b:2,c:3}
var k = $.param(obj)
//輸出:a=1&b=2&c=3



jq中的ajax全局事件
加載中loading....提示,定義好
loading......

$("#loading").ajaxStart(function(){
  $(this).show();
});
$("#loading").ajaxStop(function(){
  $(this).hide();
});

其他地方使用,仍有效,因為是全局;

另外幾個方法


ajaxComplete(fn..):請求完成時執行;

ajaxError(fn..):發生錯誤時執行,捕捉到錯誤可作為最後一個參數傳遞;

ajaxSend(fn..):請求發送前執行;

ajaxSuccess(fn...):請求成功時執行;

1、jq1.5之前,如不想某個ajax請求受全局方法影響

$.ajax({
    url:'...',
    global:false
})

1、jq1.5之後,如不想某個ajax請求受全局方法影響,則在每次發送請求之前設置

$.ajaxPrefilter(function (options) {
    options.global = true;
})
© 2026 J.chongj.chong@qq.com 模板由cho制作.