国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

iscroll-lite源碼閱讀

jackzou / 2513人閱讀

摘要:最后調用函數來判斷是否需要重置位置保存事件的類型,如果跟已經保存的事件類型不是同一類,不繼續執行函數。如果在滾動狀態,調用函數來獲取的位置,調用方法讓停在當前位置。主要是選擇動畫效果,調用或是來進行滾動根據所選的動畫效果,在每一個下調用函數

背景

最近開發移動端項目用到了iscroll,踩了不少坑,因此閱讀源碼加深下對這個工具的了解。
本次閱讀的是iscroll-lite,包含了主要的功能,比完整版少了鼠標滾輪,滾動后對齊到特定位置,按鍵綁定及定制滾動條等功能。

主體結構

以當前的5.2版本為例,前三百行是工具函數,封裝了一些工具。三百行到四百行是構造函數IScroll,起到配置和初始化的效果。四百行到一千修改了IScroll的原型,給IScroll實例加上了各種方法。最后是模塊化相關的判斷。總體的結構是很清晰的。

工具函數
_prefixStyle

用于瀏覽器兼容。

    var _vendor = (function () {
        var vendors = ["t", "webkitT", "MozT", "msT", "OT"],
            transform,
            i = 0,
            l = vendors.length;

        for ( ; i < l; i++ ) {
            transform = vendors[i] + "ransform";
            if ( transform in _elementStyle ) return vendors[i].substr(0, vendors[i].length-1);
        }

        return false;
    })();
    function _prefixStyle (style) {
        if ( _vendor === false ) return false;
        if ( _vendor === "" ) return style;
        return _vendor + style.charAt(0).toUpperCase() + style.substr(1);
    }
能力檢測

判斷瀏覽器是否支持相關的特性

    me.extend(me, {
        hasTransform: _transform !== false,
        hasPerspective: _prefixStyle("perspective") in _elementStyle,
        hasTouch: "ontouchstart" in window,
        hasPointer: !!(window.PointerEvent || window.MSPointerEvent), // IE10 is prefixed
        hasTransition: _prefixStyle("transition") in _elementStyle
    });

還有一些事件監聽方面的處理函數,處理class的函數,處理慣性的函數momentum

父元素的位置
    me.offset = function (el) {
        var left = -el.offsetLeft,
            top = -el.offsetTop;

        // jshint -W084
        while (el = el.offsetParent) {
            left -= el.offsetLeft;
            top -= el.offsetTop;
        }
        // jshint +W084

        return {
            left: left,
            top: top
        };
    };
對事件進行分類
    me.extend(me.eventType = {}, {
        touchstart: 1,
        touchmove: 1,
        touchend: 1,

...略...

        MSPointerDown: 3,
        MSPointerMove: 3,
        MSPointerUp: 3
    });
動畫處理的函數
    me.extend(me.ease = {}, {
        quadratic: {
            style: "cubic-bezier(0.25, 0.46, 0.45, 0.94)",
            fn: function (k) {
                return k * ( 2 - k );
            }
        },
        ....略....
    });
tap和click的事件模擬
    me.tap = function (e, eventName) {
        var ev = document.createEvent("Event");
        ev.initEvent(eventName, true, true);
        ev.pageX = e.pageX;
        ev.pageY = e.pageY;
        e.target.dispatchEvent(ev);
    };
    me.click = function (e) {
        var target = e.target,
            ev;

        if ( !(/(SELECT|INPUT|TEXTAREA)/i).test(target.tagName) ) {
            // https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/initMouseEvent
            // initMouseEvent is deprecated.
            ev = document.createEvent(window.MouseEvent ? "MouseEvents" : "Event");
            ev.initEvent("click", true, true);
         ...略...
            target.dispatchEvent(ev);
        }
    };
touchAction的設定
    me.getTouchAction = function(eventPassthrough, addPinch) {
        var touchAction = "none";
        if ( eventPassthrough === "vertical" ) {
            touchAction = "pan-y";
        } else if (eventPassthrough === "horizontal" ) {
            touchAction = "pan-x";
        }
        if (addPinch && touchAction != "none") {
            // add pinch-zoom support if the browser supports it, but if not (eg. Chrome <55) do nothing
            touchAction += " pinch-zoom";
        }
        return touchAction;
    };
元素的位置
    me.getRect = function(el) {
        if (el instanceof SVGElement) {
            var rect = el.getBoundingClientRect();
            return {
                top : rect.top,
                left : rect.left,
                width : rect.width,
                height : rect.height
            };
        } else {
            return {
                top : el.offsetTop,
                left : el.offsetLeft,
                width : el.offsetWidth,
                height : el.offsetHeight
            };
        }
    };
構造函數

先是獲取了wrapper和scroller,接著是一些能力檢測。

在這里通過eventPassthrough,scrollY,scrollX,freeScroll來配置滾動的方向。只有scrollY是默認為true,其他都是未設定undefined.
當eventPassthrough的值為vertical時,在垂直方向不使用iscroll的滾動,使用原生的滾動。為horizontal時,則水平方向如此。想要二維自由滾動,不能設置該值。

    this.options.eventPassthrough = this.options.eventPassthrough === true ? "vertical" : this.options.eventPassthrough;
    this.options.scrollY = this.options.eventPassthrough == "vertical" ? false : this.options.scrollY;
    this.options.scrollX = this.options.eventPassthrough == "horizontal" ? false : this.options.scrollX;
    this.options.freeScroll = this.options.freeScroll && !this.options.eventPassthrough;

