diff --git a/src/android/Trtc.java b/src/android/Trtc.java deleted file mode 100644 index 3557f83..0000000 --- a/src/android/Trtc.java +++ /dev/null @@ -1,55 +0,0 @@ -package com.chuwa.cordova.trtc; - -import android.app.Activity; -import android.content.Intent; - -import com.tencent.trtc.videocall.VideoCallingEnterActivity; -import com.tencent.trtc.videocall.VideoCallingActivity; - -import org.apache.cordova.CordovaInterface; -import org.apache.cordova.CordovaPlugin; -import org.apache.cordova.CallbackContext; - -import org.apache.cordova.CordovaWebView; -import org.json.JSONArray; -import org.json.JSONException; -import org.json.JSONObject; - - -public class Trtc extends CordovaPlugin { - - static final String ACTION_SHOW_CREATE_PAGE = "showCreatePage"; // for test - static final String ACTION_JOIN_CHANNEL = "joinChannel"; - - @Override - public boolean execute(String action, JSONArray args, CallbackContext callbackContext) throws JSONException { - Activity activity = this.cordova.getActivity(); - -// if (ACTION_SHOW_CREATE_PAGE.equals(action)) { -// Intent intent = new Intent(activity, VideoCallingEnterActivity.class); -// cordova.startActivityForResult(this, intent, 666); -// } else - if (ACTION_JOIN_CHANNEL.equals(action)) { - Intent intent = new Intent(activity, VideoCallingActivity.class); - JSONObject arg = args.getJSONObject(0); - String ROOM_ID = arg.getString("ROOM_ID"); - String USER_ID = arg.getString("USER_ID"); - int SDK_APP_ID = arg.getInt("SDK_APP_ID"); - String USER_SIG = arg.getString("USER_SIG"); - intent.putExtra("room_id", ROOM_ID); - intent.putExtra("user_id", USER_ID); - intent.putExtra("sdk_app_id", SDK_APP_ID); - intent.putExtra("user_sig", USER_SIG); - cordova.startActivityForResult(this, intent, 666); - } - - return false; - } - - @Override - public void initialize(CordovaInterface cordova, CordovaWebView webView) { - super.initialize(cordova, webView); - } - - -} diff --git a/src/android/java/com/tencent/trtc/CordovaEventKit.java b/src/android/java/com/tencent/trtc/CordovaEventKit.java new file mode 100644 index 0000000..5e7ef16 --- /dev/null +++ b/src/android/java/com/tencent/trtc/CordovaEventKit.java @@ -0,0 +1,45 @@ +package com.tencent.trtc; + +import org.apache.cordova.CordovaPlugin; +import org.json.JSONArray; +import org.json.JSONObject; + +public class CordovaEventKit { + public static CordovaEventKit kit = null; + private CordovaPlugin plugin; + + + private CordovaEventKit(CordovaPlugin plugin) { + this.plugin = plugin; + } + + public static void init(CordovaPlugin plugin){ + if(CordovaEventKit.kit == null){ + CordovaEventKit.kit = new CordovaEventKit(plugin); + } + } + + + public void fireEvent(String event, JSONArray obj){ + this.fireEvent(event,obj == null ? null obj.toString()); + } + + public void fireEvent(String event, JSONObject obj){ + this.fireEvent(event,obj == null ? null :obj.toString()); + } + + public void fireEvent(String event, String obj){ + if (plugin == null || event == null || obj == null) { + return; + } + event = event.replace('\'','_'); + String format = "window.cordova.plugin.trtc.fireEvent('%s',%s);"; + final String js = String.format(format,event, obj); + plugin.cordova.getActivity().runOnUiThread(new Runnable() { + @Override + public void run() { + plugin.webView.loadUrl("javascript:" + js); + } + }); + } +} diff --git a/src/android/java/com/tencent/trtc/CustomVideoView.java b/src/android/java/com/tencent/trtc/CustomVideoView.java new file mode 100644 index 0000000..f9ac607 --- /dev/null +++ b/src/android/java/com/tencent/trtc/CustomVideoView.java @@ -0,0 +1,132 @@ +package com.tencent.trtc; + +import android.content.Context; +import android.util.AttributeSet; +import android.view.LayoutInflater; +import android.view.View; +import android.widget.RelativeLayout; +import android.widget.TextView; + +import com.tencent.rtmp.ui.TXCloudVideoView; +import com.tencent.trtc.event.Events; +import com.tencent.trtc.videocall.UserInfo; + +public class CustomVideoView extends RelativeLayout { + + private Context mContext; + private View mView; + + private TXCloudVideoView videoView; + private TextView titleView; + + private UserInfo userInfo; + + private boolean alwaysHide = false; + private boolean mainView; + + + public CustomVideoView(Context context) { + this(context,null); + } + + public CustomVideoView(Context context, AttributeSet attrs) { + this(context, attrs,0); + } + + public CustomVideoView(Context context, AttributeSet attrs, int defStyleAttr) { + this(context, attrs, defStyleAttr,0); + } + + public CustomVideoView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) { + super(context, attrs, defStyleAttr, defStyleRes); + init(context, attrs); + } + + private void init(Context context, AttributeSet attrs) { + mContext = context; + LayoutInflater inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE); + mView = inflater.inflate(getId("custom_video_view_layout", "layout"), this, true); + videoView = (TXCloudVideoView) findViewById(getId("video", "id")); + titleView = (TextView) findViewById(getId("title","id")); + } + + + private int getId(String idName, String type) { + return getResources().getIdentifier(idName, type, mContext.getPackageName()); + } + + public void changeUser(TRTCCloud mTRTCCloud, UserInfo userInfo) { + if(userInfo == null){ + setVisibility(View.GONE); + }else { + setVisibility(alwaysHide?INVISIBLE:VISIBLE); + } +// if(this.userInfo != null){ +// if(this.userInfo.isLocal()){ +// mTRTCCloud.stopLocalPreview(); +// mTRTCCloud.stopLocalAudio(); +//// }else{ +//// mTRTCCloud.stopRemoteView(this.userInfo.getPersonid()); +// } +// } + if(isChanged(userInfo)){ + this.userInfo = userInfo; + if(this.userInfo != null){ + if(this.userInfo.isLocal()){ + mTRTCCloud.startLocalPreview(userInfo.isFrontCamera(), videoView); + mTRTCCloud.startLocalAudio(TRTCCloudDef.TRTC_AUDIO_QUALITY_SPEECH); + } else { + mTRTCCloud.startRemoteView(this.userInfo.getPersonid(), TRTCCloudDef.TRTC_VIDEO_STREAM_TYPE_SMALL, videoView); + } + if(this.userInfo.getDisplayName() == null || this.userInfo.getDisplayName().length() == 0){ + this.titleView.setText(this.userInfo.getPersonid()); + }else { + this.titleView.setText(this.userInfo.getDisplayName()); + } + this.titleView.setVisibility(VISIBLE); + }else { + this.titleView.setVisibility(INVISIBLE); + } + } + } + + private boolean isChanged(UserInfo userInfo){ + return (this.userInfo == null && userInfo !=null) || + (this.userInfo != null && !this.userInfo.equals(userInfo)); + } + + public boolean isAlwaysHide() { + return alwaysHide; + } + + public CustomVideoView setAlwaysHide(boolean alwaysHide) { + this.alwaysHide = alwaysHide; + return this; + } + + public boolean isMainView() { + return mainView; + } + + public CustomVideoView setMainView(boolean mainView) { + this.mainView = mainView; + if(!mainView){ + Events.addListener("subview.always.hide",(extra)->{ + this.alwaysHide = extra.getBoolean("alwaysHide",false); + this.setVisibility(alwaysHide?INVISIBLE:VISIBLE); + }); + } + Events.addListener("userinfo.update",(extra -> { + if(this.userInfo !=null && this.userInfo.getDisplayName() == null + && this.userInfo.getDisplayName().length() == 0 + && extra.getString("personid").equals(this.userInfo.getPersonid())){ + this.titleView.setText(extra.getString("displayName",this.userInfo.getDisplayName())); + } + })); + return this; + } + + public UserInfo getUserInfo() { + return userInfo; + } +} diff --git a/src/android/java/com/tencent/trtc/MessageObject.java b/src/android/java/com/tencent/trtc/MessageObject.java new file mode 100644 index 0000000..9bcabf9 --- /dev/null +++ b/src/android/java/com/tencent/trtc/MessageObject.java @@ -0,0 +1,10 @@ +package com.tencent.trtc; + +import org.json.JSONObject; + +public interface MessageObject { + + String getEvent(); + + JSONObject toJson(); +} diff --git a/src/android/java/com/example/basic/TRTCBaseActivity.java b/src/android/java/com/tencent/trtc/TRTCBaseActivity.java similarity index 98% rename from src/android/java/com/example/basic/TRTCBaseActivity.java rename to src/android/java/com/tencent/trtc/TRTCBaseActivity.java index 52cb357..4e015b0 100644 --- a/src/android/java/com/example/basic/TRTCBaseActivity.java +++ b/src/android/java/com/tencent/trtc/TRTCBaseActivity.java @@ -1,4 +1,4 @@ -package android.java.com.example.basic; +package com.tencent.trtc; import android.Manifest; import android.content.pm.PackageManager; diff --git a/src/android/java/com/tencent/trtc/TrtcPlugin.java b/src/android/java/com/tencent/trtc/TrtcPlugin.java new file mode 100644 index 0000000..27147ef --- /dev/null +++ b/src/android/java/com/tencent/trtc/TrtcPlugin.java @@ -0,0 +1,86 @@ +package com.tencent.trtc; + +import android.app.Activity; +import android.content.Intent; +import android.util.Log; + +import com.tencent.trtc.CordovaEventKit; +import com.tencent.trtc.event.Events; +import com.tencent.trtc.videocall.VideoCallingEnterActivity; +import com.tencent.trtc.videocall.VideoCallingActivity; + +import org.apache.cordova.CordovaInterface; +import org.apache.cordova.CordovaPlugin; +import org.apache.cordova.CallbackContext; + +import org.apache.cordova.CordovaWebView; +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.lang.reflect.Method; + + +public class TrtcPlugin extends CordovaPlugin { + private static final String TAG = TrtcPlugin.class.getSimpleName(); + + /** + * 加入会议 + * @param args + * @param callbackContext + * @throws JSONException + */ + public void joinChannel(JSONArray args, CallbackContext callbackContext) throws JSONException { + Intent intent = new Intent(this.cordova.getActivity(), VideoCallingActivity.class); + JSONObject arg = args.getJSONObject(0); + String ROOM_ID = arg.getString("ROOM_ID"); + String USER_ID = arg.getString("USER_ID"); + int SDK_APP_ID = arg.getInt("SDK_APP_ID"); + String USER_SIG = arg.getString("USER_SIG"); + intent.putExtra("room_id", ROOM_ID); + intent.putExtra("user_id", USER_ID); + intent.putExtra("sdk_app_id", SDK_APP_ID); + intent.putExtra("user_sig", USER_SIG); + cordova.startActivityForResult(this, intent, 666); + } + + + public void userInfoChange(JSONArray args,CallbackContext callbackContext) throws JSONException{ + JSONObject object = args.getJSONObject(0); + if(object != null){ + Events.fireEvent("userinfo.update",extra -> { + try { + extra.putString("personid",object.getString("personid")); + extra.putString("displayName",object.getString("displayName")); + } catch (JSONException e) { + e.printStackTrace(); + } + }); + } + } + + @Override + public boolean execute(String action, JSONArray args, CallbackContext callbackContext) throws JSONException { + cordova.getThreadPool().execute(new Runnable() { + @Override + public void run() { + try { + Method method = TrtcPlugin.class.getDeclaredMethod(action, JSONArray.class, CallbackContext.class); + method.invoke(TrtcPlugin.this, args, callbackContext); + } catch (Exception e) { + Log.e(TAG, e.toString()); + } + } + }); + return true; + } + + + @Override + public void initialize(CordovaInterface cordova, CordovaWebView webView) { + super.initialize(cordova, webView); + CordovaEventKit.init(this); + } + + +} diff --git a/src/android/java/com/tencent/trtc/debug/GenerateTestUserSig.java b/src/android/java/com/tencent/trtc/debug/GenerateTestUserSig.java deleted file mode 100644 index 0826965..0000000 --- a/src/android/java/com/tencent/trtc/debug/GenerateTestUserSig.java +++ /dev/null @@ -1,280 +0,0 @@ -package com.tencent.trtc.debug; - - -import android.util.Base64; - -import org.json.JSONException; -import org.json.JSONObject; - -import java.io.UnsupportedEncodingException; -import java.nio.charset.Charset; -import java.security.InvalidKeyException; -import java.security.NoSuchAlgorithmException; -import java.util.Arrays; -import java.util.zip.Deflater; - -import javax.crypto.Mac; -import javax.crypto.spec.SecretKeySpec; - -/** - * Module: GenerateTestUserSig - * - * Function: 用于生成测试用的 UserSig,UserSig 是腾讯云为其云服务设计的一种安全保护签名。 - * 其计算方法是对 SDKAppID、UserID 和 EXPIRETIME 进行加密,加密算法为 HMAC-SHA256。 - * - * Attention: 请不要将如下代码发布到您的线上正式版本的 App 中,原因如下: - * - * 本文件中的代码虽然能够正确计算出 UserSig,但仅适合快速调通 SDK 的基本功能,不适合线上产品, - * 这是因为客户端代码中的 SECRETKEY 很容易被反编译逆向破解,尤其是 Web 端的代码被破解的难度几乎为零。 - * 一旦您的密钥泄露,攻击者就可以计算出正确的 UserSig 来盗用您的腾讯云流量。 - * - * 正确的做法是将 UserSig 的计算代码和加密密钥放在您的业务服务器上,然后由 App 按需向您的服务器获取实时算出的 UserSig。 - * 由于破解服务器的成本要高于破解客户端 App,所以服务器计算的方案能够更好地保护您的加密密钥。 - * - * Reference:https://cloud.tencent.com/document/product/647/17275#Server - */ - -/** - * Module: GenerateTestUserSig - * - * Description: generates UserSig for testing. UserSig is a security signature designed by Tencent Cloud for its cloud services. - * It is calculated based on `SDKAppID`, `UserID`, and `EXPIRETIME` using the HMAC-SHA256 encryption algorithm. - * - * Attention: do not use the code below in your commercial app. This is because: - * - * The code may be able to calculate UserSig correctly, but it is only for quick testing of the SDK’s basic features, not for commercial apps. - * `SECRETKEY` in client code can be easily decompiled and reversed, especially on web. - * Once your key is disclosed, attackers will be able to steal your Tencent Cloud traffic. - * - * The correct method is to deploy the `UserSig` calculation code and encryption key on your project server so that your app can request from your server a `UserSig` that is calculated whenever one is needed. - * Given that it is more difficult to hack a server than a client app, server-end calculation can better protect your key. - * - * Reference: https://cloud.tencent.com/document/product/647/17275#Server - */ -public class GenerateTestUserSig { - - /** - * 配置为CDN发布、混流的域名 - * - */ - - /** - * Domain name for CDN publishing and stream mixing - */ - public static final String CDN_DOMAIN_NAME = "testuser"; - - /** - * CDN发布功能 混流bizId - */ - - /** - * `bizId` for CDN publishing and stream mixing - */ - public static final int BIZID = 111; - - /** - * CDN发布功能 混流appId - */ - - /** - * `appId` for CDN publishing and stream mixing - */ - public static final int APPID = 111; - - /** - * 腾讯云 SDKAppId,需要替换为您自己账号下的 SDKAppId。 - * - * 进入腾讯云实时音视频[控制台](https://console.cloud.tencent.com/rav ) 创建应用,即可看到 SDKAppId, - * 它是腾讯云用于区分客户的唯一标识。 - */ - - /** - * Tencent Cloud `SDKAppID`. Set it to the `SDKAppID` of your account. - * - * You can view your `SDKAppID` after creating an application in the [TRTC console](https://console.cloud.tencent.com/rav). - * `SDKAppID` uniquely identifies a Tencent Cloud account. - */ - public static final int SDKAPPID = 1400593044; - - /** - * 签名过期时间,建议不要设置的过短 - *
- * 时间单位:秒 - * 默认时间:7 x 24 x 60 x 60 = 604800 = 7 天 - */ - - /** - * Signature validity period, which should not be set too short - *
- * Unit: second
- * Default value: 604800 (7 days)
- */
- private static final int EXPIRETIME = 604800;
-
-
- /**
- * 计算签名用的加密密钥,获取步骤如下:
- *
- * step1. 进入腾讯云实时音视频[控制台](https://console.cloud.tencent.com/rav ),如果还没有应用就创建一个,
- * step2. 单击应用信息,并进一步找到“快速上手”部分。
- * step3. 点击“复制密钥”按钮,复制密钥,请将其拷贝并复制到如下的变量中
- *
- * 注意:该方案仅适用于调试Demo,正式上线前请将 UserSig 计算代码和密钥迁移到您的后台服务器上,以避免加密密钥泄露导致的流量盗用。
- * 文档:https://cloud.tencent.com/document/product/647/17275#Server
- */
-
- /**
- * Follow the steps below to obtain the key required for UserSig calculation.
- *
- * Step 1. Log in to the [TRTC console](https://console.cloud.tencent.com/rav), and create an application if you don’t have one.
- * Step 2. Find your application, click “Application Info”, and click the “Quick Start” tab.
- * Step 3. Copy and paste the key to the code, as shown below.
- *
- * Note: this method is for testing only. Before commercial launch, please migrate the UserSig calculation code and key to your backend server to prevent key disclosure and traffic stealing.
- * Reference: https://cloud.tencent.com/document/product/647/17275#Server
- */
- public static final String SECRETKEY = "f814a0480cfa95a3ffffd8891844f3cd36888166dbef5347303043744f247c06";
-
- /**
- * 计算 UserSig 签名
- *
- * 函数内部使用 HMAC-SHA256 非对称加密算法,对 SDKAPPID、userId 和 EXPIRETIME 进行加密。
- *
- * @note: 请不要将如下代码发布到您的线上正式版本的 App 中,原因如下:
- *
- * 本文件中的代码虽然能够正确计算出 UserSig,但仅适合快速调通 SDK 的基本功能,不适合线上产品,
- * 这是因为客户端代码中的 SECRETKEY 很容易被反编译逆向破解,尤其是 Web 端的代码被破解的难度几乎为零。
- * 一旦您的密钥泄露,攻击者就可以计算出正确的 UserSig 来盗用您的腾讯云流量。
- *
- * 正确的做法是将 UserSig 的计算代码和加密密钥放在您的业务服务器上,然后由 App 按需向您的服务器获取实时算出的 UserSig。
- * 由于破解服务器的成本要高于破解客户端 App,所以服务器计算的方案能够更好地保护您的加密密钥。
- *
- * 文档:https://cloud.tencent.com/document/product/647/17275#Server
- */
-
- /**
- * Calculating UserSig
- *
- * The asymmetric encryption algorithm HMAC-SHA256 is used in the function to calculate UserSig based on `SDKAppID`, `UserID`, and `EXPIRETIME`.
- *
- * @note: do not use the code below in your commercial app. This is because:
- *
- * The code may be able to calculate UserSig correctly, but it is only for quick testing of the SDK’s basic features, not for commercial apps.
- * `SECRETKEY` in client code can be easily decompiled and reversed, especially on web.
- * Once your key is disclosed, attackers will be able to steal your Tencent Cloud traffic.
- *
- * The correct method is to deploy the `UserSig` calculation code on your project server so that your app can request from your server a `UserSig` that is calculated whenever one is needed.
- * Given that it is more difficult to hack a server than a client app, server-end calculation can better protect your key.
- *
- * Reference: https://cloud.tencent.com/document/product/647/17275#Server
- */
- public static String genTestUserSig(String userId) {
- return GenTLSSignature(SDKAPPID, userId, EXPIRETIME, null, SECRETKEY);
- }
-
- /**
- * 生成 tls 票据
- *
- * @param sdkappid 应用的 appid
- * @param userId 用户 id
- * @param expire 有效期,单位是秒
- * @param userbuf 默认填写null
- * @param priKeyContent 生成 tls 票据使用的私钥内容
- * @return 如果出错,会返回为空,或者有异常打印,成功返回有效的票据
- */
-
- /**
- * Generating a TLS Ticket
- *
- * @param sdkappid `appid` of your application
- * @param userId User ID
- * @param expire Validity period, in seconds
- * @param userbuf `null` by default
- * @param priKeyContent Private key required for generating a TLS ticket
- * @return If an error occurs, an empty string will be returned or exceptions printed. If the operation succeeds, a valid ticket will be returned.
- */
- private static String GenTLSSignature(long sdkappid, String userId, long expire, byte[] userbuf, String priKeyContent) {
- long currTime = System.currentTimeMillis() / 1000;
- JSONObject sigDoc = new JSONObject();
- try {
- sigDoc.put("TLS.ver", "2.0");
- sigDoc.put("TLS.identifier", userId);
- sigDoc.put("TLS.sdkappid", sdkappid);
- sigDoc.put("TLS.expire", expire);
- sigDoc.put("TLS.time", currTime);
- } catch (JSONException e) {
- e.printStackTrace();
- }
-
- String base64UserBuf = null;
- if (null != userbuf) {
- base64UserBuf = Base64.encodeToString(userbuf, Base64.NO_WRAP);
- try {
- sigDoc.put("TLS.userbuf", base64UserBuf);
- } catch (JSONException e) {
- e.printStackTrace();
- }
- }
- String sig = hmacsha256(sdkappid, userId, currTime, expire, priKeyContent, base64UserBuf);
- if (sig.length() == 0) {
- return "";
- }
- try {
- sigDoc.put("TLS.sig", sig);
- } catch (JSONException e) {
- e.printStackTrace();
- }
- Deflater compressor = new Deflater();
- compressor.setInput(sigDoc.toString().getBytes(Charset.forName("UTF-8")));
- compressor.finish();
- byte[] compressedBytes = new byte[2048];
- int compressedBytesLength = compressor.deflate(compressedBytes);
- compressor.end();
- return new String(base64EncodeUrl(Arrays.copyOfRange(compressedBytes, 0, compressedBytesLength)));
- }
-
-
- private static String hmacsha256(long sdkappid, String userId, long currTime, long expire, String priKeyContent, String base64Userbuf) {
- String contentToBeSigned = "TLS.identifier:" + userId + "\n"
- + "TLS.sdkappid:" + sdkappid + "\n"
- + "TLS.time:" + currTime + "\n"
- + "TLS.expire:" + expire + "\n";
- if (null != base64Userbuf) {
- contentToBeSigned += "TLS.userbuf:" + base64Userbuf + "\n";
- }
- try {
- byte[] byteKey = priKeyContent.getBytes("UTF-8");
- Mac hmac = Mac.getInstance("HmacSHA256");
- SecretKeySpec keySpec = new SecretKeySpec(byteKey, "HmacSHA256");
- hmac.init(keySpec);
- byte[] byteSig = hmac.doFinal(contentToBeSigned.getBytes("UTF-8"));
- return new String(Base64.encode(byteSig, Base64.NO_WRAP));
- } catch (UnsupportedEncodingException e) {
- return "";
- } catch (NoSuchAlgorithmException e) {
- return "";
- } catch (InvalidKeyException e) {
- return "";
- }
- }
-
- private static byte[] base64EncodeUrl(byte[] input) {
- byte[] base64 = new String(Base64.encode(input, Base64.NO_WRAP)).getBytes();
- for (int i = 0; i < base64.length; ++i)
- switch (base64[i]) {
- case '+':
- base64[i] = '*';
- break;
- case '/':
- base64[i] = '-';
- break;
- case '=':
- base64[i] = '_';
- break;
- default:
- break;
- }
- return base64;
- }
-
-}
diff --git a/src/android/java/com/tencent/trtc/event/Events.java b/src/android/java/com/tencent/trtc/event/Events.java
new file mode 100644
index 0000000..2a3b51d
--- /dev/null
+++ b/src/android/java/com/tencent/trtc/event/Events.java
@@ -0,0 +1,54 @@
+package com.tencent.trtc.event;
+
+import android.os.Bundle;
+
+import org.json.JSONException;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+public class Events {
+ private static final String PREFIX = "com.tencent.trtc.event";
+ private static Map