uniapp微信小程序PC端选择文件(无法使用wx.chooseMessageFile问题)

客户要求通过小程序选择excel文件并读取数据导入,查了一下资料,微信小程序选择文件是调用API【wx.chooseMessageFile】,知道了之后马上开干,没一会做好了,却发现在电脑端微信小程序上面调用此API没有反应,一看就是不支持PC端,怎么办,用户一般都是电脑填好excel,然后导入的。要让用户填好,然后发给好友,再去手机端操作?估计被喷死。查了度娘,看到了有说写一个HTML文件,用【webview】打开,实现上传,了解了一下,开干!!


注意:webview组件账户主体类型为个人的,无法使用!!关闭这个文章的页面吧。


先上html页面效果图


步骤:用户点击上传按钮,判断是否手机端,如果是,则正常走wx.chooseMessageFile ,如果是电脑端,则进入webview打开我们写的html页面,页面使用原生input类型是file。选择文件后,将文件解析成base64并返回至小程序(也可以直接上传至后端,由于我的逻辑代码都是写在了小程序里面,所以把base64返回小程序再上传)。


1:创建一个HTML文件

复制代码创建一个text文件写入,改后缀名为html。代码如下:

html 复制代码
<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="utf-8" />
    <meta name="viewport" content="width=device-width" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1" />
  </head>
  <script type="text/javascript" src="https://res.wx.qq.com/open/js/jweixin-1.6.0.js"></script>
  <script type="text/javascript" src="https://js.cdn.aliyun.dcloud.net.cn/dev/uni-app/uni.webview.1.5.2.js"></script> 
  <body>
    <div class="panel">
      <input type="file" onChange="onFileChange()" id="fileUpload" />
      <div class="upload-view" onClick="onUploadClick()">
        <img class="upload-icon" src="" />
        <div class="upload-title">点击上传</div>
        <div class="upload-tip" id="uploadTip"></div>
      </div>
      <div class="error-tip" id="errorTip"></div>
      <div class="panel-foot">
        <button class="foot-btn-back" onClick="onBack()">返回</button>
      </div>
    </div>
  </body>
  <script type="text/javascript">
    const urlParam = {}

    // 页面生命周期onload事件
    window.onload = () => {
      this.getUrlParam()
      const { tipFileType, fileType, maxSize, sizeUnit = 'M' } = urlParam
      const fileUploadDom = document.getElementById('fileUpload')
      fileType && fileUploadDom.setAttribute('accept', fileType)
      if (maxSize) {
        document.getElementById('uploadTip').innerText = `文件大小限制为${maxSize}${sizeUnit}`
      } else {
        document.getElementById('uploadTip').setAttribute('style', 'display: none')
      }
    }

    // 文件选择完成
    function onFileChange () {
      const file = document.getElementById('fileUpload').files[0]
      if (!file) {
        return
      }
      console.log(file)
      const errorTipDom = document.getElementById('errorTip')
      errorTipDom.innerText = ''
      const { tipFileType, fileType, maxSize, sizeUnit = 'M' } = urlParam
      const { name = '', size = 0 } = file
      if (fileType) {
        const nameSplitArray = name.split('.')
        if (!fileType.split(',').includes('.' + nameSplitArray[nameSplitArray.length - 1])) {
          errorTipDom.innerText = `请上传正确的${tipFileType || ''}文件`
          return
        }
      }
      if (maxSize) {
        const sizeUnitUpper = sizeUnit.toLocaleUpperCase()
        if ((sizeUnitUpper === 'M' && size / 1024 / 1024 > maxSize) 
          || (sizeUnitUpper === 'KB' && size / 1024 > maxSize)
          || (sizeUnitUpper === 'B' && size > maxSize)) {
            errorTipDom.innerText = `上传文件不能超过${maxSize}${sizeUnitUpper}`
            return
        }
      }
      const fileReader = new FileReader()
      fileReader.readAsDataURL(file)
      fileReader.onload = () => {
        const result = (fileReader.result || '').split(',')[1]
        if (result) {
          this.onPostMessage({ code: '0', fileInfo: { base64: result, name: file.name } })
          this.onBack()
        } else {
          errorTipDom.innerText = `读取文件失败`
        }
      }
      fileReader.error = () => {
        errorTipDom.innerText = `读取文件失败`
      }
    }
    
    // 打开选择文件对话框
    function onUploadClick () {
      document.getElementById('fileUpload').click()
    }

    // 给小程序发送消息
    function onPostMessage (value) {
      uni.postMessage({
        data: value
      })
    }

    // 返回小程序,相当于小程序的webview页面触发navigateBack方法,会触发webview页面的卸载事件(onUnload)
    function onBack () {
      wx.miniProgram.navigateBack()
    }

    // 获取url中的参数
    function getUrlParam() {
      const url = location.search
      const theRequest = new Object()
      if (url.indexOf("?") != -1) {
        const str = url.substr(1)
        const strs = str.split("&")
        for (let i = 0; i < strs.length; i++) {
          urlParam[strs[i].split("=")[0]] = decodeURIComponent(strs[i].split("=")[1])
        }
      }
    }
    
  </script>