接著還有其他屬性的設定,最后調用_init,refresh,scrollTo,enable等函數

一些原形上綁定的函數
_init

只調用了_initEvents函數

_initEvents

在這里根據配置給wrapper或是window加入了鼠標事件,pointer事件和觸摸事件,transition事件等事件的監聽。這里用到了一種少見的事件綁定方法,在這里事件綁定的第三個參數不是常見的function,而是iscroll對象。而iscroll對象里有一個方法handleEvent,這種方式的好處就是可以傳遞this

handleEvent: function (e) {
        switch ( e.type ) {
            case "touchstart":
            case "pointerdown":
            case "MSPointerDown":
            case "mousedown":
                this._start(e);
                break;
            case "touchmove":
            case "pointermove":
            case "MSPointerMove":
            case "mousemove":
                this._move(e);
                break;
            case "touchend":
            case "pointerup":
            case "MSPointerUp":
            case "mouseup":
            case "touchcancel":
            case "pointercancel":
            case "MSPointerCancel":
            case "mousecancel":
              this._end(e);
            ...略...
            }
        }
           
refresh

保存wrapper和scroller的尺寸信息,及wrapper的位置信息。設定最大滾動距離,為負數。
接著判斷是否存在水平滾動或是垂直滾動

        this.maxScrollX        = this.wrapperWidth - this.scrollerWidth;
        this.hasHorizontalScroll    = this.options.scrollX && this.maxScrollX < 0;

如果瀏覽器支持pointer,使用touchAction來限制wrapper。
最后調用resetPosition函數來判斷是否需要重置位置

_start

保存事件的類型,如果跟已經保存的事件類型不是同一類,不繼續執行函數。
如果scroller在滾動狀態,調用getComputedPosition函數來獲取scroller的位置,調用_translate方法讓scroller停在當前位置。
保存當前的位置信息和觸發事件的位置信息,設置狀態信息

    this.initiated    = utils.eventType[e.type];
    this.moved        = false;
    this.distX        = 0;
    this.distY        = 0;
    this.directionX = 0;
    this.directionY = 0;
    this.directionLocked = 0;
    this.startX    = this.x;
    this.startY    = this.y;
    this.absStartX = this.x;
    this.absStartY = this.y;
    this.pointX    = point.pageX;
    this.pointY    = point.pageY;
_move

獲取事件的位置信息,對比_start中保存的位置信息,計算偏移值

    var point        = e.touches ? e.touches[0] : e,
        deltaX        = point.pageX - this.pointX,
        deltaY        = point.pageY - this.pointY,

保存新的位置信息,位移信息

    this.pointX        = point.pageX;
    this.pointY        = point.pageY;

    this.distX        += deltaX;
    this.distY        += deltaY;

如果偏移小于10像素不執行滾動。偏移的大小值和設定來根據判斷滾動的方向。最后調用_translate函數來滾動

_translate

根據設定,修改滾動條中的transform的值或是修改left和top的值來進行具體的滾動,保存信息到this.x和this.y

        if ( this.options.useTransform ) {
            this.scrollerStyle[utils.style.transform] = "translate(" + x + "px," + y + "px)" + this.translateZ;
        } else {
            x = Math.round(x);
            y = Math.round(y);
            this.scrollerStyle.left = x + "px";
            this.scrollerStyle.top = y + "px";
        }
_end

讓scroller滾動到整數位置,根據位移和觸發事件的時間的情況,摸你tap事件、click事件或是執行flick事件或調用scrollTO函數進行慣性滾動。

scrollTo

主要是選擇動畫效果,調用__translate或是_animate來進行滾動

_animate

根據所選的動畫效果,在每一個requestAnimationFrame下調用_translate函數

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/94216.html

相關文章

  • 一篇文章把本該屬于你的源碼天賦還給你

    摘要:一些方法不應該這樣不應該漫無目的地隨手拿起一分源碼,試圖去通讀。應該這樣精心挑選要閱讀的源碼項目。這最好是與你的編程語言你的工作內容你的興趣所在相關的,這樣才能更切實地感受到閱讀源碼給你帶來的益處,更有動力繼續。 showImg(https://segmentfault.com/img/bVbcvmm?w=785&h=525); 怎么閱讀源碼 沒有經驗的技術差底子薄的初級程序員,如何閱...

    chanjarster 評論0 收藏0
  • 精讀《源碼學習》

    摘要:精讀原文介紹了學習源碼的兩個技巧,并利用實例說明了源碼學習過程中可以學到許多周邊知識,都讓我們受益匪淺。討論地址是精讀源碼學習如果你想參與討論,請點擊這里,每周都有新的主題,周末或周一發布。 1. 引言 javascript-knowledge-reading-source-code 這篇文章介紹了閱讀源碼的重要性,精讀系列也已有八期源碼系列文章,分別是: 精讀《Immer.js》源...

    aboutU 評論0 收藏0
  • android高級工程師-閱讀源碼的經驗總結

    摘要:對于一個有追求的程序員來說必須讀源碼,當然閱讀源碼是一件令人頭疼的事。和差不多,想法一樣,但是和相比,搜索到的有價值結果不是很多,沒有的理想,還有一些來著百度知道。 對于一個有追求的程序員來說必須讀源碼,當然閱讀源碼是一件令人頭疼的事。閱讀別人的代碼遠比自己寫代碼要難。Linus 是Linux的早期作者,一句影響深遠的話是Read The Fucking Source Code(RTF...

    idisfkj 評論0 收藏0

發表評論

0條評論

jackzou

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<