成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

嘗試封裝一個(gè)自己的模塊

philadelphia / 512人閱讀

摘要:上一周學(xué)了閉包和模塊,這一周仍然是跟著波同學(xué),試著封裝了一個(gè)拖拽模塊。

上一周學(xué)了閉包和模塊,這一周仍然是跟著@波同學(xué),試著封裝了一個(gè)拖拽模塊。過(guò)程中經(jīng)歷了一些曲折,最開(kāi)始我是打算只用style.left的方式,但是這個(gè)需要設(shè)置position:absolute。可能對(duì)代碼造成一定影響。雖然CSS的transform會(huì)影響兼容,但這里我還是使用了這個(gè)屬性的translate來(lái)完成移動(dòng)。
只用style完成的代碼

話不多說(shuō),直接上代碼:

html和css,這里必須設(shè)置position,第一次寫這段代碼的時(shí)候忘了,結(jié)果盡管JS寫對(duì)了,效果完全出不來(lái)....真是短路了




    
    學(xué)習(xí)
    



  
重點(diǎn)!!JS
;    //這個(gè)分號(hào)是為了防止其他的模塊最后忘記加分號(hào),導(dǎo)致錯(cuò)誤。
(function() {
  
  //構(gòu)造函數(shù),屬于每一個(gè)實(shí)例
  function Drag(selector) {
    this.elem = typeof selector == "object" ? selector : document.getElementById(selector);
    //鼠標(biāo)初始位置
    this.startX = 0;
    this.startY = 0;
    //元素初始位置
    this.sourceX = 0;
    this.sourceY = 0;

    this.init();
  }

  //原型,共有的
  Drag.prototype = {
    constructor: Drag,
    init: function() {
      this.setDrag();
    },

    //用于獲取元素當(dāng)前的位置信息
    getPosition: function() {
      var that = this;
      var pos = {};
      pos = {
        x: that.elem.offsetLeft,
        y: that.elem.offsetTop
      };
      return pos;
    },
    //用來(lái)設(shè)置當(dāng)前元素的位置
    setPosition: function(pos) {
      this.elem.style.left = pos.x + "px";
      this.elem.style.top = pos.y + "px";
    },

    //該方法用來(lái)綁定事件
    setDrag: function() {
      var self = this;
      this.elem.addEventListener("mousedown", start, false);

      function start(event) {

        self.startX = event.pageX;
        self.startY = event.pageY;

        var pos = self.getPosition();

        self.sourceX = pos.x;
        self.sourceY = pos.y;

        document.addEventListener("mousemove", move, false);
        document.addEventListener("mouseup", end, false);
      }

      function move(event) {
        //總體思想:鼠標(biāo)距瀏覽器距-鼠標(biāo)距元素距離
        var currentX = event.pageX; //當(dāng)前的鼠標(biāo)x位置
        var currentY = event.pageY; //當(dāng)前的鼠標(biāo)y位置

        var distanceX = currentX - self.startX; //鼠標(biāo)移動(dòng)的距離x
        var distanceY = currentY - self.startY; //鼠標(biāo)移動(dòng)的距離y

        self.setPosition({
          x: self.sourceX + distanceX,
          y: self.sourceY + distanceY
        });

      }

      function end(event) {
        document.removeEventListener("mousemove", move);
        document.removeEventListener("mouseup", end);
      }
    }
  };
  
  //暴露在外
  window.Drag = Drag;
})();


new Drag("box");

這段代碼是比較好理解的,在一開(kāi)始看波大神的代碼時(shí),對(duì)于translate的運(yùn)用實(shí)際上我沒(méi)看太明白,因?yàn)闆](méi)想到為啥要用到正則......

雖然比較簡(jiǎn)單,但是我們還是要分析一下這段代碼的原理:

