从webview页面传值到uniapp中

官方文档已经很详细了,首先在webview页面中引入相关依赖:

  1. <!-- uniapp各平台依赖 -->
  2. <script type="text/javascript">
  3. var userAgent = navigator.userAgent;
  4. if (userAgent.indexOf('AlipayClient') > -1) {
  5. // 支付宝小程序的 JS-SDK 防止 404 需要动态加载,如果不需要兼容支付宝小程序,则无需引用此 JS 文件。
  6. document.writeln('<script src="https://appx/web-view.min.js"' + '>' + '<' + '/' + 'script>');
  7. } else if (/QQ/i.test(userAgent) && /miniProgram/i.test(userAgent)) {
  8. // QQ 小程序
  9. document.write('<script type="text/javascript" src="https://qqq.gtimg.cn/miniprogram/webview_jssdk/qqjssdk-1.0.0.js"><\/script>');
  10. } else if (/miniProgram/i.test(userAgent)) {
  11. // 微信小程序 JS-SDK 如果不需要兼容微信小程序,则无需引用此 JS 文件。
  12. document.write('<script type="text/javascript" src="https://res.wx.qq.com/open/js/jweixin-1.4.0.js"><\/script>');
  13. } else if (/toutiaomicroapp/i.test(userAgent)) {
  14. // 字节跳动小程序 JS-SDK 如果不需要兼容字节跳动小程序,则无需引用此 JS 文件。
  15. document.write('<script type="text/javascript" src="https://s3.pstatp.com/toutiao/tmajssdk/jssdk-1.0.1.js"><\/script>');
  16. } else if (/swan/i.test(userAgent)) {
  17. // 百度小程序 JS-SDK 如果不需要兼容百度小程序,则无需引用此 JS 文件。
  18. document.write('<script type="text/javascript" src="https://b.bdstatic.com/searchbox/icms/searchbox/js/swan-2.0.18.js"><\/script>');
  19. }
  20. </script>
  21. <!-- uni 的 SDK -->
  22. <script type="text/javascript" src="https://js.cdn.aliyun.dcloud.net.cn/dev/uni-app/uni.webview.1.5.2.js"></script>

然后通过uni.postMessage向uniapp传值:

  1. document.addEventListener('UniAppJSBridgeReady', function() {
  2. uni.postMessage({
  3. data: {
  4. action: 'message'
  5. }
  6. });
  7. uni.getEnv(function(res) {
  8. console.log('当前环境:' + JSON.stringify(res));
  9. });
  10. });

在uniapp中监听message:

  1. <template lang="pug">
  2. view
  3. web-view.webview(:src="url" @message="getMessage")
  4. </template>
  5. <script>
  6. export default {
  7. data() {
  8. return {
  9. url: "https://zys201811.boringkiller.cn/shianonline/webview/vod.html?data=123",
  10. }
  11. },
  12. methods: {
  13. getMessage(event) {
  14. let data = event.detail.data
  15. console.log(data);
  16. }
  17. }
  18. }
  19. </script>
  20. <style lang="stylus" scoped>
  21. $webviewHeight = 420rpx
  22. .webview
  23. width 750rpx
  24. height $webviewHeight
  25. </style>

从uniapp中动态传值到webview页面

按照官方文档,从uniapp传值到webview中,只能通过query:

  1. <template lang="pug">
  2. view
  3. <!-- #ifdef APP-PLUS -->
  4. web-view.webview(:src="url")
  5. <!-- #endif -->
  6. </template>
  7. <script>
  8. export default {
  9. data() {
  10. return {
  11. url: "https://zys201811.boringkiller.cn/shianonline/webview/vod.html?data=123",
  12. }
  13. }
  14. }
  15. </script>
  16. <style lang="stylus" scoped>
  17. $webviewHeight = 420rpx
  18. .webview
  19. width 750rpx
  20. height $webviewHeight
  21. </style>

在webview中解析query:

  1. let data = getQuery('data')
  2. console.log(data); // 获取 uni-app 传来的值
  3. // 取url中的参数值
  4. function getQuery(name) {
  5. // 正则:[找寻'&' + 'url参数名字' = '值' + '&']('&'可以不存在)
  6. let reg = new RegExp("(^|&)"+ name +"=([^&]*)(&|$)");
  7. let r = window.location.search.substr(1).match(reg);
  8. console.log(r);
  9. if(r != null) {
  10. // 对参数值进行解码
  11. return decodeURIComponent(r[2]);
  12. }
  13. return null;
  14. }

但是,我们发现,通过向web-view的src中传值,只能传一次,如果参数改变了,没法动态传到webview。

对于这种需要动态传递参数的需求,我们可以使用动态创建webview达到目的,而不是通过webview组件。

实现如下:

  1. <template lang="pug">
  2. view
  3. </template>
  4. <script>
  5. export default {
  6. data() {
  7. return {
  8. url: "https://zys201811.boringkiller.cn/shianonline/webview/vod.html",
  9. }
  10. },
  11. mounted() {
  12. // #ifdef APP-PLUS
  13. var w = plus.webview.create(this.url + '?data=good');
  14. w.show();
  15. setTimeout(() => {
  16. plus.webview.close(w);
  17. setTimeout(() => {
  18. w = plus.webview.create(this.url + '?data=123');
  19. w.show();
  20. }, 1000)
  21. }, 1000)
  22. // #endif
  23. }
  24. }
  25. </script>

以上,通过plus.webview.create创建一个webview,然后显示。如果数据更新了,可以先关闭之前的一个webview,然后重新创建一个,再显示。

也可以直接使用open刷新页面:

  1. // #ifdef APP-PLUS
  2. var w = plus.webview.open(this.url + '?data=good');
  3. setTimeout(() => {
  4. w = plus.webview.open(this.url + '?data=123');
  5. }, 1000)
  6. // #endif

相关API:

  1. // 创建窗口
  2. WebviewObject plus.webview.create( url, id, styles, extras );
  3. // 创建并打开窗口
  4. WebviewObject plus.webview.open( url, id, styles, aniShow, duration, showedCB );
  5. // 显示窗口
  6. void plus.webview.show( id_wvobj, aniShow, duration, showedCB, extras );
  7. // 隐藏窗口
  8. void plus.webview.hide( id_wvobj, aniHide, duration, extras );
  9. // 关闭窗口
  10. void plus.webview.close( id_wvobj, aniClose, duration, extras );

调用webview中的evalJs方法

动态传值还有一种解决方案,就是通过evalJs方法直接调用webview中方法。

具体实现如下:

在模板中,通过ref暴露web-view元素:

  1. <template lang="pug">
  2. web-view(:src="url" ref="wb")
  3. </template>

在mounted生命周期的时候获取此元素:

  1. // #ifdef APP-PLUS
  2. this.wb = this.$refs.wb
  3. // #endif

在需要调用webview中方法的时候使用evalJs

  1. // #ifdef APP-PLUS
  2. this.wb.evalJs(`showAlert(${this.num})`)
  3. // #endif

在webview页面定义对应的方法即可:

  1. function showAlert(num) {
  2. alert(num)
  3. }

从uniapp动态传值,可以使用这种方式。

注意:

  1. 在nvue中,只有通过ref暴露webview节点才能拿到webview本身
  2. 注意evalJs的拼写方式,官方文档是evalJS,但通过ref获取时,S应该为小写

参考资料