mirror of https://github.com/ElemeFE/element
				
				
				
			
		
			
				
	
	
		
			174 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Vue
		
	
	
			
		
		
	
	
			174 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Vue
		
	
	
<style>
 | 
						|
  .side-nav {
 | 
						|
    width: 100%;
 | 
						|
    box-sizing: border-box;
 | 
						|
    padding-right: 30px;
 | 
						|
 | 
						|
    li {
 | 
						|
      list-style: none;
 | 
						|
    }
 | 
						|
    ul {
 | 
						|
      padding: 0;
 | 
						|
      margin: 0;
 | 
						|
      overflow: hidden;
 | 
						|
    }
 | 
						|
 | 
						|
    .nav-item {
 | 
						|
      a {
 | 
						|
        font-size: 16px;
 | 
						|
        color: #5e6d82;
 | 
						|
        line-height: 40px;
 | 
						|
        height: 40px;
 | 
						|
        margin: 0;
 | 
						|
        padding: 0;
 | 
						|
        text-decoration: none;
 | 
						|
        display: block;
 | 
						|
        position: relative;
 | 
						|
        transition: all .3s;
 | 
						|
 | 
						|
        &.active {
 | 
						|
          color: #20a0ff;
 | 
						|
        }
 | 
						|
      }
 | 
						|
      .nav-item {
 | 
						|
        a {
 | 
						|
          display: block;
 | 
						|
          height: 40px;
 | 
						|
          line-height: 40px;
 | 
						|
          font-size: 13px;
 | 
						|
          padding-left: 24px;
 | 
						|
 | 
						|
 | 
						|
          &:hover {
 | 
						|
            color: #20a0ff;
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
    .nav-group__title {
 | 
						|
      font-size: 12px;
 | 
						|
      color: #99a9bf;
 | 
						|
      padding-left: 8px;
 | 
						|
      line-height: 26px;
 | 
						|
      margin-top: 10px;
 | 
						|
    }
 | 
						|
  }
 | 
						|
</style>
 | 
						|
<template>
 | 
						|
  <div class="side-nav" :style="navStyle">
 | 
						|
    <ul>
 | 
						|
      <li class="nav-item" v-for="item in data">
 | 
						|
        <a v-if="!item.path" @click="expandMenu">{{item.name}}</a>
 | 
						|
        <router-link
 | 
						|
          v-else
 | 
						|
          active-class="active"
 | 
						|
          :to="base + item.path"
 | 
						|
          exact
 | 
						|
          v-text="item.title || item.name">
 | 
						|
        </router-link>
 | 
						|
        <ul class="pure-menu-list sub-nav" v-if="item.children">
 | 
						|
          <li class="nav-item" v-for="navItem in item.children">
 | 
						|
            <router-link
 | 
						|
              class=""
 | 
						|
              active-class="active"
 | 
						|
              :to="base + navItem.path"
 | 
						|
              exact
 | 
						|
              v-text="navItem.title || navItem.name">
 | 
						|
            </router-link>
 | 
						|
          </li>
 | 
						|
        </ul>
 | 
						|
        <template v-if="item.groups">
 | 
						|
          <div class="nav-group" v-for="group in item.groups">
 | 
						|
            <div class="nav-group__title" @click="expandMenu">{{group.groupName}}</div>
 | 
						|
            <ul class="pure-menu-list">
 | 
						|
              <li
 | 
						|
                class="nav-item"
 | 
						|
                v-for="navItem in group.list"
 | 
						|
                v-if="!navItem.disabled">
 | 
						|
                <router-link
 | 
						|
                  active-class="active"
 | 
						|
                  :to="base + navItem.path"
 | 
						|
                  exact
 | 
						|
                  v-text="navItem.title"></router-link>
 | 
						|
              </li>
 | 
						|
            </ul>
 | 
						|
          </div>
 | 
						|
        </template>
 | 
						|
      </li>
 | 
						|
    </ul>
 | 
						|
  </div>
 | 
						|
</template>
 | 
						|
<script>
 | 
						|
  export default {
 | 
						|
    props: {
 | 
						|
      data: Array,
 | 
						|
      base: {
 | 
						|
        type: String,
 | 
						|
        default: ''
 | 
						|
      }
 | 
						|
    },
 | 
						|
    data() {
 | 
						|
      return {
 | 
						|
        highlights: [],
 | 
						|
        navState: [],
 | 
						|
        isSmallScreen: false
 | 
						|
      };
 | 
						|
    },
 | 
						|
    watch: {
 | 
						|
      '$route.path'() {
 | 
						|
        this.handlePathChange();
 | 
						|
      }
 | 
						|
    },
 | 
						|
    computed: {
 | 
						|
      navStyle() {
 | 
						|
        return this.isSmallScreen ? { 'padding-bottom': '60px' } : {};
 | 
						|
      }
 | 
						|
    },
 | 
						|
    methods: {
 | 
						|
      handleResize() {
 | 
						|
        this.isSmallScreen = document.documentElement.clientWidth < 768;
 | 
						|
        this.handlePathChange();
 | 
						|
      },
 | 
						|
      handlePathChange() {
 | 
						|
        if (!this.isSmallScreen) {
 | 
						|
          this.expandAllMenu();
 | 
						|
          return;
 | 
						|
        }
 | 
						|
        this.$nextTick(() => {
 | 
						|
          this.hideAllMenu();
 | 
						|
          let activeAnchor = this.$el.querySelector('a.active');
 | 
						|
          let ul = activeAnchor.parentNode;
 | 
						|
          while (ul.tagName !== 'UL') {
 | 
						|
            ul = ul.parentNode;
 | 
						|
          }
 | 
						|
          ul.style.height = 'auto';
 | 
						|
        });
 | 
						|
      },
 | 
						|
      hideAllMenu() {
 | 
						|
        [].forEach.call(this.$el.querySelectorAll('.pure-menu-list'), ul => {
 | 
						|
          ul.style.height = '0';
 | 
						|
        });
 | 
						|
      },
 | 
						|
      expandAllMenu() {
 | 
						|
        [].forEach.call(this.$el.querySelectorAll('.pure-menu-list'), ul => {
 | 
						|
          ul.style.height = 'auto';
 | 
						|
        });
 | 
						|
      },
 | 
						|
      expandMenu(event) {
 | 
						|
        if (!this.isSmallScreen) return;
 | 
						|
        let target = event.currentTarget;
 | 
						|
        if (!target.nextElementSibling || target.nextElementSibling.tagName !== 'UL') return;
 | 
						|
        this.hideAllMenu();
 | 
						|
        event.currentTarget.nextElementSibling.style.height = 'auto';
 | 
						|
      }
 | 
						|
    },
 | 
						|
    mounted() {
 | 
						|
      this.handleResize();
 | 
						|
      window.addEventListener('resize', this.handleResize);
 | 
						|
    },
 | 
						|
    beforeDestroy() {
 | 
						|
      window.removeEventListener('resize', this.handleResize);
 | 
						|
    }
 | 
						|
  };
 | 
						|
</script>
 |