MixGlobalHeader.vue 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. <template>
  2. <transition name="showHeader">
  3. <div v-if="visible" class="header-animat">
  4. <a-layout-header
  5. v-if="visible"
  6. :class="[fixedHeader && 'ant-header-fixedHeader', sidebarOpened ? 'ant-header-side-opened' : 'ant-header-side-closed', ]"
  7. :style="{ padding: '0' }">
  8. <div :class="['top-nav-header-index', theme]">
  9. <div class="header-index-wide">
  10. <div class="header-index-left">
  11. <logo class="top-nav-header" :show-title="device !== 'mobile'"/>
  12. <s-menu :class="theme === 'dark' ?'future-menu':''" v-if="device !== 'mobile'" mode="horizontal" :menu="headerMenus" :theme="theme" />
  13. </div>
  14. <user-menu class="header-index-right"></user-menu>
  15. </div>
  16. </div>
  17. </a-layout-header>
  18. </div>
  19. </transition>
  20. </template>
  21. <script>
  22. import UserMenu from '../tools/UserMenu'
  23. import SMenu from '../Menu/mixMenu'
  24. import Logo from '../tools/Logo'
  25. import { mixin } from '@/utils/mixin'
  26. export default {
  27. name: 'GlobalHeader',
  28. components: {
  29. UserMenu,
  30. SMenu,
  31. Logo
  32. },
  33. mixins: [mixin],
  34. props: {
  35. mode: {
  36. type: String,
  37. // sidemenu, topmenu
  38. default: 'sidemenu'
  39. },
  40. menus: {
  41. type: Array,
  42. required: true
  43. },
  44. theme: {
  45. type: String,
  46. required: false,
  47. default: 'dark'
  48. },
  49. collapsed: {
  50. type: Boolean,
  51. required: false,
  52. default: false
  53. },
  54. device: {
  55. type: String,
  56. required: false,
  57. default: 'desktop'
  58. }
  59. },
  60. computed: {
  61. headerMenus () {
  62. // console.log(this.menus)
  63. return this.menus.map(item => {
  64. return item
  65. })
  66. }
  67. },
  68. data () {
  69. return {
  70. visible: true,
  71. oldScrollTop: 0
  72. }
  73. },
  74. mounted () {
  75. document.addEventListener('scroll', this.handleScroll, { passive: true })
  76. },
  77. methods: {
  78. handleScroll () {
  79. if (!this.autoHideHeader) {
  80. return
  81. }
  82. const scrollTop = document.body.scrollTop + document.documentElement.scrollTop
  83. if (!this.ticking) {
  84. this.ticking = true
  85. requestAnimationFrame(() => {
  86. if (this.oldScrollTop > scrollTop) {
  87. this.visible = true
  88. } else if (scrollTop > 300 && this.visible) {
  89. this.visible = false
  90. } else if (scrollTop < 300 && !this.visible) {
  91. this.visible = true
  92. }
  93. this.oldScrollTop = scrollTop
  94. this.ticking = false
  95. })
  96. }
  97. },
  98. toggle () {
  99. this.$emit('toggle')
  100. }
  101. },
  102. beforeDestroy () {
  103. document.body.removeEventListener('scroll', this.handleScroll, true)
  104. }
  105. }
  106. </script>
  107. <style lang="less">
  108. @import '../index.less';
  109. .header-animat{
  110. position: fixed;
  111. top: 0;
  112. width: 100vw;
  113. z-index: @ant-global-header-zindex;
  114. }
  115. .showHeader-enter-active {
  116. transition: all 0.25s ease;
  117. }
  118. .showHeader-leave-active {
  119. transition: all 0.5s ease;
  120. }
  121. .showHeader-enter, .showHeader-leave-to {
  122. opacity: 0;
  123. }
  124. .future-menu>.ant-menu-item-selected>a{
  125. color: #fff !important;
  126. }
  127. </style>