</html>
<style>
  body {
    margin: 0;
  }
  .panel {
    width: 100vw;
    height: 100vh;
    display: flex;
    flex-direction: column;
    align-items: center;
    position: relative;
  }
  #fileUpload {
    display: none;
  }
  .upload-view {
    margin-top: 30%;
    width: 70%;
    display: flex;
    flex-direction: column;
    align-items: center;
    justify-content: center;
    border: 2px dashed rgba(255, 141, 26, 1);
    border-radius: 10px;
    padding: 20px 0;
  }
  .upload-icon {
    width: 135px;
    height: 135px;
  }
  .upload-title, .upload-tip {
    margin-top: 10px;
  }
  .upload-title {
    color: rgba(255, 141, 26, 1);
    font-size: 20px;
  }
  .upload-tip {
    font-size: 12px;
    color: #333;
  }
  .upload-tip::before {
    content: '*';
    color: #F00;
    margin-right: 3px;
  }
  .error-tip {
    margin-top: 30px;
    color: #F00;
    font-size: 18px;
    font-weight: 550;
  }
  .panel-foot {
    position: absolute;
    bottom: 0;
    left: 0;
    right: 0;
    padding: 10px;
  }
  .foot-btn-back {
    width: 100%;
    height: 45px;
    border-radius: 10px;
    border: 1px solid rgba(229, 229, 229, 1);
    background: transparent;
    color: rgba(166, 166, 166, 1);
    font-size: 17px;
  }
</style>

有个奇怪点,按uni官方文档说,是引入

https://gitcode.net/dcloud/uni-app/-/raw/dev/dist/uni.webview.1.5.6.js

但我在html文件里面不管是链接引入,还是下载到本地引入,都用不了uni的方法,最后换成了引入另一个才行。后面也不纠结研究了,总之能用就行。

html 复制代码
https://js.cdn.aliyun.dcloud.net.cn/dev/uni-app/uni.webview.1.5.2.js

2:调后端接口返回html文件

我这边是通过后端接口返回html页面的,到时候就是webview页面调用后端接口,相当于一个完整链接,打开html页面。好像也可以把html文件放到小程序static里面,然后直接调用吧,没试过。

后端接口:

java 复制代码
    @RequestMapping("/sys/toFileUploadView")
    public void toFileUploadView(HttpServletResponse response) {
        try (
                OutputStream outputStream = response.getOutputStream();
                FileInputStream fileInputStream = new FileInputStream("存放html文件的路径");
        ){
            outputStream.write(IOUtils.toByteArray(fileInputStream));
        } catch (IOException e) {
            log.error("打开pc端文件上传页面失败", e);
        }
    }

小程序上传按钮代码:

javascript 复制代码
onUploadExcelFile () {
	uni.getSystemInfo({
		success: res => {
			if (['windows', 'mac'].includes(res.platform)) {
				// PC端
				// 通过webview页面打开我们写的html文件
				/**
				 * url参数:
				 *  tipFileType: 上传文件不符合时,提示应该上传什么文件
				 *  fileType:支持文件的后缀名
				 *  maxSize:文件大小限制
				 *  sizeUnit:文件大小限制的单位(m, kb, b)
				 */
				// 下面这段url参数意思是:只能上传.xls或.xlsx的文件,当上传文件类型不对时,提示请上传正确的excel文件,然后文件最大只能500kb
				uni.navigateTo({
					url: `/pages/webView/webView?url=${encodeURIComponent('自己的域名或者ip 比如:https://baidu.com' + '/sys/toFileUploadView?tipFileType=excel&fileType=.xls,.xlsx&maxSize=500&sizeUnit=KB')}`
				})
			} else {
				// 手机端
				// 自己写代码调用 wx.chooseMessageFile API
			}
		}
	})
}

3:小程序webview页面

里面有很多注释说明,记得看

javascript 复制代码
<template>
	<view>
		<web-view :src="url" @message="onMessage"></web-view>
	</view>
</template>

