针对上两篇微信同声传译语音播报功能,又出现了坑

我又双叒叕来了,自己写的bug,跪着也要改完,我是真的服了

  • 首先,我们来说说是什么问题吧

    1. 上一篇文章的这张图还记得吧,不记得的,我在下面贴出来了;
    1. 我们在长度大于300的时候,根据句号或者逗号进行了切片,然后将这些片段转换成若干个短语音,通过遍历来进行完整播放(不记得的看上一篇吧);
    1. 图中的1和2都是根据句号来切成几个小片段;
    1. 在1正在播放时,切换到2进行播放,等2播放完成了之后,图标会从红色变成绿色,但是语音会接着1暂停的时机继续播放;
  • 原因呢?其实我也不确定,我稍微思考了几个可能得原因

    1. 可能是我写的遍历播放的方法有问题?
    1. 或者是因为我在切换的时候写的是pause(),暂停的方法
    1. 又或者是,wx.createInnerAudioContext()实例里面已经缓存了之前的语音地址
    1. 反正我不确定,但是问题我解决了(鼓掌吧,兄弟们)

在这里插入图片描述

解决方式

在切换图中1和2时,直接把实例清掉,然后再重新注册一个实例,简单且粗暴(我另外又偷偷的进行了一项优化)

// 我在page外定义了两个全局属性,注意:这是写在page外的哈
let radioId = 0    // 存储当前播放id
let radioOldList = []    // 存储当前正在播放的语音列表


// 在onLoad里面进行实例化
this.innerAudioContext = wx.createInnerAudioContext();

// 阅读文字,页面上的点击图标
readText: async function (e) {
    const { item } = e.currentTarget.dataset
    const that = this;
    // 判断当前id和上一次存储的id是不是相同
    if (item.id == radioId) {
    // 相同就调用pause()暂停方法,再次点击会接着之前暂停处播放
      that.innerAudioContext.pause()
    } else {
    // 不相同就调用stop()停止方法
      that.innerAudioContext.stop()
      // 同时将实例清除
      that.innerAudioContext = null
      // 重新创建一个实例
      that.innerAudioContext = wx.createInnerAudioContext()
    }
    that.data.questionList.forEach(el => {
        if (el.id == item.id) {
          el.type = 'pause'
        } else {
          el.type = 'play'
        }
    });
    that.setData({
      questionList: that.data.questionList,
      playItem: item
    })
    // 当前id和上一次存储的id相同时,跳过语音转换步骤,直接使用之前的存储的语音列表
    if (item.id == radioId) {
      that.readStart(radioOldList, item.id)
      return
    }
    let list = splitStringByLength(item.audiodialogue, 300)
    let radioList = list.map(el => {
      return new Promise(resolve => {
        plugin.textToSpeech({
          lang: "zh_CN",
          tts: true,
          content: el,
          success: function (res) {
            resolve(res.filename)
          },
          fail: function (res) {
            wx.showToast({
              title: '语音转换失败',
            })
          }
        })
      })
    })
    Promise.all(radioList).then(res => {
      that.readStart(res, item.id)
    })
  },
  // 开始阅读
  readStart: async function (radioList, id) {
    const that = this
    // 将当前播放的语音列表存储起来
    radioOldList = radioList
    for (let text of radioList) {
      this.innerAudioContext.src = text;
      this.innerAudioContext.onPlay(() => {
        console.log('开始播放当前片段', 'onPlay');
      });
      this.innerAudioContext.onError((err) => {
        console.error('音频播放出错', err);
      });
      this.innerAudioContext.onEnded(async () => {
        // 如果是最后一个片段,这里可以结束,否则不需要await
        if (text === radioList[radioList.length - 1]) {
          that.data.questionList.forEach(el => {
            if (el.id == id) {
              el.type = 'play'
            }
          })
          that.setData({
            questionList: that.data.questionList,
          })
          that.innerAudioContext.stop()
          return
        }
      });
      // 将当前播放的id存储起来
      radioId = id
      // 确保前一个音频播放结束后再播放下一个
      await new Promise(resolve => {
        this.innerAudioContext.onEnded(resolve);
        this.innerAudioContext.play();
      });
    }
  },
