ホームページ >ウェブフロントエンド >Vue.js >Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

青灯夜游
青灯夜游転載
2021-07-08 10:56:192022ブラウズ

Vuejs で、基になる DOM を操作するようにいくつかの命令をカスタマイズします。次の記事では、開発効率を向上させるために直接使用できる、一般的に使用されるいくつかのカスタム命令コードを要約して共有します。

Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

[関連する推奨事項: 「vue.js チュートリアル 」]

1. 要素のクリック範囲拡張コマンド v- ExpandClick

このディレクティブを使用すると、要素のクリック範囲を暗黙的に拡張できますが、疑似要素を借用して実装されているため、ページ上の要素の配置やレイアウトには影響しません。

渡すことができるパラメータは次のとおりです: 上、右、下、左の拡張範囲、単位 px、デフォルトの拡張は 10 ピクセルです。ディレクティブのコードは次のとおりです:

export default function (el, binding) {
    const s = document.styleSheets[document.styleSheets.length - 1]
    const DEFAULT = -10 // 默认向外扩展10px
    const ruleStr = `content:"";position:absolute;top:-${top || DEFAULT}px;bottom:-${bottom || DEFAULT}px;right:-${right || DEFAULT}px;left:-${left || DEFAULT}px;`
    const [top, right, bottom, left] = binding.expression && binding.expression.split(',') || []
    const classNameList = el.className.split(' ')
    el.className = classNameList.includes('expand_click_range') ? classNameList.join(' ') : [...classNameList, 'expand_click_range'].join(' ')
    el.style.position = el.style.position || "relative"
    if (s.insertRule) {
        s.insertRule('.expand_click_range::before' + '{' + ruleStr + '}', s.cssRules.length)
    } else { /* IE */
        s.addRule('.expand_click_range::before', ruleStr, -1)
    }
}

パラメータ属性:

Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

これにより、テンプレート内の任意の要素で新しい v-expandClick プロパティを使用できるようになります。

<div v-expandClick="20,30,40,50" @click="glabClickoutside"> 点击范围扩大</div>

2. テキスト コンテンツのコピー コマンド v-copy

このコマンドを使用して、要素のテキスト コンテンツをコピーします (コマンドはシングルクリックをサポートしています) copy v-copy およびダブルクリック copy v-copy.dblclick、アイコンをクリックして v-copy.icon をコピー (3 つのモード)、パラメーターが渡されない場合、デフォルトで click copy が使用されます。

ディレクティブのコードは次のとおりです:

export default {
  bind (el, binding) {
    // 双击触发复制
    if (binding.modifiers.dblclick) {
      el.addEventListener(&#39;dblclick&#39;, () => handleClick(el.innerText))
      el.style.cursor = &#39;copy&#39;
    }
    // 点击icon触发复制
    else if (binding.modifiers.icon) {
      if (el.hasIcon) return
      const iconElement = document.createElement(&#39;i&#39;)
      iconElement.setAttribute(&#39;class&#39;, &#39;el-icon-document-copy&#39;)
      iconElement.setAttribute(&#39;style&#39;, &#39;margin-left:5px&#39;)
      el.appendChild(iconElement)
      el.hasIcon = true
      iconElement.addEventListener(&#39;click&#39;, () => handleClick(el.innerText))
      iconElement.style.cursor = &#39;copy&#39;
    }
    // 单击触发复制
    else {
      el.addEventListener(&#39;click&#39;, () => handleClick(el.innerText))
      el.style.cursor = &#39;copy&#39;
    }
  }
}

function handleClick (text) {
  // 创建元素
  if (!document.getElementById(&#39;copyTarget&#39;)) {
    const copyTarget = document.createElement(&#39;input&#39;)
    copyTarget.setAttribute(&#39;style&#39;, &#39;position:fixed;top:0;left:0;opacity:0;z-index:-1000;&#39;)
    copyTarget.setAttribute(&#39;id&#39;, &#39;copyTarget&#39;)
    document.body.appendChild(copyTarget)
  }

  // 复制内容
  const input = document.getElementById(&#39;copyTarget&#39;)
  input.value = text
  input.select()
  document.execCommand(&#39;copy&#39;)
  // alert(&#39;复制成功&#39;)
}

パラメータ属性:

Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

その後、任意の項目に対して new を使用できます。テンプレート内の要素 v-copy プロパティは次のとおりです:

<div v-copy> 单击复制 </div>
<div v-copy.dblclick> 双击复制 </div>
<div v-copy.icon> icon复制 </div>

3. 要素の全画面コマンド v-screenfull
全画面コマンド、要素をクリックして全画面表示を実行します/全画面操作を終了します。 element-ui の全画面アイコン el-icon-full-screen を要素の後に挿入するかどうかをサポートします。

ディレクティブのコードは次のとおりです:

import screenfull from &#39;screenfull&#39;

export default {
  bind (el, binding) {
    if (binding.modifiers.icon) {
      if (el.hasIcon) return
      // 创建全屏图标
      const iconElement = document.createElement(&#39;i&#39;)
      iconElement.setAttribute(&#39;class&#39;, &#39;el-icon-full-screen&#39;)
      iconElement.setAttribute(&#39;style&#39;, &#39;margin-left:5px&#39;)
      el.appendChild(iconElement)
      el.hasIcon = true
  }
    el.style.cursor = el.style.cursor || &#39;pointer&#39;
    // 监听点击全屏事件
    el.addEventListener(&#39;click&#39;, () => handleClick())
  }
}

function handleClick () {
  if (!screenfull.isEnabled) {
    alert(&#39;浏览器不支持全屏&#39;)
    return
  }
  screenfull.toggle()
}

パラメータ属性:

Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

その後、任意の項目に対して new を使用できます。

<div v-screenfull.icon> 全屏 </div>

4. 要素の説明の指示 v-tooltip

element- と同様に、要素に説明を追加します。 ui の el-tooltip (疑問符アイコンをマウスでオーバーレイすると、説明テキストが表示されます)。

Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

ディレクティブのコードは次のとおりです:

import Vue from &#39;vue&#39;
export default function (el, binding) {
    if (el.hasIcon) return
    const iconElement = structureIcon(binding.arg, binding.value)
    el.appendChild(iconElement)
    el.hasIcon = true
}

function structureIcon (content, attrs) {
    // 拼接绑定属性
    let attrStr = &#39;&#39;
    for (let key in attrs) {
        attrStr += `${key}=${attrs[key]} `
    }
    const a = `<el-tooltip content=${content} ${attrStr}><i class="el-icon-question" style="margin:0 10px"></i></el-tooltip>`
    // 创建构造器
    const tooltip = Vue.extend({
        template: a
    })
    // 创建一个 tooltip 实例并返回 dom 节点
    const component = new tooltip().$mount()
    return component.$el
}

パラメータ属性:

Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

Then次のように、コマンド内の任意の要素で新しい v-tooltip プロパティを使用できます:

<div v-tooltip:content=&#39;tootipParams&#39;> 提示 </div>

例:

<div v-tooltip:提示内容为XXX1> 提示1</div>
<div v-tooltip:提示内容为XXX=&#39;tootipParams&#39;> 提示2 </div>

コマンドの element-ui でサポートされているパラメーターを渡します:

data() {
    return {
        tootipParams: {
            placement: &#39;top&#39;,
            effect: &#39;light&#39;,
        }
    }
}

5. テキストが省略コマンドを超えています v-ellipsis

テキストの内容が幅 (デフォルト 100 ピクセル) を超える場合、このコマンドを使用してテキストの内容を省略形式に自動的に変更します。 CSS を使用するのと同等:

width: 100px;
whiteSpace: nowrap
overflow: hidden;
textOverflow: ellipsis;

命令を使用した効果:

Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

命令のコードは次のとおりです:

export default function (el, binding) {
    el.style.width = binding.arg || 100 + &#39;px&#39;
    el.style.whiteSpace = &#39;nowrap&#39;
    el.style.overflow = &#39;hidden&#39;;
    el.style.textOverflow = &#39;ellipsis&#39;;
}

パラメーター属性:

Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

次に、次のように、テンプレート内の任意の要素で新しい v-ellipsis プロパティを使用できます:

<div v-ellipsis:100> 需要省略的文字是阿萨的副本阿萨的副本阿萨的副本阿萨的副本</div>

6。先頭に戻るコマンド v-backtop
このコマンドを使用して、ページまたは指定した要素を先頭に戻します。

オプションで指定された要素。指定されていない場合は、グローバル ページの先頭に戻ります。オプションで、要素がオフセットされた後に戻る要素を何ピクセル表示するか、たとえば 400 ピクセルスクロールした後に「トップに戻る」ボタンを表示します。

Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

ディレクティブのコードは次のとおりです:

export default {
  bind (el, binding, vnode) {
    // 响应点击后滚动到元素顶部
    el.addEventListener(&#39;click&#39;, () => {
    const target = binding.arg ? document.getElementById(binding.arg) : window
    target.scrollTo({
      top: 0,
      behavior: &#39;smooth&#39;
      })
    })
  },
  update (el, binding, vnode) {
    // 滚动到达参数值才出现绑定指令的元素
    const target = binding.arg ? document.getElementById(binding.arg) : window
    if (binding.value) {
      target.addEventListener(&#39;scroll&#39;, (e) => {
        if (e.srcElement.scrollTop > binding.value) {
          el.style.visibility = &#39;unset&#39;
        } else {
          el.style.visibility = &#39;hidden&#39;
        }
      })
    }
    // 判断初始化状态
    if (target.scrollTop < binding.value) {
      el.style.visibility = &#39;hidden&#39;
    }
  },
  unbind (el) {
    const target = binding.arg ? document.getElementById(binding.arg) : window
    target.removeEventListener(&#39;scroll&#39;)
    el.removeEventListener(&#39;click&#39;)
  }
}

パラメータ属性:

Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

Then次のように、任意の要素で新しい v-backtop プロパティを使用して、ID が app の要素が 400 ピクセルスクロールされた後にバインディング命令要素を表示できます。

<div  v-backtop:app="400"> 回到顶部 </div>

は、次のように使用することもできます。バインディング命令は常に表示されます。要素であり、グローバル ページです。先頭に戻る:

<div  v-backtop> 回到顶部 </div>

7. 空状態コマンド v-empty

このコマンドを使用して、デフォルトの空の状態を表示します。デフォルトの画像 (オプション、デフォルトでは画像なし)、デフォルトのテキスト コンテンツ (オプション、デフォルトではデータなし)、および空の状態を示すかどうか (必須) を渡すことができます。

Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

命令のコードは次のとおりです:

import Vue from "vue";
export default {
  update (el, binding, vnode) {
    el.style.position = el.style.position || &#39;relative&#39;
    const { offsetHeight, offsetWidth } = el
    const { visible, content, img } = binding.value
    const image = img ? `<img  src="${img}"    style="max-width:90%"  style="max-width:90%" alt="Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)" ></img>` : &#39;&#39;
    const defaultStyle = "position:absolute;top:0;left:0;z-index:9999;background:#fff;display:flex;justify-content: center;align-items: center;"
    const empty = Vue.extend({
    template: `<div style="height:${offsetHeight}px;width:${offsetWidth}px;${defaultStyle}">
      <div style="text-align:center">
        <div>${image}</div>
        <div>${content || &#39;暂无数据&#39;}</div>
      </div>
    </div>`
    })
    const component = new empty().$mount().$el
    if (visible) {
      el.appendChild(component)
    } else {
      el.removeChild(el.lastChild)
    }
  },
}

パラメータ属性:

1Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

その後、次のようにオブジェクト emptyValue を渡して、テンプレート内の任意の要素で新しい v-empty プロパティを使用できます。

<div style="height:500px;width:500px" v-empty="emptyValue"> 原本内容

需要传入一个参数对象,例如显示文字为:暂无列表,图片路径为 ../../assets/images/blue_big.png,控制标示 visible:

emptyValue = {
  content: &#39;暂无列表&#39;,
  img: require(&#39;../../assets/images/blue_big.png&#39;),
  visible: true,
},

8、徽标指令 v-badge

使用该指令在元素右上角显示徽标。

支持配置徽标的背景颜色、徽标形状;支持传入徽标上显示的数字。

1Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

指令的代码如下:

import Vue from &#39;vue&#39;

const SUCCESS = &#39;#72c140&#39;
const ERROR = &#39;#ed5b56&#39;
const WARNING = &#39;#f0af41&#39;
const INFO = &#39;#4091f7&#39;
const HEIGHT = 20
let flag = false
export default {
  update (el, binding, vnode) {
    const { modifiers, value } = binding
    const modifiersKey = Object.keys(modifiers)
    let isDot = modifiersKey.includes(&#39;dot&#39;)
    let backgroundColor = &#39;&#39;
    if (modifiersKey.includes(&#39;success&#39;)) {
      backgroundColor = SUCCESS
    } else if (modifiersKey.includes(&#39;warning&#39;)) {
      backgroundColor = WARNING
    } else if (modifiersKey.includes(&#39;info&#39;)) {
      backgroundColor = INFO
    } else {
      backgroundColor = ERROR
    }

    const targetTemplate = isDot 
        ? `<div style="position:absolute;top:-5px;right:-5px;height:10px;width:10px;border-radius:50%;background:${backgroundColor}"></div>` 
        : `<div style="background:${backgroundColor};position:absolute;top:-${HEIGHT / 2}px;right:-${HEIGHT / 2}px;height:${HEIGHT}px;min-width:${HEIGHT}px;border-radius:${HEIGHT / 2}px;text-align:center;line-height:${HEIGHT}px;color:#fff;padding:0 5px;">${value}</div>`
        
    el.style.position = el.style.position || &#39;relative&#39;
    const badge = Vue.extend({
      template: targetTemplate
    })
    const component = new badge().$mount().$el
    if (flag) {
      el.removeChild(el.lastChild)
    }
    el.appendChild(component)
    flag = true
  }
}

参数 Attributes:

1Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

然后你可以在模板中任何元素上使用新的 v-badge property,如下:

<div v-badge.dot.info="badgeCount" style="height:50px;width:50px;background:#999"> </div>

9、拖拽指令 v-drag

使用该指令可以对元素进行拖拽。

指令的代码如下:

export default {
  let _el = el
  document.onselectstart = function() {
    return false  //禁止选择网页上的文字
  }
  
  _el.onmousedown = e => {
    let disX = e.clientX - _el.offsetLeft //鼠标按下,计算当前元素距离可视区的距离
    let disY = e.clientY - _el.offsetTop
    document.onmousemove = function(e){     
      let l = e.clientX - disX
      let t = e.clientY - disY;
      _el.style.left = l + "px"
      _el.style.top = t + "px"
    }
    document.onmouseup = e => {
      document.onmousemove = document.onmouseup = null
    }
    return false
  }
}

然后你可以在模板中任何元素上使用新的 v-drag property,如下:

<div v-drag> 支持拖拽的元素 </div>

10、响应缩放指令 v-resize

使用该指令可以响应元素宽高改变时执行的方法。

指令的代码如下:

export default {
  bind(el, binding) {
    let width = &#39;&#39;, height = &#39;&#39;;
    function isReize() {
      const style = document.defaultView.getComputedStyle(el);
      if (width !== style.width || height !== style.height) {
        binding.value();  // 执行传入的方法
      }
      width = style.width;
      height = style.height;
     }
     el.__timer__ = setInterval(isReize, 300); // 周期性监听元素是否改变
  },
  unbind(el) {
    clearInterval(el.__timer__);
  }
}

参数 Attributes:

1Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

然后你可以在模板中任何元素上使用新的 v-resize property,如下:

// 传入 resize() 方法
<div v-resize="resize"></div>

11、字符串整形指令 v-format

使用该指令可以修改字符串,如使用 v-format.toFixed 保留两位小数、 v-format.price 将内容变成金额(每三位逗号分隔),可以同时使用,如 v-format.toFixed.price。

例如将数字 243112.331 变成 243112.33,或 243,112.33。

指令的代码如下:

export default {
  update (el, binding, vnode) {
    const { value, modifiers } = binding
    if (!value) return
    let formatValue = value
    if (modifiers.toFixed) {
      formatValue = value.toFixed(2)
    }
    console.log(formatValue)
    if (modifiers.price) {
      formatValue = formatNumber(formatValue)
    }
    el.innerText = formatValue
  },
}



function formatNumber (num) {
  num += &#39;&#39;;
  let strs = num.split(&#39;.&#39;);
  let x1 = strs[0];
  let x2 = strs.length > 1 ? &#39;.&#39; + strs[1] : &#39;&#39;;
  var rgx = /(\d+)(\d{3})/;
  while (rgx.test(x1)) {
    x1 = x1.replace(rgx, &#39;$1&#39; + &#39;,&#39; + &#39;$2&#39;);
  }
  return x1 + x2
}

参数 Attributes:

1Vuejs でよく使用されるカスタム命令をいくつか共有します (概要)

然后你可以在模板中任何元素上使用新的 v-format property,如下:

<div> 123 </div>

如何使用这些指令?

为了便于管理指令,我们将每个指令都存在于单独的 js 文件中。在项目的 src 下建一个 directives 目录,目录下新建 index.js 文件用于引入并注册指令。

├── src
|  ├── directive
|  |  ├── index.js
|  |  ├── backtop.js
|  |  ├── badge.js
|  |  ├── copy.js
|  |  ├── ellipsis.js
|  |  ├── empty.js
|  |  ├── expandClick.js
|  |  ├── screenfull.js
|  |  └── tooltips.js
|  ├── main.js

举个例子:

directives 目录下新建 ellipsis.js 文件:

export default function (el, binding) {
    el.style.width = binding.arg || 100 + &#39;px&#39;
    el.style.whiteSpace = &#39;nowrap&#39;
    el.style.overflow = &#39;hidden&#39;;
    el.style.textOverflow = &#39;ellipsis&#39;;
}

directives 的 index.js 文件中引入 ellipsis 指令并注册:

import Vue from &#39;vue&#39;
import ellipsis from &#39;./ellipsis&#39; // 引入指令
// import other directives

const directives = {
  ellipsis
  // other directives
}

Object.keys(directives).forEach(name => Vue.directive(name, directives[name]))

最后在 mian.js 中引入 index.js 文件:

import &#39;@/directives/index&#39;

这样就可以正常使用这些指令了:

import &#39;@/directives/index&#39;

更多编程相关知识,请访问:编程入门!!

以上がVuejs でよく使用されるカスタム命令をいくつか共有します (概要)の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事はsegmentfault.comで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。