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

資訊專欄INFORMATION COLUMN

手把手教你寫vue裁切預(yù)覽組件

FreeZinG / 671人閱讀

摘要:版本裁切工具,包含預(yù)覽功能最終效果源碼地址第一步先用安裝腳手架不會安裝的看官網(wǎng)初始化第二步創(chuàng)建文件新建里新建,在配置訪問路由具體看源碼最終生成的文件結(jié)構(gòu)如下圖第三步注冊組件引用所有插件導(dǎo)入插件入口文件如果已安裝就跳過注冊插件全

vue版本裁切工具,包含預(yù)覽功能

最終效果: https://qiuyaofan.github.io/vue-crop-demo/

源碼地址: https://github.com/qiuyaofan/vue-crop

第一步:先用vue-cli安裝腳手架(不會安裝的看 vue-cli官網(wǎng))
// 初始化vue-cli
vue init webpack my-plugin
第二步:創(chuàng)建文件
新建src/views/validSlideDemo.vue,

src/components里新建VueCrop/index.js,VueCrop.vue,

在routes/index.js配置訪問路由(具體看github源碼)

最終生成的文件結(jié)構(gòu)如下圖:

第三步:注冊組件
1.引用所有插件:src/components/index.js
// 導(dǎo)入插件入口文件
import VueCrop from "./VueCrop/index.js"
const install = function (Vue, opts = {}) {
  /* 如果已安裝就跳過 */
  if (install.installed) return
  
  // 注冊插件
  Vue.component(VueCrop.name, VueCrop)
}

// 全局情況下注冊插件
if (typeof window !== "undefined" && window.Vue) {
  install(window.Vue)
}

export {
  install,
  // 此處是為了兼容在vue內(nèi)多帶帶引入這個插件,如果是main.js全局引入就可以去掉
  VueCrop
}
2.全局調(diào)用插件:src/main.js ( vue plugins官方文檔解說install)
import Vue from "vue"
import App from "./App"
import router from "./router"

// 新加的:導(dǎo)入入口文件
import { install } from "src/components/index.js"

// 全局調(diào)用,相當(dāng)于調(diào)用 `MyPlugin.install(Vue)`
Vue.use(install)

Vue.config.productionTip = false
/* eslint-disable no-new */
new Vue({
  el: "#app",
  router,
  components: { App },
  template: ""
})
3.VueCrop入口文件調(diào)用VueCrop.vue:src/components/VueCrop/index.js
// 導(dǎo)入vue
import VueCrop from "./VueCrop.vue"

// Vue.js 的插件應(yīng)當(dāng)有一個公開方法 install 。這個方法的第一個參數(shù)是 Vue 構(gòu)造器
VueCrop.install = function (Vue) {
  // 注冊組件
  Vue.component(VueCrop.name, VueCrop)
}

export default VueCrop
小結(jié):我一開始一直有個誤解,以為myPlugin.install是vue的一個方法,其實不是,他只是我們構(gòu)造plugin識的一個公開方法,可以理解為原生js中的構(gòu)造函數(shù)的方法:
function MyPlugin(){
  console.info("構(gòu)造函數(shù)")
}
MyPlugin.prototype.install=function(vue,options){
    console.info("構(gòu)造器vue:"+vue);
}

而真正注冊組件的是:Vue.component()

所以,vue插件注冊的過程是:

1.調(diào)用main.js中:
import { install } from "src/components/index.js"
vue.use(install)

2.index.js添加install方法,調(diào)用Vue.component注冊組件

3.組件內(nèi)的index.js同所有組件的index.js一樣
第四步:設(shè)計開發(fā)自己的組件,構(gòu)建組件結(jié)構(gòu)
在此之前,可以先了解下組件的命名規(guī)范等,可參考文章 掘金:Vue前端開發(fā)規(guī)范,其中第2點有詳細(xì)講解

首先,確定自己的調(diào)用方式和需要暴露的參數(shù)


>

其中,@afterCrop="afterCrop"是裁切完成的回調(diào)函數(shù),其他是屬性配置