<script>
	export default {
		data() {
			return {
				url: '',
				webMessageList: void (0)
			}
		},
		onLoad(options) {
			this.url = decodeURIComponent(options.url)
			if (options.title) {
				uni.setNavigationBarTitle({
					title: decodeURIComponent(options.title)
				})
			}
		},

		/**
		 * 外部页面(html页面)调用wx.miniProgram.navigateBack()后触发
		 */
		onUnload () {
			const pages = getCurrentPages()
			if (pages.length > 1) {
				const { route } = pages[pages.length - 2]
				/**
				 * 判断小程序路由栈里当前webview页面的上一个页面是否是指定页面
				 */
				if (['你的页面'].includes(route)) {
					if ((this.webMessageList || []).length) {
						const { fileInfo = {} } = this.webMessageList[0]
						if (fileInfo.base64) {
							/**
							 * 如果是指定页面并有接收到消息,则取出文件的base64
							 * 这里注意:如果你是打算通过uni.setStroageSync设置缓存,然后指定页面的onShow方法接收处理,
							 * 			我只能告诉你,如果文件太大,缓存还没设置完,
							 * 			此页面已经卸载完成了,然后触发上一个页面的onShow的时候,你取缓存就是空,
							 * 			如果你说加setTimeout延迟获取缓存,那延迟的时间该是多少呢,不可靠!
							 * 			我的做法是设置uni.setStroageSync缓存的同时,设置一个store缓存,值简单点,true就行了
							 * 			设置store缓存肯定会在页面卸载完成前设置完成。
							 * 			对应页面的show方法看文章后面!
							 */
							// store.commit('store缓存key', true)
							// uni.setStroageSync('文件内容缓存key', fileInfo.base64)
							// uni.showLoading({
							// 	mask: true,
							// 	title: '请稍候...'
							// })
						}
					}
				}
			}
		},

		methods: {

			/**
			 * 接收webview打开的外部页面发送过来的消息
			 * 有点不好的是,外部页面发送的消息这里并不能实时获取,必须触发页面返回也就是调用navigateBack等方法才会接收到外部页面的消息
			 * 在我们写的html页面里面调用wx.miniProgram.navigateBack() 会触发webview页面的消息接收
			 * 调用wx.miniProgram.navigateBack()后执行方法顺序是:onMessage -> onUnload
			 * 也就是接收到外部页面消息后,webview页面就马上关闭了
			 */
			onMessage ({ detail }) {
				const { data = [] } = detail || {}
				if (data.length) {
					this.webMessageList = data
				}
			}
		}
	}
</script>

<style>

</style>

4:对应功能页面的show方法

javascript 复制代码
async onShow () {
	if ('取store缓存') {
		const waitTime = (millisecond) => {
			return new Promise((resolve, reject) => {
				setTimeout(() => {
					resolve()
				}, millisecond)
			})
		}
		while (true) {
			if ('store缓存不存在或者不为true') {
				uni.hideLoading()
				break
			}
			const base64 = uni.getStorageSync('文件base64内容缓存key')
			if (base64) {
				// 取到文件内容了,自己处理了...
				store.commit('store缓存key', false)
				uni.hideLoading()
				break
			}
			// 本次循环遍历结束,文件base64内容缓存还没设置完成,等待500毫秒,继续下一次检查
			await waitTime(500)
		}
	}
}

好了,到这就结束了,有问题留言。如果你们碰到小程序上线后webview页面通过后端接口打不开html文件的,检查一下小程序后台管理业务域名这些配置上没有。


码字不易,于你有利,勿忘点赞

相关推荐
快乐的二进制鸭7 小时前
uniapp实现app的pdf预览
pdf·uni-app
盛夏绽放8 小时前
微信小程序地图map全方位解析
微信小程序·小程序
qq_316837759 小时前
uniapp 打包安卓 集成高德地图
uni-app
阿福的工作室9 小时前
uniapp录制语音
uni-app
初尘屿风11 小时前
基于微信小程序的电影院订票选座系统的设计与实现,SSM+Vue+毕业论文+开题报告+任务书+指导搭建视频
vue.js·微信小程序·小程序
貂蝉空大11 小时前
uni-app开发app时 使用uni.chooseLocation遇到的问题
uni-app
林同学++11 小时前
uniapp多端适配
uni-app
kidding72311 小时前
uniapp引入uview组件库(可以引用多个组件)
前端·前端框架·uni-app·uview
qq_3168377511 小时前
uniapp 安卓10+ 选择并上传文件
uni-app
合法的咸鱼11 小时前
uniapp 使用unplugin-auto-import 后, vue文件报红问题
前端·vue.js·uni-app