1. 程式人生 > >vue 自定義指令心得

vue 自定義指令心得

//各個快捷鍵對應的code
const hotKeyList = {
  "S": 83,
  "O": 79,
  "Q": 81,
  "R": 82,
  'ESC': 27,
};

function bindEvent(e, el, arg) {
  if(!el || !arg){
    return
  }
  if (el.displayFlag) {
    // console.log(el) 監聽組合快捷鍵ctrl
    let keyCode = e.keyCode || e.which || e.charCode;
    let ctrlKey = e.ctrlKey || e.metaKey;
    
if (arg === 'ESC' && keyCode === hotKeyList[arg]) { el.click() e.preventDefault(); e.returnValue = false; return false; } else { if (ctrlKey && keyCode === hotKeyList[arg]) { el.click() e.preventDefault(); e.returnValue = false
; return false; } } } } // 遞迴檢測觸發的按鈕是否會顯示 function checkDisplay(el) { if (el.localName === "body") { return true } if (el.style.display === 'none') { return false } else { if (el.style.display === 'block') { return true } else { return checkDisplay(el.parentNode) //
遞迴一定要加retur } } } export default { name: "shortcutKey", inserted() { console.log('inserted') //最好不要用bind,因為bind的el.parentNode是取不到的,用insert或是update都可以 }, update(el, binding) { // console.log('update') if (hotKeyList[binding.arg]) { setTimeout(() => { el.displayFlag = checkDisplay(el) if (!el.displayFlag) { window.removeEventListener('keydown', e => bindEvent(e, el, binding.arg)) } else { window.addEventListener('keydown', e => bindEvent(e, el, binding.arg)) } }, 300) } }, unbind() { //解綁事件,應該是有問題的,因為自定義指令定義的變數不是區域性的,第二個指令會覆蓋調之前定義的 window.removeEventListener('keydown', bindEvent) } };
import Vue from "vue";

const isServer = Vue.prototype.$isServer;
const on = (function () {
  if (!isServer && document.addEventListener) {
    return function (element, event, handler) {
      if (element && event && handler) {
        element.addEventListener(event, handler, false);
      }
    };
  }
  return function (element, event, handler) {
    if (element && event && handler) {
      element.attachEvent("on" + event, handler);
    }
  };
})();

const nodeList = [];
const ctx = "@@clickoutsideContext";

let startClick;
let seed = 0;

!Vue.prototype.$isServer && on(document, "mousedown", e => (startClick = e));

!Vue.prototype.$isServer &&
  on(document, "mouseup", e => {
    nodeList.forEach(node => node[ctx].documentHandler(e, startClick));
  });

function createDocumentHandler(el, binding, vnode) {
  return function (mouseup = {}, mousedown = {}) {
    if (
      !vnode ||
      !vnode.context ||
      !mouseup.target ||
      !mousedown.target ||
      el.contains(mouseup.target) ||
      el.contains(mousedown.target) ||
      el === mouseup.target ||
      (vnode.context.popperElm &&
        (vnode.context.popperElm.contains(mouseup.target) ||
          vnode.context.popperElm.contains(mousedown.target)))
    )
      return;

    if (
      binding.expression &&
      el[ctx].methodName &&
      vnode.context[el[ctx].methodName]
    ) {
      vnode.context[el[ctx].methodName]();
    } else {
      el[ctx].bindingFn && el[ctx].bindingFn();
    }
  };
}

/**
 * v-clickoutside
 * @desc 點選元素外面才會觸發的事件
 * @example
 * ```vue
 * <div v-clickoutside="handleClose">
 * ```
 */
export default {
  name: "clickoutside",
  bind(el, binding, vnode) {
    nodeList.push(el);
    const id = seed++;
    el[ctx] = {
      id,
      documentHandler: createDocumentHandler(el, binding, vnode),
      methodName: binding.expression,
      bindingFn: binding.value
    };


  },

  update(el, binding, vnode) {


    el[ctx].documentHandler = createDocumentHandler(el, binding, vnode);
    el[ctx].methodName = binding.expression;
    el[ctx].bindingFn = binding.value;

    // 判斷指令中是否綁定了函式
    if (binding.expression) {
      // 如果綁定了函式 則呼叫那個函式,此處binding.value就是handleClose方法
      binding.value(startClick);
    }

  },

  unbind(el) {
    let len = nodeList.length;

    for (let i = 0; i < len; i++) {
      if (nodeList[i][ctx].id === el[ctx].id) {
        nodeList.splice(i, 1);
        break;
      }
    }
    delete el[ctx];
  }
};