在組件src/components/VueCrop/VueCrop.vue內(nèi),可以用this.$emit("afterCrop")觸發(fā)demo里的afterCrop事件

組件結(jié)構(gòu)上,主要分為:裁切主體部分(VueCrop.vue),選框組件(VueCropTool.vue),裁切框?qū)挾?、位置坐?biāo)等計算(VueCropMove.js),拖拽事件注冊公共js(components/utils/draggable.js)

當(dāng)前裁切插件的總體思路

裁切插件的裁切主體由圖片,選框,預(yù)覽結(jié)構(gòu)組成

選框(VueCropTool.vue)負(fù)責(zé)拖拽改變其大小,坐標(biāo)位置等并返回給VueCrop.vue

主體計算數(shù)值同步預(yù)覽顯示(c-crop--preview)

主體觸發(fā)調(diào)用頁面(VueCropDemo.vue)的afterCrop事件,從而傳遞參數(shù)返回裁切后的url,left,top,bottom,right,x,y,w,h等

備注:此組件不具備真實的裁切功能,最終的裁切是傳遞給后臺去裁,你如果想擴展可以在afterCrop函數(shù)里根據(jù)坐標(biāo)等信息進行處理

接下來我們對各個組件和js進行講解

1.draggable.js是參照element里的,修改了一部分,源碼如下
export default function (element, options) {
  const moveFn = function (event) {
    if (options.drag) {
      options.drag(event)
    }
  }
  // mousedown fn
  const downFn = function (event) {
    if (options.start) {
        // 調(diào)用參數(shù)中start函數(shù)
      options.start(event)
    }
  }
  // mouseup fn
  const upFn = function (event) {
    document.removeEventListener("mousemove", moveFn)
    document.removeEventListener("mouseup", upFn)
    document.onselectstart = null
    document.ondragstart = null

    if (options.end) {
        // 調(diào)用參數(shù)中end函數(shù)
      options.end(event)
    }
  }
  // 綁定事件
  element.addEventListener("mousedown", event => {
    if (options.stop && options.stop(event, element) === false) {
      return false
    }
    document.onselectstart = function () {
      return false
    }
    document.ondragstart = function () {
      return false
    }
    document.addEventListener("mousedown", downFn)
    document.addEventListener("mousemove", moveFn)
    document.addEventListener("mouseup", upFn)
  })
}
VueCropTool.vue使用如下
draggable(this.$el.querySelector(".c-crop--drap_screen"), {
    start: (event) => {
      this.startPos = [event.x, event.y]
    },
    drag: (event) => {
      this.handleDragLocation(event)
    },
    end: (event) => {
      this.handleDragLocation(event)
    }
})
2.裁切主體部分(VueCrop.vue全部源碼鏈接)
//script部分


3.裁切框部分(VueCropTool.vue全部源碼鏈接)
//script部分

4.計算裁切框的js(VueCropMove.js全部源碼鏈接)
// 12種形態(tài),四條邊,邊的中點,邊的四個角。e:東,w:西,n:北,s:南,ne:東南以此類推
const movePos = {
  0: e,
  4: e,
  1: s,
  5: s,
  2: w,
  6: w,
  3: n,
  7: n,
  8: ne,
  9: se,
  10: sw,
  11: nw
}
let width, height, result, ratio