1.自執(zhí)行函數(shù)里有一個(gè)構(gòu)造函數(shù)Drag(),在構(gòu)造函數(shù)里我們?cè)O(shè)置的方法和屬性時(shí)每一個(gè)構(gòu)造函數(shù)實(shí)例獨(dú)有的,比如他們的位置信息等。而在原型里的有三個(gè)方法:分別是獲取元素位置信息的getPosition()、設(shè)置元素位置的setPosition()和綁定事件的setDrag(),這三個(gè)因?yàn)槭枪玫模瑸榱斯?jié)省資源,我們就放在原型里了。

2.這段代碼執(zhí)行的原理是:當(dāng)鼠標(biāo)按下時(shí),獲取元素初始位置信息sourceX/Y、鼠標(biāo)初始位置信息startX/Y;當(dāng)鼠標(biāo)移動(dòng)完成時(shí),獲取鼠標(biāo)新的位置currentX/Y,兩個(gè)鼠標(biāo)位置相減就能得到鼠標(biāo)移動(dòng)的距離distanceX/Y,這同時(shí)也是元素移動(dòng)的距離,然后,我們把這個(gè)值賦給元素的style.left/top。元素的拖拽就實(shí)現(xiàn)了。

transform和style的結(jié)合
由于技術(shù)的發(fā)展,越來(lái)越多的設(shè)備開(kāi)始支持CSS3了,加上style的資源占用的更多,效率方面存在問(wèn)題,所以我們考慮使用transform。
瀏覽器的兼容寫法

我們首先在函數(shù)Drag()前加上私有屬性:

var transform = getTransform();

在下面再加上私有方法:

function getTransform() {
    var transform = "",
      divStyle = document.createElement("div").style,
      transformArr = ["transform", "webkitTransform", "MozTransform", "msTransform", "oTransform"],
      i = 0,
      l = transformArr.length;

    for (; i < l; i++) {
      if (transformArr[i] in divStyle) {
        return transform = transformArr[i];
      }
    }
    return transform;
  }
PS:記住createElement()這個(gè)方法,下次判斷瀏覽器兼容時(shí)用得著!

我們還需要在getPosition()的下面加上一個(gè)函數(shù),用同樣的形式:

getTranslate: function() {
      var val = {};
      var transformValue = document.defaultView.getComputedStyle(this.elem, false)[transform];
      if(transformValue=="none"){
        val={x:0,y:0};
      }else{
        var transformArr = transformValue.match(/-?d+/g);
        val = {
          x: Number(transformArr[4]),
          y: Number(transformArr[5])
        };
      }


      return val;
    },
PS:之所以要判斷transformValue是否為none,是因?yàn)樵诔跏蓟癄顟B(tài)是,元素未被設(shè)置transform屬性,這樣正則之后的數(shù)組是找不到[4][5] 的,所以我們讓val的兩個(gè)屬性為0,也就是稍后會(huì)成為的transform的translateX和translateY的值。

繼續(xù)寫代碼。上面一段我們用來(lái)提取translate的X、Y值??聪旅嬉欢危?/p>

getPosition: function() {
      var that = this;
      var pos = {};
      if(transform){
        var val=this.getTranslate();
        pos={
          x:val.x,
          y:val.y
        };
      }else{
        pos = {
          x: that.elem.offsetLeft,
          y: that.elem.offsetTop
        };
      }
      return pos;
    },
注意上面一段代碼我們修改的的內(nèi)容,在這里我們?cè)黾恿艘粋€(gè)判斷:即當(dāng)支持transform屬性的瀏覽器存在時(shí),我們會(huì)用transform屬性修改元素的值,把之前在getTranslate中得到的x、y賦值給pos的x、y。

在上面一段代碼中,我們會(huì)根據(jù)瀏覽器的情況,用不同的方法取到相同的值,val的值來(lái)自getTranslate(),是我們從元素的transform中提取出來(lái)的。同樣,在下面的setPosition()中,我們也要設(shè)置if判斷。

setPosition: function(pos) {
      if (transform) {
        this.elem.style[transform] = "translate(" + pos.x + "px" + "," + pos.y + "px)";
      } else {
        this.elem.style.left = pos.x + "px";
        this.elem.style.top = pos.y + "px";
      }
    },

