Mouseover or hover vue.js

前端 未结 13 1696
温柔的废话
温柔的废话 2021-01-30 02:13

I would like to show a div when hovering over an element in vue.js. But I can\'t seem to get it working.

It looks like there is no event for hover or mouseover in vue.j

相关标签:
13条回答
  • 2021-01-30 02:27

    It's possible to toggle a class on hover strictly within a component's template, however, it's not a practical solution for obvious reasons. For prototyping on the other hand, I find it useful to not have to define data properties or event handlers within the script.

    Here's an example of how you can experiment with icon colors using Vuetify.

    new Vue({
      el: '#app'
    })
    <link href="https://fonts.googleapis.com/css?family=Roboto:100,300,400,500,700,900|Material+Icons" rel="stylesheet">
    <link href="https://cdn.jsdelivr.net/npm/vuetify/dist/vuetify.min.css" rel="stylesheet">
    <script src="https://cdnjs.cloudflare.com/ajax/libs/vue/2.5.17/vue.js"></script>
    <script src="https://cdn.jsdelivr.net/npm/vuetify/dist/vuetify.js"></script>
    
    <div id="app">
      <v-app>
        <v-toolbar color="black" dark>
          <v-toolbar-items>
            <v-btn icon>
              <v-icon @mouseenter="e => e.target.classList.toggle('pink--text')" @mouseleave="e => e.target.classList.toggle('pink--text')">delete</v-icon>
            </v-btn>
            <v-btn icon>
              <v-icon @mouseenter="e => e.target.classList.toggle('blue--text')" @mouseleave="e => e.target.classList.toggle('blue--text')">launch</v-icon>
            </v-btn>
            <v-btn icon>
              <v-icon @mouseenter="e => e.target.classList.toggle('green--text')" @mouseleave="e => e.target.classList.toggle('green--text')">check</v-icon>
            </v-btn>
          </v-toolbar-items>
        </v-toolbar>
      </v-app>
    </div>

    0 讨论(0)
  • 2021-01-30 02:29

    Though I would give an update using the new composition api.

    Component

    <template>
      <div @mouseenter="hovering = true" @mouseleave="hovering = false">
        {{ hovering }}
      </div>
    </template>
    
    <script>
      import { ref } from '@vue/compsosition-api'
    
      export default {
        setup() {
          const hovering = ref(false)
          return { hovering }
        }
      })
    </script>
    

    Reusable Composition Function

    Creating a useHover function will allow you to reuse in any components.

    export function useHover(target: Ref<HTMLElement | null>) {
      const hovering = ref(false)
    
      const enterHandler = () => (hovering.value = true)
      const leaveHandler = () => (hovering.value = false)
    
      onMounted(() => {
        if (!target.value) return
        target.value.addEventListener('mouseenter', enterHandler)
        target.value.addEventListener('mouseleave', leaveHandler)
      })
    
      onUnmounted(() => {
        if (!target.value) return
        target.value.removeEventListener('mouseenter', enterHandler)
        target.value.removeEventListener('mouseleave', leaveHandler)
      })
    
      return hovering
    }
    

    Here's a quick example calling the function inside a Vue component.

    <template>
      <div ref="hoverRef">
        {{ hovering }}
      </div>
    </template>
    
    <script lang="ts">
      import { ref } from '@vue/compsosition-api'
      import { useHover } from './useHover'
    
      export default {
        setup() {
          const hoverRef = ref(null)
          const hovering = useHover(hoverRef)
          return { hovering, hoverRef }
        }
      })
    </script>
    

    You can also use a library such as @vuehooks/core which comes with many useful functions including useHover.

    0 讨论(0)
  • 2021-01-30 02:29

    This worked for me for nuxt

    <template>
      <span
        v-if="item"
        class="primary-navigation-list-dropdown"
        @mouseover="isTouchscreenDevice ? null : openDropdownMenu()"
        @mouseleave="isTouchscreenDevice ? null : closeDropdownMenu()"
      >
        <nuxt-link
          to="#"
          @click.prevent.native="openDropdownMenu"
          v-click-outside="closeDropdownMenu"
          :title="item.title"
          :class="[
            item.cssClasses,
            { show: isDropdownMenuVisible }
          ]"
          :id="`navbarDropdownMenuLink-${item.id}`"
          :aria-expanded="[isDropdownMenuVisible ? true : false]"
          class="
            primary-navigation-list-dropdown__toggle
            nav-link
            dropdown-toggle"
          aria-current="page"
          role="button"
          data-toggle="dropdown"
        >
          {{ item.label }}
        </nuxt-link>
        <ul
          :class="{ show: isDropdownMenuVisible }"
          :aria-labelledby="`navbarDropdownMenuLink-${item.id}`"
          class="
            primary-navigation-list-dropdown__menu
            dropdown-menu-list
            dropdown-menu"
        >
          <li
            v-for="item in item.children" :key="item.id"
            class="dropdown-menu-list__item"
          >
            <NavLink
              :attributes="item"
              class="dropdown-menu-list__link dropdown-item"
            />
          </li>
        </ul>
      </span>
    </template>
    
    <script>
    import NavLink from '@/components/Navigation/NavLink';
    
    export default {
      name: "DropdownMenu",
      props: {
        item: {
          type: Object,
          required: true,
        },
      },
      data() {
        return {
          isDropdownMenuVisible: false,
          isTouchscreenDevice: false
        };
      },
      mounted() {
        this.detectTouchscreenDevice();
      },
      methods: {
        openDropdownMenu() {
          if (this.isTouchscreenDevice) {
            this.isDropdownMenuVisible = !this.isDropdownMenuVisible;
          } else {
            this.isDropdownMenuVisible = true;
          }
        },
    
        closeDropdownMenu() {
          if (this.isTouchscreenDevice) {
            this.isDropdownMenuVisible = false;
          } else {
            this.isDropdownMenuVisible = false;
          }
        },
    
        detectTouchscreenDevice() {
          if (window.PointerEvent && ('maxTouchPoints' in navigator)) {
            if (navigator.maxTouchPoints > 0) {
              this.isTouchscreenDevice = true;
            }
          } else {
            if (window.matchMedia && window.matchMedia("(any-pointer:coarse)").matches) {
              this.isTouchscreenDevice = true;
            } else if (window.TouchEvent || ('ontouchstart' in window)) {
              this.isTouchscreenDevice = true;
            }
          }
          return this.isTouchscreenDevice;
        }
      },
      components: {
        NavLink
      }
    };
    </script>
    
    <style scoped lang="scss">
    .primary-navigation-list-dropdown {
      &__toggle {
        color: $white;
    
        &:hover {
          color: $blue;
        }
      }
    
      &__menu {
        margin-top: 0;
      }
    
      &__dropdown {
    
      }
    }
    
    .dropdown-menu-list {
      &__item {
    
      }
    
      &__link {
        &.active,
        &.nuxt-link-exact-active {
          border-bottom: 1px solid $blue;
        }
      }
    }
    </style>
    
    0 讨论(0)
  • 2021-01-30 02:33

    Here is a working example of what I think you are asking for.

    http://jsfiddle.net/1cekfnqw/3017/

     <div id="demo">
            <div v-show="active">Show</div>
            <div @mouseover="mouseOver">Hover over me!</div>
        </div>
    
    var demo = new Vue({
        el: '#demo',
        data: {
            active: false
        },
        methods: {
            mouseOver: function(){
                this.active = !this.active;   
            }
        }
    });
    
    0 讨论(0)
  • 2021-01-30 02:33

    To show child or sibling elements it's possible with CSS only. If you use :hover before combinators (+, ~, >, space). Then the style applies not to hovered element.

    HTML

    <body>
      <div class="trigger">
        Hover here.
      </div>
      <div class="hidden">
        This message shows up.
      </div>
    </body>
    

    CSS

    .hidden { display: none; }
    .trigger:hover + .hidden { display: inline; }
    
    0 讨论(0)
  • 2021-01-30 02:35

    i feel above logics for hover is incorrect. it just inverse when mouse hovers. i have used below code. it seems to work perfectly alright.

    <div @mouseover="upHere = true" @mouseleave="upHere = false" >
        <h2> Something Something </h2>
        <some-component v-show="upHere"></some-component>
    </div>
    

    on vue instance

    data : {
        upHere : false
    }
    

    Hope that helps

    0 讨论(0)
提交回复
热议问题