关键代码:
  • 存储上一次的播放id和播放列表
    在这里插入图片描述

  • readText方法内,判断当前播放的id是否和存储的相同,相同就跳过语音转换步骤

  • 如果不相同,先停止当前的播放并清除实例后,重新创建实例

在这里插入图片描述

总结:

全是坑,一个坑填完还有另外好多坑排着队等着,咋整?站着填不完,那就跪着填吧。就这样吧,累了

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/650426.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

dp + 计数,1954D - Colored Balls

一、题目 1、题目描述 2、输入输出 2.1输入 2.2输出 3、原题链接 Problem - 1954D - Codeforces 二、解题报告 1、思路分析 本题前置题目: 1953. 你可以工作的最大周数 通过前置题目可以知道如何计算两两不同数对序列的最大长度 我们记最大数量为ma&#xf…

景源畅信电商:做抖店的成本高吗?

在当今数字化时代,抖音不仅仅是一个分享短视频的平台,更是一个充满商机的市场。随着抖音用户量的激增,越来越多的商家和个人开始关注在抖音上开设店铺,即所谓的“抖店”。那么,做抖店的成本高吗?这个问题困扰着许多初…

剖析【C++】——类与对象(上)超详解——小白篇

目录 1.面向过程和面向对象的初步认识 1.面向过程(Procedural Programming) 2.面向对象(Object-Oriented Programming) 概念: 特点: 总结 2.C 类的引入 1.从 C 语言的结构体到 C 的类 2.C 中的结构…

InTouch历史报警、历史事件按时段查询,导出

简介:本插件基于上位机组态InTouch的历史报警、操作记录而开发 适用InTouch版本:不限 适用Windows系统:不限 适用数据库:SQL Server 标记名点数:不限 配套软件安装:Excel、WPS、SQL Server 功能&…

【C++初阶】—— 类和对象 (上)

📝个人主页🌹:EterNity_TiMe_ ⏩收录专栏⏪:C “ 登神长阶 ” 🌹🌹期待您的关注 🌹🌹 类和对象 1. 初步认识C2. 类的引入3. 类的定义声明和定义全部放在类体中声明和定义分开存放 4.…

内网横向移动小补充 --->PTK

大家别急,我的基于资源的约束性委派攻击还在写,这个东西一时半会讲不清楚,所以我在这里先来补充一点横向移动以前没说好的东西!!! 在更啦,别催啦~~~~ 还记得我之前在内网渗透里面讲过这个PTK&a…

二分例题(D.负重越野,I.路径规划)

这两天的训练赛都有一道二分的题&#xff0c;但是都没往二分上面想&#xff0c;同样不知道怎么二分。 D. Fast and Fat 思路 二分的关键也就是check函数怎么写了&#xff0c;求队伍最大速度&#xff0c;可以分为速度>mid和<mid两部分&#xff0c;再判断&#xff0c;能不…

乡村振兴的乡村旅游新模式:挖掘乡村旅游资源,创新旅游开发方式,打造乡村旅游新品牌,助力美丽乡村建设

目录 一、引言 二、乡村旅游资源挖掘 1、自然景观资源 2、人文历史资源 3、农业产业资源 三、旅游开发方式创新 1、多元化旅游产品 2、体验式旅游模式 3、智慧旅游建设 四、乡村旅游新品牌打造 1、品牌定位与策划 2、品牌传播与推广 3、品牌维护与提升 五、助力美…

【面试必看】Java并发

并发 1. 线程 1. 线程vs进程 进程是程序的一次执行过程&#xff0c;是系统运行程序的基本单位&#xff0c;因此进程是动态的。 系统运行一个程序即是一个进程从创建&#xff0c;运行到消亡的过程。在 Java 中&#xff0c;当我们启动 main 函数时其实就是启动了一个 JVM 的进…

英语新概念2-回译法-lesson16

第一次回译 if you ___ your car on a wrong place, the traffic police man will find you quickly. If he do not give you the ticket,you are lucky.However,the ___ not all like this,The police man is __ sometimes.I had a holiday in Sweden, I found a ___ in my c…