這一段沒(méi)什么好講的,就是用不同的形式賦值而已。

到這里,這個(gè)模塊就封裝完畢了。接下來(lái)讓我們看看完整代碼:

;
(function() {
  //私有屬性
  var transform = getTransform();
  //構(gòu)造函數(shù),屬于每一個(gè)實(shí)例
  function Drag(selector) {
    this.elem = typeof selector == "object" ? selector : document.getElementById(selector);
    //鼠標(biāo)初始位置
    this.startX = 0;
    this.startY = 0;
    //元素初始位置
    this.sourceX = 0;
    this.sourceY = 0;

    this.init();
  }

  //原型,共有的
  Drag.prototype = {
    constructor: Drag,
    init: function() {
      this.setDrag();
    },

    //用于獲取元素當(dāng)前的位置信息
    getPosition: function() {
      var that = this;
      var pos = {};
      if(transform){
        var val=this.getTranslate();
        pos={
          x:val.x,
          y:val.y
        };
      }else{
        pos = {
          x: that.elem.offsetLeft,
          y: that.elem.offsetTop
        };
      }
      return pos;
    },

    //獲取translate值
    getTranslate: function() {
      var val = {};
      var transformValue = document.defaultView.getComputedStyle(this.elem, false)[transform];
      if(transformValue=="none"){
        val={x:0,y:0};
      }else{
        var transformArr = transformValue.match(/-?d+/g);
        val = {
          x: Number(transformArr[4]),
          y: Number(transformArr[5])
        };
      }


      return val;
    },
    //用來(lái)設(shè)置當(dāng)前元素的位置
    setPosition: function(pos) {
      if (transform) {
        this.elem.style[transform] = "translate(" + pos.x + "px" + "," + pos.y + "px)";
      } else {
        this.elem.style.left = pos.x + "px";
        this.elem.style.top = pos.y + "px";
      }
    },

    //該方法用來(lái)綁定事件
    setDrag: function() {
      var self = this;
      this.elem.addEventListener("mousedown", start, false);

      function start(event) {

        self.startX = event.pageX;
        self.startY = event.pageY;

        var pos = self.getPosition();

        self.sourceX = pos.x;
        self.sourceY = pos.y;

        document.addEventListener("mousemove", move, false);
        document.addEventListener("mouseup", end, false);
      }

      function move(event) {
        //總體思想:鼠標(biāo)距瀏覽器距-鼠標(biāo)距元素距離
        var currentX = event.pageX; //當(dāng)前的鼠標(biāo)x位置
        var currentY = event.pageY; //當(dāng)前的鼠標(biāo)y位置

        var distanceX = currentX - self.startX; //鼠標(biāo)移動(dòng)的距離x
        var distanceY = currentY - self.startY; //鼠標(biāo)移動(dòng)的距離y

        self.setPosition({
          x: self.sourceX + distanceX,
          y: self.sourceY + distanceY
        });

      }

      function end(event) {
        document.removeEventListener("mousemove", move);
        document.removeEventListener("mouseup", end);
      }
    }
  };
  //私有方法,用來(lái)獲取transform的兼容寫法
  function getTransform() {
    var transform = "",
      divStyle = document.createElement("div").style,
      transformArr = ["transform", "webkitTransform", "MozTransform", "msTransform", "oTransform"],
      i = 0,
      l = transformArr.length;

    for (; i < l; i++) {
      if (transformArr[i] in divStyle) {
        return transform = transformArr[i];
      }
    }
    return transform;
  }
  //暴露在外
  window.Drag = Drag;
})();


new Drag("box");

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/107087.html

