先从movable-view开始说起吧. movable-view是小程序自定义的组件.其描述为:”可移动的视图容器,在页面中可以拖拽滑动”. 官方文档地址:https://mp.weixin.qq.com/debug/wxadoc/dev/component/movable-view.html.

值得注意的是文档中有一段备注: “当movable-view小于movable-area时,movable-view的移动范围是在movable-area内;当movable-view大于movable-area时,movable-view的移动范围必须包含movable-area(x轴方向和y轴方向分开考虑)”. 也就是说父容器movable-area是可以比子容器movable-view小的,但是子容器的移动范围必须包括父容器.

先看官方实例代码:

 1 <view class="section">
 2   <view class="section__title">movable-view区域小于movable-area</view>
 3   <movable-area style="height: 200px; 200px;background: red;">
 4     <movable-view style="height: 50px;  50px; background: blue;" x="{{x}}" y="{{y}}" direction="all">
 5     </movable-view>
 6   </movable-area>
 7   <view class="btn-area">
 8     <button size="mini" bindtap="tap">click me to move to (30px, 30px)</button>
 9   </view>
10   <view class="section__title">movable-view区域大于movable-area</view>
11   <movable-area style="height: 100px; 100px;background: red;" direction="all">
12     <movable-view style="height: 200px;  200px; background: blue;">
13     </movable-view>
14   </movable-area>
15 </view>

这里面有个错误,应该是编写人的一点小失误吧. 第二个movable-area的属性direction应该写在movable-view上. 

1  <movable-area style="height: 100px; 100px;background: red;" >
2     <movable-view style="height: 200px;  200px; background: blue;" direction="all">
3     </movable-view>
4   </movable-area>

看下效果:

1) 当movable-view区域小于movable-area时,子容器movable-view只能在父容器内移动.  下图的效果是设置了属性 out-of-bounds=”true”的效果. out-of-bounds可以染子容器到达父容器边界时有个超出边界然后回弹的动画. 并不是真正能让子容器移动到父容器以外.

2) 当movable-view区域大于movable-area时,子容器移动的范围必须包括父容器. 

微信小程序movable-view移动图片和双指缩放-风君雪科技博客                                                  微信小程序movable-view移动图片和双指缩放-风君雪科技博客

第二种情况中,把父容器看做手机屏幕可视区域,子容器看做要查看的长图,大图. 就可以实现拖动查看图片的效果. 如果图片时动态加载的,不是固定的图片,就要兼容图片宽高小于屏幕可视宽高和图片宽高大于可视屏幕宽高的可能性,也就是要考虑到以上两种情况.

我们要在movable组件加载的同时设置好movable-view的宽高,因为movable组件加载成功后再去改变movable-view的大小,可移动区域是不会变的. 我们可以通过页面中要查看的图片的onload事件中获取图片宽高(目前我只发现bindload事件能获取到图片宽高),然后存储起来imgWidth和imgHeight. 当用户点击图片时,在bindtap事件中设置好movable-view的宽高,同时将movable-area的弹窗wx;if设置为true.

1   <!-- 要查看的图片列表 -->
2           <view class="flex-wrap flex-pic">
3               <view class="picList">            
4                   <image  wx:for="{{item.imglist}}" wx:for-item="itemImg" wx:key="*this" id="{{'rfnd_'+index}}" bindload="imageOnload" src="{{ itemImg}}" bindtap="showResizeModal" data-src="{{itemImg}}"></image>           
5               </view>
6           </view>

因为要查看的是一个图片列表, 我用了一个数组去存储每个图片的宽高,然后通过图片id来关联 

 1   /**
 2    * 加载图片
 3    */
 4   imageOnload:function(e){
 5     var id = e.currentTarget.id
 6     this.data.imgIdList[id] = {
 7       e.detail.width,
 8       height:e.detail.height
 9     }
10     
11   },

 模板页面

 1 <!--components/resizePicModal/resizePicModal.wxml-->
 3 <template name="resizePic">
 4 
 5   <scroll-view class="backdrop"> 
 6     <view class="close-icon" bindtap="closeResizeModal"> 
 7       取消预览
 8     </view>
 9     <movable-area style="100%;height:100%;"  >
