index.vue 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. <template>
  2. <movable-area :style="[getAreaStyle]">
  3. <movable-view
  4. v-for="(item, index) in list"
  5. :animation="animation"
  6. :direction="direction"
  7. :key="item.key"
  8. :damping="damping"
  9. :x="item.x"
  10. :y="item.y"
  11. :disabled="longpress ? disabled : false"
  12. @longpress="handleLongpress"
  13. @touchstart="handleDragStart(index)"
  14. @change="handleMoving"
  15. @touchend="handleDragEnd"
  16. :style="[getViewStyle]"
  17. class="base-drag-wrapper"
  18. :class="{ active: activeIndex === index }"
  19. >
  20. <!-- #ifdef MP-WEIXIN -->
  21. <view class="drag-item">
  22. <view :class="['tab-item']" @longpress="toggleEdit"
  23. @tap="deleteTab(item)">
  24. <image v-if="item.isLock && isActiveEdit" :src="icons.lock" class="right-icon" />
  25. <image v-if="getMovableItem(item)" :src="icons.edit" class="right-icon" />
  26. <image :src="item.iconUrl" class="icon" />
  27. <view :class="['name', getMovableItem(item) && 'shake']">{{ item.name }}</view>
  28. </view>
  29. </view>
  30. <!-- #endif -->
  31. <!-- #ifndef MP-WEIXIN -->
  32. <slot name="item" :element="item" :index="index"></slot>
  33. <!-- #endif -->
  34. </movable-view>
  35. </movable-area>
  36. </template>
  37. <script>
  38. export default {
  39. props: {
  40. column: {
  41. type: Number,
  42. default: 3
  43. },
  44. value: {
  45. type: Array,
  46. default: () => []
  47. },
  48. width: {
  49. type: String,
  50. default: '100%'
  51. },
  52. height: {
  53. type: String,
  54. default: 'auto'
  55. },
  56. itemKey: {
  57. type: String,
  58. required: true
  59. },
  60. itemHeight: {
  61. type: String,
  62. default: '100px'
  63. },
  64. direction: {
  65. type: String,
  66. default: 'all',
  67. validator: value => {
  68. return ['all', 'vertical', 'horizontal', 'none'].includes(value);
  69. }
  70. },
  71. animation: {
  72. type: Boolean,
  73. default: true
  74. },
  75. damping: {
  76. type: Number,
  77. default: 20
  78. },
  79. longpress: {
  80. type: Boolean,
  81. default: true
  82. }
  83. },
  84. data() {
  85. return {
  86. list: [],
  87. disabled: true,
  88. activeIndex: -1,
  89. moveToIndex: -1,
  90. oldIndex: -1,
  91. tempDragInfo: {
  92. x: '',
  93. y: ''
  94. },
  95. cloneList: []
  96. };
  97. },
  98. computed: {
  99. getAreaStyle() {
  100. const width = this.getRealWidth(this.width);
  101. return { width: width + 'px', height: this.height !== 'auto' ? this.height : Math.round(this.list.length / this.column) * this.getItemHeight + 'px' };
  102. },
  103. getViewStyle() {
  104. const { itemHeight, getItemWidth } = this;
  105. return { width: getItemWidth + 'px', height: itemHeight };
  106. },
  107. getItemHeight() {
  108. return parseFloat(this.itemHeight);
  109. },
  110. getItemWidth() {
  111. if (this.column === 0) return;
  112. const width = this.getRealWidth(this.width);
  113. return (parseFloat(width) / this.column).toFixed(2);
  114. }
  115. },
  116. methods: {
  117. deleteTab(tab) {
  118. if (tab.isLock) return
  119. const index = this.activeTabs.findIndex(t => t.name === tab.name)
  120. if (index !== -1) {
  121. const removedTab = this.activeTabs.splice(index, 1)[0]
  122. this.toBeSelectTabs.push(removedTab)
  123. this.addPlaceholders()
  124. }
  125. },
  126. toggleEdit() {
  127. this.isActiveEdit = !this.isActiveEdit
  128. },
  129. //获取实际的宽度
  130. getRealWidth(width) {
  131. if (width.includes('%')) {
  132. const windowWidth = uni.getSystemInfoSync().windowWidth;
  133. width = windowWidth * (parseFloat(width) / 100);
  134. }
  135. return width;
  136. },
  137. initList(list = []) {
  138. const newList = this.deepCopy(list);
  139. this.list = newList.map((item, index) => {
  140. const [x, y] = this.getPosition(index);
  141. return {
  142. ...item,
  143. x,
  144. y,
  145. key: Math.random() + index
  146. };
  147. });
  148. this.cloneList = this.deepCopy(this.list);
  149. },
  150. //长按
  151. handleLongpress() {
  152. this.disabled = false;
  153. },
  154. //拖拽开始
  155. handleDragStart(index) {
  156. this.activeIndex = index;
  157. this.oldIndex = index;
  158. },
  159. //拖拽中
  160. handleMoving(e) {
  161. if (e.detail.source !== 'touch') return;
  162. const { x, y } = e.detail;
  163. Object.assign(this.tempDragInfo, { x, y });
  164. const currentX = Math.floor((x + this.getItemWidth / 2) / this.getItemWidth);
  165. const currentY = Math.floor((y + this.getItemHeight / 2) / this.getItemHeight);
  166. this.moveToIndex = Math.min(currentY * this.column + currentX, this.list.length - 1);
  167. if (this.oldIndex !== this.moveToIndex && this.oldIndex !== -1 && this.moveToIndex !== -1) {
  168. const newList = this.deepCopy(this.cloneList);
  169. newList.splice(this.moveToIndex, 0, ...newList.splice(this.activeIndex, 1));
  170. this.list.forEach((item, index) => {
  171. if (index !== this.activeIndex) {
  172. const itemIndex = newList.findIndex(val => val[this.itemKey] === item[this.itemKey]);
  173. [item.x, item.y] = this.getPosition(itemIndex);
  174. }
  175. });
  176. this.oldIndex = this.moveToIndex;
  177. }
  178. },
  179. //获取当前的位置
  180. getPosition(index) {
  181. const x = (index % this.column) * this.getItemWidth;
  182. const y = Math.floor(index / this.column) * this.getItemHeight;
  183. return [x, y];
  184. },
  185. //拖拽结束
  186. handleDragEnd(e) {
  187. if (this.disabled) return;
  188. if (this.moveToIndex !== -1 && this.activeIndex !== -1 && this.moveToIndex !== this.activeIndex) {
  189. this.cloneList.splice(this.moveToIndex, 0, ...this.cloneList.splice(this.activeIndex, 1));
  190. } else {
  191. this.$set(this.list[this.activeIndex], 'x', this.tempDragInfo.x);
  192. this.$set(this.list[this.activeIndex], 'y', this.tempDragInfo.y);
  193. }
  194. this.initList(this.cloneList);
  195. const endList = this.list.map(item => this.omit(item, ['x', 'y', 'key']));
  196. this.$emit('input', endList);
  197. this.$emit('end', endList);
  198. this.activeIndex = -1;
  199. this.oldIndex = -1;
  200. this.moveToIndex = -1;
  201. this.disabled = true;
  202. },
  203. deepCopy(source) {
  204. return JSON.parse(JSON.stringify(source));
  205. },
  206. /**
  207. * 排除掉obj里面的key值
  208. * @param {object} obj
  209. * @param {Array|string} args
  210. * @returns {object}
  211. */
  212. omit(obj, args) {
  213. if (!args) return obj;
  214. const newObj = {};
  215. const isString = typeof args === 'string';
  216. const keys = Object.keys(obj).filter(item => {
  217. if (isString) {
  218. return item !== args;
  219. }
  220. return !args.includes(item);
  221. });
  222. keys.forEach(key => {
  223. if (obj[key] !== undefined) newObj[key] = obj[key];
  224. });
  225. return newObj;
  226. }
  227. },
  228. watch: {
  229. value: {
  230. handler() {
  231. this.initList(this.value);
  232. },
  233. immediate: true,
  234. deep: true
  235. }
  236. }
  237. };
  238. </script>
  239. <style lang="scss" scoped>
  240. .base-drag-wrapper {
  241. opacity: 1;
  242. z-index: 1;
  243. &.active {
  244. opacity: 0.7;
  245. transform: scale(1.3);
  246. z-index: 99;
  247. }
  248. }
  249. </style>