微信小程序 卡片动画 for循环

wxml页面:

 1 <view class="container">
 2     <view wx:for="{{info}}" data-index="{{index}}" class="card" id="{{item.id}}" style="position: absolute; left:{{item.left}}rpx; top:{{item.top + index*10}}rpx;display:{{display1}}; z-index:{{100-index}}; opacity:1;" bindtouchstart="viewTouchInside" bindtouchmove="viewDidMove" bindtouchend="viewTouchUpDownInside" animation="{{item.animation}}">
 3         <view style="100%;height:auto;margin-top:13%;"><rich-text nodes="{{item.profile}}"></rich-text></view>
 4     </view>
 5     <view style="height:100rpx;100%;  position: fixed;bottom:100rpx;">
 6         <view class="left-view" bindtap="LastPage" ><image src="/images/left.png"  ></image></view>
 7         <view style="float:left;60%;">
 8         <label class="progress-view">进度{{currentTime}}/{{duration}}</label>
 9          <progress class="progress" percent="{{currentTime/duration * 100}}" color="#F9F9FF" backgroundColor="#943134"></progress></view>
10         <view class="right-view" bindtap="NextPage" ><image src="/images/right.png"  ></image></view>
11     </view>
12 </view>

wxss:

  1 .container {
  2   /* height: 100%; */
  3   overflow: hidden;
  4 
  5 }
  6 
  7 page {
  8   /* height: 100%; */
  9   overflow: hidden;
 10   background-color: black;
 11   background: -moz-linear-gradient(top, #FE7676 0%, #FE4E4E 100%);
 12   background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, #FE4E4E), color-stop(100%, #FE4E4E));
 13   background: -webkit-linear-gradient(top, #FE7676 0%, #FE4E4E 100%);
 14   background: -o-linear-gradient(top, #FE7676 0%, #FE4E4E 100%);
 15   background: -ms-linear-gradient(top, #FE7676 0%, #FE4E4E 100%);
 16   background: linear-gradient(to bottom, #FE7676 0%, #FE4E4E 100%);
 17 }
 18 
 19 .card {
 20   width: 600rpx;
 21   height: 800rpx;
 22   box-sizing: border-box;
 23   background-color: white;
 24   text-align: center;
 25   box-shadow: 2px 2px 8px #aaa;
 26   border-radius: 10rpx;
 27   padding: 0 50rpx;
 28   background-image: url('');
 29   background-position: bottom;
 30 
 31 }
 32 
 33 .card .name {
 34   font-size: 20px;
 35   font-weight: bolder;
 36   margin: 60rpx 0 20rpx;
 37 }
 38 
 39 .card .location {
 40   font-size: 28rpx;
 41   margin: 30rpx 0 20rpx;
 42   line-height: 40rpx;
 43   text-align: left;
 44   color: #535353;
 45 
 46 }
 47 
 48 .name1 {
 49   font-size: 26rpx;
 50   font-weight: bolder;
 51 }
 52 
 53 .card .like-img {
 54   width: 50%;
 55   height: 45rpx;
 56   margin: 10rpx 0 40rpx;
 57 }
 58 
 59 .school-sign-img {
 60   width: 280rpx;
 61   height: 440rpx;
 62   position: absolute;
 63   left: 320rpx;
 64   top: 360rpx;
 65 }
 66 
 67 .allLikeView {
 68   width: 100rpx;
 69   height: 100rpx;
 70   box-sizing: border-box;
 71   background-color: white;
 72   padding: 30rpx;
 73   box-shadow: 0px 2px 8px #aaa;
 74   border-radius: 50rpx;
 75   margin: 800rpx;
 76 }
 77 
 78 .allLikeView .all-like-img {
 79   width: 40rpx;
 80   height: 40rpx;
 81 }
 82 
 83 .left-view {
 84   float: left;
 85   width: 20%;
 86   text-align: center;
 87   padding-top: 12rpx;
 88 }
 89 
 90 .left-view image {
 91   width: 32px;
 92   height: 32px;
 93 }
 94 
 95 .right-view {
 96   float: right;
 97   width: 20%;
 98   text-align: center;
 99   padding-top: 12rpx;
100 }
101 
102 .right-view image {
103   width: 32px;
104   height: 32px;
105 }
106 
107 .progress-view {
108   color: #ffffff;
109   text-align: center;
110   width: 100%;
111   font-size: 24rpx;
112   display: block;
113   padding-bottom: 12rpx;
114 }
View Code

js:

  1 import util from '../../utils/util.js';
  2 const marginHori = 74
  3 const marginVerti = 100
  4 
  5 Page({
  6 
  7   data: {
  8     course_id: 0,
  9     course_type: 1,
 10     info: [],
 11     currentTime: 0, //当前卡片的索引
 12     duration: 1, //持续时间,卡片总数;
 13 
 14     startX: 0, //滑动开始x轴
 15     startY: 0, //滑动开始y轴
 16 
 17     windowWidth: 0, //屏幕宽度
 18     windowHeight: 0, //屏幕高度
 19 
 20     currentId: 1,
 21   },
 22 
 23   onLoad: function (options) {
 24 
 25     wx.setNavigationBarTitle({
 26       title: '卡片动画',
 27     })
 28     var that = this
 29 
 30     //获取屏幕的宽度和高度
 31     wx.getSystemInfo({
 32       success: function (res) {
 33         that.setData({
 34           windowWidth: res.windowWidth,
 35           windowHeight: res.windowHeight
 36         })
 37       }
 38     });
 39 
 40     var course_id = options.id; //课程id
 41     var course_type = options.type; //课程类型
 42     this.setData({
 43       course_id: parseInt(course_id),
 44       course_type: parseInt(course_type),
 45     });
 46 
 47     //服务接口数据
 48     this.getInfo();
 49 
 50   },
 51   getInfo: function () {
 52     let self = this;
 53     util.request(util.apiUrl + '接口获取数据列表', 'POST', {
 54       course_id: self.data.course_id,
 55       course_type: self.data.course_type,
 56       currentTime: self.data.currentTime,
 57     }).then(res => {
 58       //
 59     }).catch(res => {
 60       var info = res.data.list;
 61 
 62       //获取服务器数据
 63       info.data.forEach((item, index) => {
 64         // info.data[index].animation = animation; //保存动画
 65         info.data[index].left = marginHori //卡片坐标 距左边距离
 66         info.data[index].top = marginVerti //卡片坐标 距上边距离
 67         if ((parseInt(info.currentTime) - 1) > index) { //数组索引index从0开始,所以需要-1
 68           info.data[index].left = 10000; //卡片坐标 距左边距离
 69           info.data[index].top = 10000; //卡片坐标 距上边距离
 70         }
 71       });
 72 
 73       self.setData({ //保存卡片数据
 74         info: info.data
 75       })
 76 
 77       self.setData({
 78         currentTime: info.currentTime,
 79         duration: parseInt(info.duration),
 80       });
 81       console.log(self.data.info);
 82     });
 83   },
 84 
 85   //获取课程进度
 86   getProcess: function () {
 87     let self = this;
 88     util.request(util.apiUrl + '接口', 'POST', {
 89       course_id: self.data.course_id, //当前课程id
 90       current: self.data.currentTime, //已看的索引
 91       duration: self.data.duration, //课程总共的卡片数量
 92       read_id: self.data.currentId, //当前索引
 93     }).then(res => {
 94       //
 95     }).catch(res => {
 96       console.log(res)
 97     });
 98   },
 99 
100   onReady: function () { },
101 
102   onShow: function () { },
103 
104   //手指触摸动作开始
105   viewTouchInside: function (event) {
106 
107     var that = this
108     var index = event.currentTarget.dataset.index; //下标
109     var item = that.data.info[index].animation; //获取每一个的动画
110     var pointX = event.touches[0].clientX
111     var pointY = event.touches[0].clientY
112 
113     that.setData({
114       startX: pointX,
115       startY: pointY,
116       currentId: event.currentTarget.id
117     })
118   },
119 
120   //手指触摸后移动
121   viewDidMove: function (event) {
122 
123     var that = this
124 
125     var pointX = event.touches[0].clientX
126     var pointY = event.touches[0].clientY
127 
128     var widthCenter = that.data.windowWidth / 2
129     var heightCenter = that.data.windowHeight / 2
130 
131     var perX = (pointX - that.data.startX) / widthCenter
132     var perY = (pointY - that.data.startY) / heightCenter
133     var maxPer = (Math.abs(perX) > Math.abs(perY)) ? Math.abs(perX) : Math.abs(perY)
134 
135     var index = event.currentTarget.dataset.index; //下标
136     var item = that.data.info[index].animation; //获取每一个的动画
137 
138     var animationRotate = wx.createAnimation({
139       duration: 100,
140       timingFunction: 'ease-out',
141     })
142     animationRotate.scale(1).rotate(perX * 20).step();
143     var x = marginHori + pointX - that.data.startX
144     var y = marginVerti + pointY - that.data.startY
145 
146     that.data.info[index].left = x,
147       that.data.info[index].top = y,
148       that.data.info[index].animation = animationRotate.export();
149 
150     that.setData({
151       info: that.data.info
152     })
153   },
154 
155   //手指触摸动作结束
156   viewTouchUpDownInside: function (event) {
157 
158     var that = this
159     console.log("that.data.currentId " + that.data.currentId);
160     var endX = event.changedTouches[0].clientX
161     var endY = event.changedTouches[0].clientY
162 
163     var distanceX = endX - that.data.startX
164     var distanceY = endY - that.data.startY
165 
166     var index = event.currentTarget.dataset.index; //下标 这个下标从0开始,所以不需要-1
167     var item = that.data.info[index].animation; //获取每一个的动画
168 
169     if (distanceX > 93.75) {
170       that.removeCard('right', index + 1); //往右移除卡片
171     } else if (distanceX < -93.75) {
172       that.removeCard('left', index + 1); //往左移除卡片
173     } else if (distanceY < -100) {
174       that.removeCard('up', index + 1); //往上移除卡片
175     } else if (distanceY > 100) {
176       that.removeCard('down', index + 1); //往下移除卡片
177     }
178 
179     var animation = wx.createAnimation({
180       duration: 100,
181       timingFunction: 'ease-out',
182     })
183 
184     //移动的范围不大,回到原点
185     if (distanceX < 93.75 && distanceX > -93.75 && distanceY > -150 && distanceY < 150) {
186 
187       that.data.info[index].left = marginHori;
188       that.data.info[index].top = marginVerti;
189       that.setData({
190         info: that.data.info
191       })
192       animation.scale(1).step();
193       that.data.info[index].animation = animation.export();
194       that.setData({ //保存动画内容并渲染到页面
195         info: that.data.info
196       })
197     }
198 
199   },
200 
201   //移除卡片
202   removeCard: function (direction, index) {
203 
204     console.log('removeCard', direction);
205     var that = this
206 
207     var animation = wx.createAnimation({
208       duration: 250,
209       timingFunction: 'linear',
210     })
211 
212     if (direction == 'right') { //往右滑动
213       animation.translateX(400).rotate(10).opacity(0).step()
214       animation.translateX(0).rotate(0).step()
215     } else if (direction == 'left') { //往左滑动
216       animation.translateX(-400).rotate(-45).opacity(0).step()
217       animation.translateX(0).rotate(0).step()
218     } else if (direction == 'up') { //往上滑动
219       animation.translateY(-400).opacity(0).step()
220       animation.translateY(0).step()
221     } else if (direction == 'down') { //往下滑动
222       animation.translateY(-60).opacity(0).step()
223       animation.translateY(0).opacity(1).rotate(0).step()
224     }
225 
226 
227     if (direction == 'down') { //下拉
228 
229       //index有可能为1
230       if (index > 1) {
231         that.data.info[index - 1 - 1].animation = animation.export(); //下一页,即将移除的卡片动画
232       }
233       that.setData({
234         info: that.data.info,
235       })
236       //数组索引从0开始,所以-1,把上一页数据显示出来,所以-1
237 
238       if (index > 1) {
239         that.data.info[index - 1 - 1].top = marginVerti;
240         that.data.info[index - 1 - 1].left = marginHori;
241       }
242       that.data.info[index - 1].top = marginVerti;
243       that.data.info[index - 1].left = marginHori;
244       index = index == 1 ? 1 : index - 1;
245       setTimeout(function () {
246         that.setData({
247           info: that.data.info,
248           currentTime: index
249         })
250       }.bind(that), 250);
251     } else {
252 
253       that.data.info[index - 1].animation = animation.export(); //下一页,即将移除的卡片动画
254       //开始执行动画
255       that.setData({
256         info: that.data.info
257       });
258 
259       that.data.info[index - 1].top = 10000;
260       that.data.info[index - 1].left = 10000;
261 
262 
263       setTimeout(function () {
264         that.setData({
265           info: that.data.info,
266         })
267       }.bind(that), 250);
268 
269       console.log("index=" + (index + 1));
270       that.setData({
271         currentTime: index + 1
272       });
273 
274       that.getProcess();
275       if ((parseInt(that.data.currentTime)) > that.data.duration) {
276         wx.redirectTo({
277           url: '../result/index',
278         })
279       }
280     }
281   },
282 
283   //左箭头 上一张卡片
284   LastPage: function () {
285     var that = this;
286     console.log("that.data.currentTime" + that.data.currentTime);
287     //测试使用
288     // that.setData({
289     //   currentTime: 1,
290     // });
291     // that.getProcess();
292     if (that.data.currentTime == 1) { //当前是第一页 返回
293       return;
294     }
295     that.data.info.forEach((item, i) => {
296       that.data.info[i].animation = ''; //清除之前的卡片动画
297     });
298 
299     //核心动画
300     var animation = wx.createAnimation({
301       duration: 250,
302       timingFunction: 'linear',
303     })
304     animation.translateY(-60).opacity(0).step();
305     animation.translateY(0).opacity(1).rotate(0).step();
306 
307     //数组索引从0开始,所以-1,把上一页数据显示出来,所以-1
308     that.data.info[that.data.currentTime - 1 - 1].animation = animation.export(); //保存动画到数组
309 
310     that.setData({
311       info: that.data.info
312     })
313 
314     that.data.info[that.data.currentTime - 1 - 1].top = marginVerti; //上一页,恢复到原始的距顶部距离
315     that.data.info[that.data.currentTime - 1 - 1].left = marginHori; //上一页,恢复到原始的距左部距离
316     setTimeout(function () {
317       that.setData({
318         info: that.data.info
319       })
320     }.bind(that), 250);
321 
322     that.setData({
323       currentTime: that.data.currentTime - 1,
324       currentId: that.data.currentId - 1
325     })
326 
327     that.getProcess(); //服务器接口保存卡片浏览当前页码
328 
329   },
330 
331   //右箭头 下一张卡片
332   NextPage: function () {
333 
334     var that = this;
335     //console.log('lastpage=that.data.currentTime=' + that.data.currentTime);
336     var currentTime = parseInt(that.data.currentTime);
337     if (currentTime >= that.data.duration) {
338       wx.redirectTo({
339         url: '../result/index',
340       })
341     }
342     that.removeCard('right', currentTime); //往右移除卡片
343 
344   }
345 })
原文地址:https://www.cnblogs.com/Jessie-candy/p/13450801.html