10         <movable-view direction="all" 
11         out-of-bounds="true"  x="{{img.x}}" y="{{img.y}}" >
12         <image  mode="widthFix" class="dtl-img" src="{{img.currentSrc}}"></image>
13         </movable-view>
14         
15     </movable-area> 
16   </scroll-view> 
17  </template>
 1   /**
 2    * 打开弹窗
 3    */
 4   showResizeModal: function (e) {
 5     var src = e.currentTarget.dataset.src;
 6     var x = 0
 7     var y =0
 8     try {
 9       var width = this.imgIdList[e.currentTarget.id].width; //图片原宽
10       var height = this.imgIdList[e.currentTarget.id].height; //图片原高
11      
12       //小程序默认固定宽320px,获取top和left值,使图片居中显示
13       height = height * (320 / width);
14       width = 320;
15     
16       x = (app.windowWidth - width) / 2 
17       y = (app.windowHeight - height) / 2
18 
19     } catch (e) { }
20     var img = {
21       x: x,
22       y: y,
26 currentSrc: src, 27 }; 28 this.setData({ img: img, isCheckDtl: true }); 29 30 },

 部分CSS代码

.backdrop{
  background: rgba(0, 0, 0, 1);
  width:100%;
  height: 100%;
  position: fixed;
  top:0;
  left:0;
}

以上基本上可以完成一个点击查看图片的需求.

然而如果再支持双指缩放的话,movable-view实现不了.我暂没想出来怎么实现,如果有人知道,希望能够指点迷津. 主要原因是因为还是我上文提到的那句话:”movable组件加载成功后再去改变movable-view的大小,可移动区域是不会变的“.缩放后图片大小肯定会改变的. 缩小还好,一旦放大,可移动区域还是原来的不会改变.想象一下,如果一张宽度刚刚好时屏幕可视宽度的图片,放大后,这张图片就只能在屏幕可视宽度windowWidth的范围中移动,看不到左也看不到右边超出的部分.

所以如果既要可移动图片又要可缩放,就不能用movable-view组件了,自己写个吧. 原来bindtouchmove会触发页面的滚动条,但是现在微信好像已经修复了这个BUG,我今天在真机上测试没有出现这个问题. 

 自定义控件resizePicModal.wxml:

 1 <!-- 缩放 -->
 2 <template name="resizePic">
 3   <scroll-view class="backdrop"  catchtouchmove="bindTouchMove" catchtouchend="bindTouchEnd" bindtouchstart="bindTouchStart" > 
 4     <view class="close-icon" bindtap="closeResizeModal"> 
 5       取消预览
 6     </view>
 7        <image catchtouchmove="bindTouchMove" bindtouchend="bindTouchEnd" bindtouchstart="bindTouchStart"  
 8       style=" transform: scale({{img.baseScale}}); position:absolute; top:{{img.top}}px; left:{{img.left}}px; "
 9       mode="widthFix" class="dtl-img" src="{{img.currentSrc}}"></image> 
10 
11    
12   </scroll-view> 
13  </template>

