uni-drawer.vue 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. <template>
  2. <view v-if="visibleSync" :class="{ 'uni-drawer--visible': showDrawer }" class="uni-drawer">
  3. <view class="uni-drawer__mask" :class="{ 'uni-drawer__mask--visible': showDrawer && mask }" @tap="close('mask')" />
  4. <view class="uni-drawer__content" :class="{'uni-drawer--right': rightMode,'uni-drawer--left': !rightMode, 'uni-drawer__content--visible': showDrawer}">
  5. <slot />
  6. </view>
  7. </view>
  8. </template>
  9. <script>
  10. export default {
  11. name: 'UniDrawer',
  12. props: {
  13. /**
  14. * 显示状态
  15. */
  16. visible: {
  17. type: Boolean,
  18. default: false
  19. },
  20. /**
  21. * 显示模式(左、右),只在初始化生效
  22. */
  23. mode: {
  24. type: String,
  25. default: ''
  26. },
  27. /**
  28. * 蒙层显示状态
  29. */
  30. mask: {
  31. type: Boolean,
  32. default: true
  33. },
  34. maskClick:{
  35. type: Boolean,
  36. default: true
  37. },
  38. },
  39. data() {
  40. return {
  41. visibleSync: false,
  42. showDrawer: false,
  43. rightMode: false,
  44. watchTimer: null
  45. }
  46. },
  47. watch: {
  48. visible(val) {
  49. if (val) {
  50. this.open()
  51. } else {
  52. this.close()
  53. }
  54. }
  55. },
  56. created() {
  57. this.visibleSync = this.visible
  58. setTimeout(() => {
  59. this.showDrawer = this.visible
  60. }, 100)
  61. this.rightMode = this.mode === 'right'
  62. },
  63. methods: {
  64. close(type) {
  65. if(type=='mask' && !this.maskClick){
  66. return
  67. }
  68. this._change('showDrawer', 'visibleSync', false)
  69. },
  70. open() {
  71. this._change('visibleSync', 'showDrawer', true)
  72. },
  73. _change(param1, param2, status) {
  74. this[param1] = status
  75. if (this.watchTimer) {
  76. clearTimeout(this.watchTimer)
  77. }
  78. this.watchTimer = setTimeout(() => {
  79. this[param2] = status
  80. this.$emit(status ? 'open' : 'close')
  81. }, status ? 50 : 300)
  82. }
  83. }
  84. }
  85. </script>
  86. <style lang="scss" scoped>
  87. // 抽屉宽度
  88. $drawer-width: 220px;
  89. .uni-drawer {
  90. /* #ifndef APP-NVUE */
  91. display: block;
  92. /* #endif */
  93. position: fixed;
  94. top: 0;
  95. left: 0;
  96. right: 0;
  97. bottom: 0;
  98. overflow: hidden;
  99. z-index: 999;
  100. }
  101. .uni-drawer__content {
  102. /* #ifndef APP-NVUE */
  103. display: block;
  104. /* #endif */
  105. position: absolute;
  106. top: 0;
  107. width: $drawer-width;
  108. bottom: 0;
  109. background-color: $uni-bg-color;
  110. transition: transform 0.3s ease;
  111. }
  112. .uni-drawer--left {
  113. left: 0;
  114. transform: translateX(-$drawer-width);
  115. }
  116. .uni-drawer--right {
  117. right: 0;
  118. transform: translateX($drawer-width);
  119. }
  120. .uni-drawer__content--visible {
  121. transform: translateX(0px);
  122. }
  123. .uni-drawer__mask {
  124. /* #ifndef APP-NVUE */
  125. display: block;
  126. /* #endif */
  127. opacity: 0;
  128. position: absolute;
  129. top: 0;
  130. left: 0;
  131. bottom: 0;
  132. right: 0;
  133. background-color: $uni-bg-color-mask;
  134. transition: opacity 0.3s;
  135. }
  136. .uni-drawer__mask--visible {
  137. /* #ifndef APP-NVUE */
  138. display: block;
  139. /* #endif */
  140. opacity: 1;
  141. }
  142. </style>