tag.vue 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. <template>
  2. <view class="app-container">
  3. <view class="list-container">
  4. <block v-for="(item, index) in tagList" :key='index'>
  5. <view class="item">
  6. <view class="top">
  7. <view class="title">{{item.tagGroupName}}</view>
  8. </view>
  9. <view class="tags">
  10. <block v-for="(it, idx) in item.tags" :key='idx'>
  11. <view class="tag" :class="it.tagged ? 'current':''" @tap="changeTag(index, idx)">{{it.tagName}}</view>
  12. </block>
  13. </view>
  14. </view>
  15. </block>
  16. <view class="item">
  17. <view class="top">
  18. <view class="title">自定义标签</view>
  19. </view>
  20. <view class="tags">
  21. <block v-for="(item, index) in customTagList" :key='index'>
  22. <view class="tag current">{{item}}<image src="@/static/icon/close2.png" class="close" @tap="deleteTag(index)"></image></view>
  23. </block>
  24. <view class="tag add" @tap="isShowDialog = true"><image src="@/static/icon/add.png"></image>新建标签</view>
  25. </view>
  26. </view>
  27. </view>
  28. <view class="bottom-container">
  29. <view class="button" @tap="submitForm">保存</view>
  30. </view>
  31. <view class="global-mask" v-show="isShowDialog"></view>
  32. <view class="global-dialog" v-show="isShowDialog" style="top: 40%;">
  33. <view class="title">新建标签</view>
  34. <view class="input">
  35. <input type="text" placeholder="标签名称" v-model="newTagName">
  36. </view>
  37. <view class="btn">
  38. <view class="left" @tap="isShowDialog = false">取消</view>
  39. <view class="right" @tap="addTag()">确定</view>
  40. </view>
  41. </view>
  42. </view>
  43. </template>
  44. <script>
  45. export default {
  46. data() {
  47. return {
  48. unionId: null,
  49. tagList: [],
  50. customTagList: [],
  51. isShowDialog: false,
  52. newTagName: '',
  53. oldTagIds: [],
  54. }
  55. },
  56. onLoad({unionId}) {
  57. this.unionId = unionId;
  58. this.getTagList();
  59. },
  60. methods: {
  61. // 获取标签列表
  62. getTagList() {
  63. this.$axios({
  64. url: '/tag/wx/tag/list',
  65. method: 'get',
  66. params: {
  67. unionId: this.unionId
  68. },
  69. }).then(res => {
  70. this.tagList = res.data.tagGroupBeanList;
  71. this.customTagList = res.data.tagNameList;
  72. for (let i = 0; i < this.tagList.length; i++) {
  73. for (let j = 0; j < this.tagList[i].tags.length; j++) {
  74. if(this.tagList[i].tags[j].tagged) {
  75. this.oldTagIds.push(this.tagList[i].tags[j].tagId);
  76. }
  77. }
  78. }
  79. console.log(this.oldTagIds);
  80. })
  81. },
  82. // 新建标签
  83. addTag() {
  84. if(!this.newTagName) {
  85. return this.$toast('请填写标签名称');
  86. }
  87. this.customTagList.push(this.newTagName);
  88. this.newTagName = '';
  89. this.isShowDialog = false;
  90. },
  91. // 删除标签
  92. deleteTag(index) {
  93. this.customTagList.splice(index, 1);
  94. },
  95. // 切换标签
  96. changeTag(index, idx) {
  97. this.tagList[index].tags[idx].tagged = !this.tagList[index].tags[idx].tagged;
  98. },
  99. // 保存
  100. submitForm() {
  101. let tagList = [];
  102. let oldTagIds = this.oldTagIds;
  103. let newTagIds = [];
  104. let delTagIds = [];
  105. // 查出选中的标签
  106. for (let i = 0; i < this.tagList.length; i++) {
  107. for (let j = 0; j < this.tagList[i].tags.length; j++) {
  108. if(this.tagList[i].tags[j].tagged) {
  109. tagList.push(this.tagList[i].tags[j]);
  110. newTagIds.push(this.tagList[i].tags[j].tagId);
  111. }
  112. }
  113. }
  114. // 查出被删除的标签
  115. for (let k = 0; k < oldTagIds.length; k++) {
  116. if(newTagIds.indexOf(oldTagIds[k]) < 0) {
  117. delTagIds.push(oldTagIds[k]);
  118. }
  119. }
  120. let params = {
  121. unionId: this.unionId,
  122. saveTags: this.customTagList,
  123. delTags: delTagIds,
  124. addTags: tagList
  125. }
  126. this.$axios({
  127. url: '/tag/custom/edit',
  128. type: 'application/json',
  129. params,
  130. isLoading: 1,
  131. }).then(res => {
  132. this.$successToast('操作成功');
  133. setTimeout(() => {
  134. uni.$emit('refreshCustomerData');
  135. uni.navigateBack({
  136. delta: 1
  137. })
  138. }, 1000)
  139. })
  140. },
  141. }
  142. }
  143. </script>
  144. <style lang="scss">
  145. .app-container {
  146. background: #F4F2F2;
  147. padding: 20rpx 20rpx 120rpx;
  148. box-sizing: border-box;
  149. }
  150. .list-container {
  151. .item {
  152. border-bottom: 1px solid #E5E5E5;
  153. .top {
  154. .title {
  155. font-size: 28rpx;
  156. color: #666666;
  157. line-height: 70rpx;
  158. }
  159. }
  160. .tags {
  161. display: flex;
  162. flex-wrap: wrap;
  163. .tag {
  164. display: flex;
  165. align-items: center;
  166. justify-content: center;
  167. font-size: 24rpx;
  168. padding: 0 16rpx;
  169. height: 48rpx;
  170. line-height: 48rpx;
  171. border-radius: 10rpx;
  172. margin-right: 20rpx;
  173. margin-bottom: 20rpx;
  174. background: #ffffff;
  175. color: #666666;
  176. border: 1px solid #eaeaea;
  177. .close {
  178. width: 20rpx;
  179. height: 20rpx;
  180. display: block;
  181. margin-left: 20rpx;
  182. }
  183. &:last-child {
  184. margin-right: 0;
  185. }
  186. &.current {
  187. background: #FFD6BB;
  188. color: #FE781F;
  189. border: 1px solid rgba($color: #FE781F, $alpha: 0.4);
  190. }
  191. &.add {
  192. image {
  193. width: 20rpx;
  194. height: 20rpx;
  195. display: block;
  196. margin-right: 10rpx;
  197. }
  198. }
  199. }
  200. }
  201. }
  202. }
  203. .global-dialog {
  204. .input {
  205. display: flex;
  206. justify-content: center;
  207. margin-bottom: 50rpx;
  208. input {
  209. width: 500rpx;
  210. height: 84rpx;
  211. background: #F3F3F3;
  212. border-radius: 10rpx;
  213. padding: 0 20rpx;
  214. }
  215. }
  216. }
  217. .bottom-container {
  218. position: fixed;
  219. bottom: 0;
  220. left: 0;
  221. width: 100%;
  222. height: 100rpx;
  223. background: #FFFFFF;
  224. padding: 0 20rpx;
  225. display: flex;
  226. justify-content: center;
  227. align-items: center;
  228. box-sizing: border-box;
  229. .button {
  230. width: 100%;
  231. height: 72rpx;
  232. text-align: center;
  233. line-height: 72rpx;
  234. border-radius: 72rpx;
  235. background: linear-gradient(-90deg,#ff3f42 0%, #fe781f 100%);
  236. font-size: 28rpx;
  237. color: #FFFFFF;
  238. }
  239. }
  240. </style>