forked from xiangyuecn/Recorder
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnative-config.js
218 lines (170 loc) · 7.05 KB
/
native-config.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
/*
app-support/app.js中Native测试用的配置例子,用于调用App的原生接口来录音
【本文件的作用】:实现app.js内Native中Config的四个标注为需实现的接口(这几个接口是app-native-support.js需要的),提供本文件可免去修改app.js源码。
本例子提供了一个JsBridge实现,并且本文件所在目录内还有Android和IOS的demo项目,app原生层已实现相应的接口,copy源码改改就能用。
此文件需要在app.js之前进行加载,【注意】【如果你App原生层实现不是用的demo中提供的接口文件,需自行重写本配置代码】
支持在iframe中使用(含跨域)。
https://github.com/xiangyuecn/Recorder
*/
(function(){
"use strict";
/**【需修改】请使用自己的js文件目录,不要用github的不稳定。RecordApp会自动从这个目录内进行加载相关的实现文件、Recorder核心、编码引擎,会自动默认加载哪些文件,请查阅app.js内所有Platform的paths配置;如果这些文件你已手动全部加载,这个目录配置可以不用**/
window.RecordAppBaseFolder=window.PageSet_RecordAppBaseFolder||"https://xiangyuecn.gitee.io/recorder/src/";
//Install Begin:在RecordApp准备好时执行这些代码
window.OnRecordAppInstalled=window.Native_RecordApp_Config=function(){
window.Native_RecordApp_Config=null;
window.IOS_Weixin_RecordApp_Config&&IOS_Weixin_RecordApp_Config();//如果ios-weixin-config.js也引入了的话,也需要初始化
var App=RecordApp;
var CLog=App.CLog;
var platform=App.Platforms.Native;
var config=platform.Config;
CLog("native-config init");
/******JsBridge简易版*******/
/*JsBridge名称定义
Android为addJavascriptInterface注入到全局,提供RecordAppJsBridge.request(jsonString)
IOS为userContentController绑定对象实现(此对象仅供识别,由于messageHandlers没有同步返回值(同步能实现异步,异步只能异步到死),因此不能参与数据交互,数据交互使用重写prompt实现)
*/
var JsBridgeName="RecordAppJsBridge";
var AppJsBridgeRequest=window.AppJsBridgeRequest=function(action,args,call){
var p=window.top;
var pfn=0;
try{//让顶层window去处理,如果跨域无权限就算了
pfn=p.AppJsBridgeRequest;
}catch(e){
CLog("检测到跨域iframe,AppJsBridgeRequest将由Native通过执行postMessage转发来兼容数据的返回",3);
};
if(pfn && pfn!=AppJsBridgeRequest){
return pfn(action,args,call);
};
args||(args={});
var callback="";
if(call){
callback=Callbacks.length+"";
Callbacks.push(call);
};
var json={status:"",message:"",callback:callback,value:null};//接口调用返回数据格式标准
var data=JSON.stringify({action:action,args:args,callback:callback});
//APP差异化处理
var val="";
if(window[JsBridgeName]){//Android
val=window[JsBridgeName].request(data);
}else if( ((window.webkit||{}).messageHandlers||{})[JsBridgeName+"IsSet"] ){//IOS
val=prompt(data);
}else{//非App环境
json.message="非app,不能调用接口";
};
val=val&&JSON.parse(val)||json;
return val;//同步返回结果,异步返回会走AppJsBridgeRequest.Call
};
var Callbacks=[""];
//app异步回调
AppJsBridgeRequest.Call=function(msg){
if(Callbacks[msg.callback]){
Callbacks[msg.callback](msg);
Callbacks[msg.callback]=null;
}else{
//NOOP
};
};
//app事件回调
AppJsBridgeRequest.Record=function(pcmDataBase64,sampleRate){
NativeRecordReceivePCM(pcmDataBase64,sampleRate);
};
//尝试注入顶层window,用于接收Native回调数据,此处特殊处理一下,省得跨域的iframe无权限
//本JsBridge因为多了异步回调方法,因此对NativeRecordReceivePCM封装了一层,没有用它自带的postMessage转发兼容实现,其实这里的代码和它的代码是一样的。
try{
window.top.AppJsBridgeRequest=AppJsBridgeRequest;
}catch(e){
var tipsFn=function(){
CLog("检测到跨域iframe,AppJsBridgeRequest无法注入到顶层,已监听postMessage,Native通过执行postMessage转发来兼容数据返回",3);
if(window.parent!=window.top){
CLog("RecordApp Native Config示例不支持跨域iframe超过1层,因为没有处理中间的iframe的window的postMessage转发",1);
};
};
setTimeout(tipsFn,8000);
tipsFn();
addEventListener("message",function(e){//纯天然,无需考虑origin
var data=e.data;
if(data&&data.type=="AppJsBridgeRequest"){
var action=data.action;
data=data.data;
if(action=="Call"){
AppJsBridgeRequest.Call(data);
}else if(action=="Record"){
AppJsBridgeRequest.Record(data.pcmDataBase64, data.sampleRate);
}else{
CLog("AppJsBridgeRequest未知postMessage:"+action,3);
};
};
});
};
/******JsBridge简单实现 End*******/
/*********实现app.js内Native中Config的接口*************/
config.IsApp=function(call){
/*识别为app环境*/
if(window[JsBridgeName]||((window.webkit||{}).messageHandlers||{})[JsBridgeName+"IsSet"]){
call(true);
return;
};
call(false);//非app
};
config.JsBridgeRequestPermission=function(success,fail){
//异步接口,录音权限检测,返回值int:1支持,2不支持,3用户拒绝
AppJsBridgeRequest("recordPermission",{},function(json){
if(json.status!="success"){
fail(json.message);
return;
};
if(json.value==1){
success();
}else if(json.value==3){
fail("用户拒绝了录音权限",true);
}else{
fail("不支持录音");
};
});
};
var aliveInt;
config.JsBridgeStart=function(set,success,fail){
//异步接口,开始录音
AppJsBridgeRequest("recordStart",{param:set},function(json){
if(json.status!="success"){
fail(json.message);
return;
};
success();
//激活定时心跳,如果超过10秒未发心跳,app将会停止录音,防止未stop导致泄露
aliveInt=setInterval(function(){
//同步接口
var val=AppJsBridgeRequest("recordAlive");
//console.log("心跳已发出:"+JSON.stringify(val));
},5000);
});
};
config.JsBridgeStop=function(success,fail){
clearInterval(aliveInt);//关掉定时心跳
//异步接口,结束录音
AppJsBridgeRequest("recordStop",{},function(json){
if(json.status!="success"){
fail(json.message);
return;
};
success();
});
};
/*********接口实现END*************/
};
//Install End
//如果已加载RecordApp,手动进行触发
if(window.RecordApp){
OnRecordAppInstalled();
};
})();
console.error("【注意】本网站正在使用RecordApp的native-config.js测试用的配置例子,这个配置如果要使用到你的网站,需要自己重写或修改后才能使用");
//别的站点引用弹窗醒目提示
if(!/^file:|:\/\/[^\/]*(jiebian.life|git\w+.io)(\/|$)/.test(location.href)
&& !localStorage["DisableAppSampleAlert"]
&& !window.AppSampleAlert){
window.AppSampleAlert=1;
alert("【注意】当前网站正在使用RecordApp测试用的配置例子*.config.js,需要自己重写或修改后才能使用");
};