相關(guān)文章

  • 有贊移動(dòng) iOS 組件化(模塊化)架構(gòu)設(shè)計(jì)實(shí)踐

    摘要:一背景業(yè)務(wù)組件化或者叫模塊化作為移動(dòng)端應(yīng)用架構(gòu)的主流方式之一,近年來(lái)一直是業(yè)界積極探索和實(shí)踐的方向。有贊移動(dòng)團(tuán)隊(duì)自年起也在不斷嘗試各種組件化方案,在有贊微商城,有贊零售,有贊美業(yè)等多個(gè)應(yīng)用中進(jìn)行了實(shí)踐。相比組件,個(gè)人感覺(jué)稱之為模塊更為合適。 一、背景 業(yè)務(wù)組件化(或者叫模塊化)作為移動(dòng)端應(yīng)用架構(gòu)的主流方式之一,近年來(lái)一直是業(yè)界積極探索和實(shí)踐的方向。有贊移動(dòng)團(tuán)隊(duì)自16年起也在不斷嘗試各種...

    Thanatos 評(píng)論0 收藏0
  • 前端基礎(chǔ)進(jìn)階(十):面向?qū)ο髮?shí)戰(zhàn)之封裝拖拽對(duì)象

    摘要:前面幾篇文章,我跟大家分享了的一些基礎(chǔ)知識(shí),這篇文章,將會(huì)進(jìn)入第一個(gè)實(shí)戰(zhàn)環(huán)節(jié)利用前面幾章的所涉及到的知識(shí),封裝一個(gè)拖拽對(duì)象。不封裝對(duì)象直接實(shí)現(xiàn)利用原生封裝拖拽對(duì)象通過(guò)擴(kuò)展來(lái)實(shí)現(xiàn)拖拽對(duì)象。 showImg(https://segmentfault.com/img/remote/1460000008699587); 前面幾篇文章,我跟大家分享了JavaScript的一些基礎(chǔ)知識(shí),這篇文章,...

    Eidesen 評(píng)論0 收藏0
  • 對(duì)于組件化自己理解

    摘要:也是的獨(dú)特性,使得組件化成為了可能。簡(jiǎn)單的說(shuō)就是組件內(nèi)部標(biāo)簽對(duì)內(nèi)語(yǔ)義化,組件自定義標(biāo)簽對(duì)外語(yǔ)義化對(duì)內(nèi)語(yǔ)義化保存自定義標(biāo)簽具有正確的語(yǔ)義,自定義標(biāo)簽對(duì)外語(yǔ)義是對(duì)內(nèi)部標(biāo)簽組合出的功能概括。 組件化 這里首先介紹WebComponents標(biāo)準(zhǔn),以下為騰訊alloyteam團(tuán)隊(duì)的一篇文章里的內(nèi)容。 模板能力,WebComponent提供原生的模板能力 ShadowDOm封裝組件獨(dú)立的內(nèi)部結(jié)構(gòu)...

    jerry 評(píng)論0 收藏0
  • 使用 Resilience4j 框架實(shí)現(xiàn)重試機(jī)制

    摘要:重試會(huì)增加的響應(yīng)時(shí)間。提供了輔助方法來(lái)為包含遠(yuǎn)程調(diào)用的函數(shù)式接口或表達(dá)式創(chuàng)建裝飾器。如果我們想創(chuàng)建一個(gè)裝飾器并在代碼庫(kù)的不同位置重用它,我們將使用。 在本文中,我們將從快速介紹 Resilience4j 開(kāi)始,然后深入探討其 Retry 模塊。我們將了解何時(shí)、如何使用它,以及它提供的功能。在此過(guò)程中,我們還將學(xué)...

    番茄西紅柿 評(píng)論0 收藏2637
  • 我是如何學(xué)習(xí)游戲引擎?

    摘要:下面列舉了游戲開(kāi)發(fā)中常見(jiàn)的崗位以及兩條常見(jiàn)的協(xié)作開(kāi)發(fā)的流水線其實(shí)學(xué)習(xí)游戲引擎,前期對(duì)于任何崗位來(lái)說(shuō)路線都是相似的,基本上就是一個(gè)熟悉基本操作理解基本概念拓展專業(yè)知識(shí)的過(guò)程。當(dāng)然這不是絕對(duì)的,任何引擎的開(kāi)始階段和大成階段都是相似的。 這是【游戲開(kāi)發(fā)那些事】第51篇原創(chuàng) 前言:游戲引擎,表面...

    未東興 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<