// 獲取某種形態(tài)類型的寬或高最大值
function getMaxSize (json, startJson, dire, type) {
  if (type === "w") {
    switch (dire) {
      case "e":
      case "s":
      case "n":
      case "ne":
      case "se":
        return json.screen.right - json.l
      case "w":
      case "nw":
      case "sw":
        return startJson.r - json.screen.left
    }
  } else if (type === "h") {
    switch (dire) {
      case "n":
      case "nw":
      case "ne":
        return startJson.b - json.screen.top
      case "s":
      case "w":
      case "e":
      case "sw":
      case "se":
        return json.screen.bottom - startJson.t
    }
  }
}
// 判斷是否有ratio,返回修改后的尺寸
function setRatioSize (type, json, startJson, ratio, width, height) {
  if (ratio) {
    if (width / ratio >= height) {
      var maxHeight = getMaxSize(json, startJson, type, "h")
      height = width / ratio
      if (height > maxHeight) {
        height = maxHeight
        width = height * ratio
      }
    } else {
      var maxWidth = getMaxSize(json, startJson, type, "w")
      width = height * ratio
      if (width > maxWidth) {
        width = maxWidth
        height = width / ratio
      }
    }
  }
  return {
    width: width,
    height: height
  }
}
// 拖拽東邊,高度是不變的,除非有比例拖拽時
function e (_this, json, startJson) {
  ratio = _this.cropJson.r
  width = range(getWidth(json, startJson, "e"), getMaxSize(json, startJson, "e", "w"))
  if (ratio) {
      // 有比例時,計算高度,并對比最大值是否超出
    height = range(width / ratio, getMaxSize(json, startJson, "e", "h"))
    result = setRatioSize("e", json, startJson, ratio, width, height)
    setSize(_this, result)
  } else {
    _this.width = width
  }
  return _this
}

// 拖拽南邊,寬度是不變的,除非有比例拖拽時
function s (_this, json, startJson) {
  ratio = _this.cropJson.r
  height = range(getHeight(json, startJson, "s"), getMaxSize(json, startJson, "s", "h"))
  if (ratio) {
    // 有比例時,計算寬度,并對比最大值是否超出
    width = range(height * ratio, getMaxSize(json, startJson, "s", "w"))
    result = setRatioSize("s", json, startJson, ratio, width, height)
    setSize(_this, result)
  } else {
    _this.height = height
  }

  return _this
}

// 以下同上,以此類推
function w (_this, json, startJson) {
  ratio = _this.cropJson.r
  width = range(getWidth(json, startJson, "w"), getMaxSize(json, startJson, "w", "w"))
  if (ratio) {
    height = range(width / ratio, getMaxSize(json, startJson, "w", "h"))
    result = setRatioSize("w", json, startJson, ratio, width, height)
    setSize(_this, result)
    _this.left = getLeft(_this, json, startJson)
  } else {
    _this.width = width
    _this.left = rangeMax(json.x - json.screen.left, startJson.r)
  }
  return _this
}
function n (_this, json, startJson) {
  ratio = _this.cropJson.r
  height = range(getHeight(json, startJson, "n"), getMaxSize(json, startJson, "n", "h"))
  if (ratio) {
    width = range(height * ratio, getMaxSize(json, startJson, "n", "w"))
    result = setRatioSize("n", json, startJson, ratio, width, height)
    setSize(_this, result)
    _this.top = getTop(_this, json, startJson)
  } else {
    _this.height = height
    _this.top = rangeMax(json.y - json.screen.top, startJson.b)
  }
  return _this
}

function ne (_this, json, startJson) {
  height = range(getHeight(json, startJson, "n"), getMaxSize(json, startJson, "ne", "h"))
  width = range(getWidth(json, startJson, "e"), getMaxSize(json, startJson, "ne", "w"))
  result = setRatioSize("ne", json, startJson, _this.cropJson.r, width, height)
  setSize(_this, result)
  _this.top = getTop(_this, json, startJson)
  return _this
}
function se (_this, json, startJson) {
  height = range(getHeight(json, startJson, "s"), getMaxSize(json, startJson, "se", "h"))
  width = range(getWidth(json, startJson, "e"), getMaxSize(json, startJson, "se", "w"))
  result = setRatioSize("se", json, startJson, _this.cropJson.r, width, height)
  setSize(_this, result)
  return _this
}
function sw (_this, json, startJson) {
  width = range(getWidth(json, startJson, "w"), getMaxSize(json, startJson, "sw", "w"))
  height = range(getHeight(json, startJson, "s"), getMaxSize(json, startJson, "sw", "h"))
  result = setRatioSize("sw", json, startJson, _this.cropJson.r, width, height)
  setSize(_this, result)
  _this.left = getLeft(_this, json, startJson)
  return _this
}
function nw (_this, json, startJson) {
  width = range(getWidth(json, startJson, "w"), getMaxSize(json, startJson, "nw", "w"))
  height = range(getHeight(json, startJson, "n"), getMaxSize(json, startJson, "nw", "h"))
  result = setRatioSize("nw", json, startJson, _this.cropJson.r, width, height)
  setSize(_this, result)
  _this.left = getLeft(_this, json, startJson)
  _this.top = getTop(_this, json, startJson)
  return _this
}

