Vue JS: директива click click click

Я знаю эти обработчики событий vue:

@click : mouse left-click
@dblclick : mouse double click

Что может быть обработчиком/директивой для обнаружения щелчка правой кнопкой мыши? Необходимо реализовать пользовательское контекстное меню в представлении Vue Tree.

Спасибо.

Ответ 1

<button @contextmenu="handler($event)">r-click</button>

methods : {
    handler: function(e) {
        //do stuff
        e.preventDefault();
     }
}

@contextmenu сделает трюк. preventDefault следует избегать отображения контекстного меню по умолчанию.

Короче, как указано в комментарии:

<button @contextmenu.prevent="handler">r-click</button>

Теперь модификатор prevent заботится о предотвращении поведения по умолчанию.

Ответ 2

Вот также ответ на "пользовательское" слово - идея, как поймать идентификаторы правых элементов

<html>
  <head>
     <style>
        html, body, #app {
          margin: 0;
          width: 100%;
          height: 100%;
          display: flex;
          text-align: center;
          align-items: center;
          justify-content: center;
          background-color: #343338;
          font-family: Roboto, Tahoma, sans-serif;
        }
        .title {
          color: white;
          display: block;
          font-size: 25px;
        }

        .cls-context-menu {
           display:none;
           position:absolute;
        }
        .cls-context-menu ul, #context-menu li {
            list-style:none;
            margin:0; padding:0;
            background:white;
        }
        .cls-context-menu { border:solid 1px #CCC;}
        .cls-context-menu li {
           border-bottom:solid 1px #CCC;
           display:block;
           padding:5px 12px;
           text-decoration:none;
           color:blue;
        }
        .cls-context-menu li:hover{
            background: blue;
            color: #FFF;
        }
        .cls-context-menu li:last-child { border:none; }

        .context-menu-icon {
          top: 1px;
          position: relative;
          margin-right: 2px;
        }
        .cls-context-menu-item {
          cursor: pointer;
          display:block;
          padding:20px;
          background:#ECECEC;
        }



  </style>
       <!-- <link rel="stylesheet" type="text/css"
            href="../../poc/right-click-vmenu/right-click-menu.css" /> -->
  </head>
  <body>


  <template id="context-menu-item-template">
    <li class="cls-context-menu-item">
      <slot></slot>
    </li>
  </template>

  <template id="context-menu-template">
     <div id="div-context-menu" :nid="cid" class="cls-context-menu" style="left: 160px; top: 57px; display: none">
       <ul>
         <context-menu-item>
           <a @click="addNewItem($event)"><img class="context-menu-icon" src="" width="12">
            add new </a>
         </context-menu-item>
         <context-menu-item>
           <a @click="updateItem($event)"><img class="context-menu-icon" src="" width="12">
           edit </a>
         </context-menu-item>
         <context-menu-item>
           <a @click="removeItem($event)"><img class="context-menu-icon" src="" width="10">
           remove</a>
         </context-menu-item>
       </ul>
     </div>
  </template>

  <div id="app" @click="hideContextMenu()" @contextmenu = "showContextMenu($event)">
    <div>
      <h3 id="title-1" class="title">Num-01 Click with right-a on body</h3>
      <h3 id="title-2" class="title">Num-02 Click with right-a on body</h3>
     </div>
     <context-menu></context-menu>
  </div>

     <div id="js_libs">
        <script src="https://unpkg.com/axios/dist/axios.min.js"></script>
        <script src="https://unpkg.com/[email protected]/dist/vue.js"></script>
        <script src="https://unpkg.com/[email protected]/dist/vuex.js"></script>
     </div>
     <script>

  Vue.component('context-menu-item', {
    template: '#context-menu-item-template',
    props: {
      icon: ''
    }
  });

  Vue.component('context-menu', {
     template: '#context-menu-template'
     , props: {
        icon: ''
        , nid: null
     }
     , data() {
           return {
              cid: null
           }
     }
     , methods: {
        addNewItem: function (e) {
           var itemId = e.target.parentElement.parentElement.parentElement.cid
           var msg = "UPDATE clicked for item-id : " + itemId
           console.log ( msg ) ; alert ( msg ) ;
      }
      , updateItem: function (e) {
           var itemId = e.target.parentElement.parentElement.parentElement.cid
           var msg = "UPDATE clicked for item-id : " + itemId
           console.log ( msg ) ; alert ( msg ) ;
      }
      , removeItem: function (e) {
           var itemId = e.target.parentElement.parentElement.parentElement.cid
           var msg = "REMOVE clicked for item-id : " + itemId
           console.log ( msg ) ; alert ( msg ) ;
      }
     }
  });

  var vm = new Vue({
    el: '#app'
     , methods: {
        showContextMenu: function (e) {
           if ( e.target.className.startsWith("title") ) {
              e.preventDefault();
              var menu = document.getElementById("div-context-menu");
              menu.style.left = e.pageX + 'px'
              menu.style.top = e.pageY + 'px'
              menu.style.display = 'block'
              menu.cid = e.target.id.replace(/title-/,"")
              console.log ( "cid")
              console.log ( menu.cid )
           }
        }
        , hideContextMenu: function () {
           document.getElementById("div-context-menu").style.display = "none"
        }
     }
  });
     </script>
  </body>
  </html>