关于上面提到的 注册表 和为什么通过自定义URL Protocol就可以调用本地的应用程序的介绍在网上已经有很多相关的博客了,我会将一些我觉得比较详细和清晰的文章贴出来,就不详细介绍其中的原理了。
通过自定义URL协议在Web网页中启动本地应用程序
前端网页如何打开一个PC本地应用
但是,上面说到的这些都不是今天的重点,今天的重点是我们如何得知我们有没有成功唤醒本地应用。
如果你曾经再网上搜索过这个问题,那么你得到的最终解决方案大概率是他—— protocolcheck.js
protocolcheck.js根据各种浏览器在通过自定义协议调用本地应用的不同反应,整理了各个浏览器对应的hacks,并通过检测用户所使用的浏览器来调用对应的hacks来检测浏览器是否有成功唤醒应用:
虽然protocolcheck. js已经将刚刚浏览器对应的检测方法封装起来,但这些方法毕竟大多数都是hacks,并且已经有三四年没有进行更新了,随着浏览器的更新肯定会逐渐暴露出许多问题(亲测最新版的chrome和Firefox都已失效)。
显然这不是一个很好的解决方案,正当我想和产品经理去探讨这个需求的可行性时,我想到了百度网盘似乎也有类似的功能,那么百度网盘是怎么实现的呢?
通过百度网盘web端的唤醒功能一段时间的研究,我发现在唤醒过程中他会不断发送同一个请求,在唤醒成功后便不在发送,查看该请求的响应我们可以看到该请求返回了errno,request_id,status三个字段,其中,在唤醒成功时,除了唤醒成功前的最后一个请求的status为1之外,其他请求的status均为2。
而在唤醒失败时,则所有请求的status均为2。
不难判断出,百度云盘是通过web端和客户端直接通讯来判断出是否唤醒成功,但他们是如何进行通讯的呢?这个问题我们先按下不表,但这种方式又让我联想到了EN和QQ邮箱的快速登录功能,当我们启动了 QQ客户端,再去访问 QQ 邮箱登录页,页面会提示我们已经登录了的 QQ,点击即可登录。
响应内容
这个请求返回了登录信息,并且是通过向127.0.0.1:4301获得的,那么问题来了,这个服务是谁提供的呢?
真相大白!由上面的信息我们可以判断出是QQ的客户端开启了一个HTTP Server,当浏览器打开页面时,会向这个Server发起请求来判断用户是否已经登录并返回对应的用户信息。
前提:要唤醒的本地应用是自家开发的。
通过百度云盘和快速登录的案例,我们可以得到唤醒本地应用的最终方案:
问题看似被解决了,但我们其实还需要解决web端与本地应用的HTTP Server的跨域问题:
方案一:通过后台代理,这是百度云盘使用的方式;
方案二:通过jsonp,这是qq邮箱快速登录使用的方式;
方案一成本较高,需要加多一层代理层,并且需要想办法生成一个唯一标识来标记当前浏览器;方案二没有唯一标识的问题,但需要提前对端口号进行约定,并且有可能出现端口号被占用的情况。最终基于成本和端口被占用的概率考虑,我选择了方案二,虽然不算完美,但也算是能够满足大多数用户的需求了。
1、首先,threejs是JavaScript编写的WebGL第三方库,提供了非常多的3D显示功能。打开软件使用语音识别等已有的技术。2、其次,从音频中提取音位信息,并输入给训练好的模型。
3、最后,输出对应的人脸表情即可实现。
我的html结构是这样的class="app-voice-you" voiceSrc="xx.mp3">
class="app-voice-headimg" src="xx.png" />
style="width: 60%" class="app-voice-state-bg">
class="app-voice-state app-voice-pause">
class="app-voice-time app-voice-unread">
1'6"
id="audio_my" src="">
Your browser does not support the audio tag.
核心功能就是语音播放,主要包括了以下几个功能点:
红点表明未听语音,语音听过后,红点会消失
将“未读”状态的样式独立出来,“已读”的时候,把样式删除就行。结合本地存储处理就搞定了。
//this是点击的语音的document
var app_voice_time = this.getElementsByClassName("app-voice-time")[0]
if(app_voice_time.className.indexOf("app-voice-unread") != -1){
//存在红点时,把红点样式删除
app_voice_time.className = app_voice_time.className.replace("app-voice-unread","")
}
第一次听语音,会自动播放下一段语音
这里主要是使用HTML5的audio控件的“语音播放完”事件
语音播放完,找到下一个语音,播放下一个语音
//语音播放完事件(PAGE.audio是audio控件的document)
PAGE.audio.addEventListener('ended', function () {
//循环获取下一个节点
PAGE.preVoice = PAGE.currentVoice
var currentVoice = PAGE.currentVoice
while(true){
currentVoice = currentVoice.nextElementSibling//下一个兄弟节点
//已经到达最底部
if(!currentVoice){
PAGE.preVoice.getElementsByClassName("app-voice-state")[0].className = "app-voice-state app-voice-pause"
return false
}
var voiceSrc = currentVoice.getAttribute("voiceSrc")
if(voiceSrc &&voiceSrc != ""){
break
}
}
if(!PAGE.autoNextVoice){
PAGE.preVoice.getElementsByClassName("app-voice-state")[0].className = "app-voice-state app-voice-pause"
return false
}
PAGE.currentVoice = currentVoice
//获取得到下一个语音节点,播放
PAGE.audio.src = voiceSrc
PAGE.audio.play()
PAGE.Event_PlayVoice()
}, false)
每段语音可以暂停、继续播放、重新播放
这个比较简单,但是也是比较多逻辑。需要变换样式告诉用户,怎样是继续播放/重新播放。
播放中的语音有动画,不是播放中的语音则会停止动画。
这里主要是CSS3动画的应用
.app-voice-pause,.app-voice-play{
height: 18px
background-repeat: no-repeat
background-image: url(../img/voice.png)
background-size: 18px auto
opacity: 0.5
}
.app-voice-you .app-voice-pause{
/*从未播放*/
background-position: 0px -39px
}
.app-voice-you .app-voice-play{
/*播放中(不需要过渡动画)*/
background-position: 0px -39px
-webkit-animation: voiceplay 1s infinite step-start
-moz-animation: voiceplay 1s infinite step-start
-o-animation: voiceplay 1s infinite step-start
animation: voiceplay 1s infinite step-start
}
@-webkit-keyframes voiceplay {
0%,
100% {
background-position: 0px -39px
}
33.333333% {
background-position: 0px -0px
}
66.666666% {
background-position: 0px -19.7px
}
}