QT安装和配置[安装注意点][QT找不到python27.dll][缩小空间]

安装注意点 本文摘录于&#xff1a;https://blog.csdn.net/Python_0011/article/details/131699443只是做学习备份之用&#xff0c;绝无抄袭之意&#xff0c;有疑惑请联系本人&#xff01; 双击"qt-online-installer-windows-x64-4.8.0.exe"文件后输入账号选择如下控…

spring boot3整合邮件服务实现邮件发送功能

⛰️个人主页: 蒾酒 &#x1f525;系列专栏&#xff1a;《spring boot实战》 目录 内容概要 开通服务 依赖引入 配置属性 创建邮件发送工具类 测试 最近发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家…

小白不知道怎么投稿?记住这个好方法

作为一名单位信息宣传员,我最初踏上这条道路时,满心憧憬着通过文字传递我们单位的精彩瞬间,让社会听见我们的声音。然而,理想与现实之间的距离,却在一次次邮箱投稿的石沉大海中渐渐清晰。那时的我,像所有“小白”一样,以为只要用心撰写稿件,通过电子邮件发给各大媒体,就能收获满…

大规模团队的数据库开发,如何用OceanBase工具快速建立企业级账号体系

前言 为了让数据库开发的安全性与可靠性得以充分保障&#xff0c;数据库开发工具的管控能力显得尤为关键。构建一个健全的账号体系&#xff0c;能够协助开发团队实现对数据库开发工具的全方位管控&#xff0c;从而有效防范各类数据安全隐患&#xff0c;确保数据库开发的顺利进…

BFS解决最短路问题(详解)

目录 BFS简介 && 框架&#xff1a; 一.二叉树的最小深度 二&#xff1a;迷宫中里入口最近的出口&#xff1a; 三.最小基因变化: 四&#xff1a;单词接龙&#xff1a; ​五&#xff1a;为高尔夫比赛砍树&#xff1a; BFS简介 && 框架&#xff1a; 说到BFS…

【JAVASE】接口(上)

一&#xff1a;接口的概念 在现实生活中&#xff0c;接口的例子比比皆是&#xff0c;比如&#xff1a;笔记本上上的USB接口。 电脑上的USB口上可以插:U盘、鼠标、键盘等。 电源插座插孔上可以插入&#xff1a;电脑、电视机等。 通过以上例子可以看出&#xff1a;接口就是公共…

stm32学习-AD单通道

配置流程 初始化 1.开启RCC时钟&#xff08;ADC、GPIO和ADCCLK的预分频器的时钟&#xff09; void RCC_ADCCLKConfig(uint32_t RCC_PCLK2); 作用&#xff1a;配置预分频器系数。&#xff08;可以对APB2的72MHz时钟选择2/4/6/8分频&#xff0c;输入到ADCCLK中&#xff09; …

连续数组 ---- 前缀和

题目链接 题目: 分析: 如果我们直接做这道题, 有点麻烦如果我们将所有的0都变成1, 那么找到数量相同的0和1的最长连续子数组, 就是找最长连续子数组的和为0, 那么就和之前的题目<找到和为k的子数组>有点像, 需要用前缀和哈希表来完成在[0,i-1]区间找到前缀和 sum[i] - …

Android 逆向学习【1】——版本/体系结构/代码学习

#Android 历史版本 参考链接&#xff1a;一篇文章让你了解Android各个版本的历程 - 知乎 (zhihu.com) 三个部分&#xff1a;api等级、版本号、代号&#xff08;这三个东西都是指的同一个系统&#xff09; API等级&#xff1a;在APP开发的时候写在清单列表里面的 版本号&…

2024年短视频评论区批量爬取采集软件

一、背景说明 前言 评论区引流&#xff0c;顾名思义&#xff0c;是通过在视频下方进行留言评论、回复评论&#xff0c;吸引用户的注意&#xff0c;从而和你的账号产生互动、交易。比如&#xff0c;在一个关于健身的视频下方&#xff0c;留言分享自己的健身经验或者提出问题。…