JS: resizePicModal.js

  1 /**
  2  * 使用方法:
  3  * 1) WXHTML要缩放的图片 必须 传入 src 以及绑定 bindtap事件,
  4  *    e.g:     
  5  *    <image bindtap="toggleDtl" data-src="{{item}}" wx:for="{{productCard.arrImg}}" wx:key="*this" src="{{item}}" style="100%" mode="widthFix"></image>
  6  * 2) WXHTML 要引入Modal模板(isCheckDtl无需再定义):
  7  *      <view wx:if="{{isCheckDtl}}">
  8  *        <import src="/components/resizePicModal/resizePicModal.wxml"/>
  9  *        <template is="resizePic" data="{{img}}"></template>
 10  *      </view>
 11  * 3) JS页面要引入JS文件,覆盖当前页面的事件:
 12  *    var resizePicModalService =  require ('../../components/resizePicModal/resizePicModal.js')
 13  *    var resizePicModal = {}
 14  * 4) 在onLoad事件中,实例化ResizePicModal
 15  *        resizePicModal = new resizePicModalService.ResizePicModal()
 16  */
 17 var app = getApp()
 18 let modalEvent = {
 19   distanceList: [0, 0],//存储缩放时,双指距离.只有两个数据.第一项为old distance.最后一项为new distance
 20   disPoint: { x: 0, y: 0 },//手指touch图片时,在图片上的位置
 21   imgIdList:{},
 22   
 23   /**
 24    * 打开弹窗
 25    */
 26   showResizeModal: function (e) {
 27     var src = e.currentTarget.dataset.src;
 28     var x = 0
 29     var y =0
 30     try {
 31       var width = this.imgIdList[e.currentTarget.id].width; //图片原宽
 32       var height = this.imgIdList[e.currentTarget.id].height; //图片原高
 33      
 34       //小程序固定宽320px
 35       height = height * (320 / width);
 36       width = 320;
 37 
 38       x = (app.windowWidth - width) / 2 //> 0 ? (app.windowWidth - width) / 2 : 0;
 39       y = (app.windowHeight - height) / 2// > 0 ? (app.windowHeight - height) / 2 : 0;
 40 
 41     } catch (e) { }
 42     var img = {
 43       top: y,
 44       left: x,
 45       x: x, y: y,
 46        '100%',
 47       baseScale: 1,
 48       currentSrc: src,
 49     };
 50     this.setData({ img: img, isCheckDtl: true });
 51     
 52   },
 53   /**
 54    * 关闭弹窗
 55    */
 56   closeResizeModal:function(){
 57     this.setData({  isCheckDtl: false })
 58   },
 59   /**
 60    * 加载图片
 61    */
 62   imageOnload:function(e){
 63     var id = e.currentTarget.id
 64     this.imgIdList[id] = {
 65       e.detail.width,
 66       height:e.detail.height
 67     }
 68     
 69   },
 70   /**
 71    * bindtouchmove
 72    */
 73   bindTouchMove: function (e) {
 74     if (e.touches.length == 1) {//一指移动当前图片
 75       this.data.img.left = e.touches[0].clientX - this.disPoint.x
 76       this.data.img.top = e.touches[0].clientY - this.disPoint.y
 77 
 78       this.setData({ img: this.data.img })
 79     }
 80   
 82     if (e.touches.length == 2) {//二指缩放
 83       var xMove = e.touches[1].clientX - e.touches[0].clientX
 84       var yMove = e.touches[1].clientY - e.touches[0].clientY
 85       var distance = Math.sqrt(xMove * xMove + yMove * yMove);//开根号
 86       this.distanceList.shift()
 87       this.distanceList.push(distance)
 88       if (this.distanceList[0] == 0) { return }
 89       var distanceDiff = this.distanceList[1] - this.distanceList[0]//两次touch之间, distance的变化. >0,放大图片.<0 缩小图片
 90       // 假设缩放scale基数为1:  newScale = oldScale + 0.005 * distanceDiff
 91       var baseScale = this.data.img.baseScale + 0.005 * distanceDiff
 92       if(baseScale>0){
 93         this.data.img.baseScale = baseScale
 94         var imgWidth = baseScale * parseInt(this.data.img.imgWidth) 
 95         var imgHeight = baseScale * parseInt(this.data.img.imgHeight)
 96         this.setData({ img: this.data.img })
 97       }else{
 98         this.data.img.baseScale = 0
 99         this.setData({ img: this.data.img })
100       }
101       
102     }
103 
104   },
105   /**
106    * bindtouchend
107    */
108   bindTouchEnd: function (e) {
109     if (e.touches.length == 2) {//二指缩放
110       this.setData({ isCheckDtl: true })
111     }
112   },
113   /**
114    * bindtouchstart
115    */
116   bindTouchStart: function (e) {
117     this.distanceList = [0, 0]//回复初始值
118     this.disPoint = { x: 0, y: 0 }
119     if (e.touches.length == 1) {
120       this.disPoint.x = e.touches[0].clientX - this.data.img.left
121       this.disPoint.y = e.touches[0].clientY - this.data.img.top
122     }
123 
124   }
125 }
126 
127 function ResizePicModal(){
128   let pages = getCurrentPages()
129   let curPage = pages[pages.length - 1]
130   Object.assign(curPage, modalEvent)//覆盖原生页面事件
131   this.page = curPage
132   curPage.resizePicModal = this
133   return this
134 }
135 module.exports = {
136   ResizePicModal
137 }

业务页面wxml:引入自定义控件模板

 1   <view class="flex-wrap flex-pic">
 2               <view class="picList">            
 3                   <image  wx:for="{{item.imglist}}" wx:for-item="itemImg" wx:key="*this" id="{{'rfnd_'+index}}" bindload="imageOnload" src="{{ itemImg}}" bindtap="showResizeModal" data-src="{{itemImg}}"></image>           
 4               </view>
 5   </view>
 6 <!-- 缩放 -->
 7  <view wx:if="{{isCheckDtl}}">
 8    <import src="/components/resizePicModal/resizePicModal.wxml"/>
 9   <template is="resizePic" data="{{img}}"></template>
10  </view>

业务页面js,引用js文件,实例化resizePicModal 

1 var that
2 var resizePicModal = {}
3 var app = getApp()
4 var resizePicModalService = require('../../components/resizePicModal/resizePicModal.js')

 1  /**
 2    * 生命周期函数--监听页面加载
 3    */
 4   onLoad: function (options) {
 5     that = this 8     resizePicModal = new resizePicModalService.ResizePicModal()
}

 新手一枚,有未完善支出谢谢指点出来.