// 匹配范圍
function range (value, max) {
  value = value > max ? max : value
  return value < 20 ? 20 : value
}
// 最大值
function rangeMax (value, max) {
  return value > max ? max : value
}
// top
function getTop (_this, json, startJson) {
  return rangeMax(startJson.b - _this.height - json.screen.top, startJson.b)
}
// left
function getLeft (_this, json, startJson) {
  return rangeMax(startJson.r - _this.width - json.screen.left, startJson.r)
}
// height:只存在于s||n類型
function getHeight (json, startJson, type) {
  return type === "n" ? startJson.b - json.y : json.y - startJson.t
}
// width:只存在于w||e類型
function getWidth (json, startJson, type) {
  return type === "w" ? startJson.r - json.x : json.x - startJson.l
}
// setSize
function setSize (_this, result) {
  _this.width = result.width
  _this.height = result.height
}

export default movePos

今天就分享到這里啦~喜歡這個插件可以去 github star~

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

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

相關(guān)文章

  • 把手你寫 Vue UI 組件庫@vue2.0

    摘要:手把手教你寫組件庫最近在研究的實現(xiàn),發(fā)現(xiàn)網(wǎng)上很少有關(guān)于插件具體實現(xiàn)的文章,官方的文檔也只是一筆帶過,對于新手來說并不算友好。 手把手教你寫 Vue UI 組件庫 最近在研究 muse-ui 的實現(xiàn),發(fā)現(xiàn)網(wǎng)上很少有關(guān)于 vue 插件具體實現(xiàn)的文章,官方的文檔也只是一筆帶過,對于新手來說并不算友好。 筆者結(jié)合官方文檔,與自己的摸索總結(jié),以最簡單的 FlexBox 組件為例子,帶大家入門 v...

    Keagan 評論0 收藏0
  • 把手教你用jsx封裝Vue中的復(fù)雜組件(網(wǎng)易云音樂實戰(zhàn)項目需求)

    摘要:終極解決方案所以我們要統(tǒng)一環(huán)境,直接使用渲染我們的組件,文檔可以參照音樂標(biāo)題歌手專輯時長省去一些細(xì)節(jié)注意需要放在中,的透傳也不要忘了,這樣我們在外部想使用的一些屬性和事件才比較方便。 背景介紹 最近在做vue高仿網(wǎng)易云音樂的項目,在做的過程中發(fā)現(xiàn)音樂表格這個組件會被非常多的地方復(fù)用,而且需求比較復(fù)雜的和靈活。 預(yù)覽地址 源碼地址 圖片預(yù)覽 歌單詳情 showImg(https://se...

    HitenDev 評論0 收藏0
  • 把手教你擼個vue2.0彈窗組件

    摘要:組件結(jié)構(gòu)同組件結(jié)構(gòu)通過方法獲取元素的大小及其相對于視口的位置,之后對提示信息進行定位??梢杂脕磉M行一些復(fù)雜帶校驗的彈窗信息展示,也可以只用于簡單信息的展示??梢酝ㄟ^屬性來顯示任意標(biāo)題,通過屬性來修改顯示區(qū)域的寬度。 手把手教你擼個vue2.0彈窗組件 在開始之前需要了解一下開發(fā)vue插件的前置知識,推薦先看一下vue官網(wǎng)的插件介紹 預(yù)覽地址 http://haogewudi.me/k...

    mrli2016 評論0 收藏0

發(fā)表評論

0條評論

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