Android WebRTC完整入門教程01: 使用相機

WebRTC安卓端沒有官方教程从藤,甚至連API文檔都沒有催跪。這是一件奇怪的事,畢竟WebRTC是Google開發(fā)的夷野。目前官方文檔和Demo都只有web端的懊蒸,雖然寫得簡單易懂,整體用法也和安卓端相同悯搔,但是具體細節(jié)還是有巨大的差異骑丸。

當然,仔細找Google和Github上還是能找到一些不錯的教程妒貌,我這里將它們結合一下通危,組成一個完整的入門教程,并帶有可運行Demo.

首先介紹一些基本概念
RTC(Real Time Communication): 實時通信
WebRTC: 基于web的實時通信
Signaling: 信令, 一些描述媒體或網(wǎng)絡的字符串
SDP(Session Description Protocol): 會話描述協(xié)議, 主要描述媒體信息
ICE(Interactive Connectivity Establishment): 交互式連接建立
STUN(Session Traversal Utilities for NAT): NAT會話穿透工具
TURN(Traversal Using Relays around NAT): 中繼穿透NAT

接下來是使用方法

添加WebRTC庫

在module的build.gradle中添加依賴灌曙,這個是官方打包的最新版本(201901)菊碟。當然你也可以 自己構建.

dependencies {
    ...
    implementation 'org.webrtc:google-webrtc:1.0.26131'
}

添加權限

    <uses-permission android:name="android.permission.CAMERA" />
    <uses-permission android:name="android.permission.RECORD_AUDIO" />

注意Android6.0以上需要到設置里開啟相機和麥克風權限。因為跟主題無關, 所以這里沒加申請權限代碼.

添加SurfaceViewRenderer

它是SurfaceView的子類

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="cc.rome753.wat.MainActivity">

    <org.webrtc.SurfaceViewRenderer
        android:id="@+id/localView"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

</android.support.constraint.ConstraintLayout>

使用相機

主要步驟如下

  1. 創(chuàng)建PeerConnectionFactory
  2. 創(chuàng)建并啟動VideoCapturer
  3. 用PeerConnectionFactory創(chuàng)建VideoSource
  4. 用PeerConnectionFactory和VideoSource創(chuàng)建VideoTrack
  5. 初始化視頻控件SurfaceViewRenderer
  6. 將VideoTrack展示到SurfaceViewRenderer中
public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        // create PeerConnectionFactory
        PeerConnectionFactory.InitializationOptions initializationOptions =
                PeerConnectionFactory.InitializationOptions.builder(this).createInitializationOptions();
        PeerConnectionFactory.initialize(initializationOptions);
        PeerConnectionFactory peerConnectionFactory = PeerConnectionFactory.builder().createPeerConnectionFactory();

        // create AudioSource
        AudioSource audioSource = peerConnectionFactory.createAudioSource(new MediaConstraints());
        AudioTrack audioTrack = peerConnectionFactory.createAudioTrack("101", audioSource);

        EglBase.Context eglBaseContext = EglBase.create().getEglBaseContext();

        SurfaceTextureHelper surfaceTextureHelper = SurfaceTextureHelper.create("CaptureThread", eglBaseContext);
        // create VideoCapturer
        VideoCapturer videoCapturer = createCameraCapturer();
        VideoSource videoSource = peerConnectionFactory.createVideoSource(videoCapturer.isScreencast());
        videoCapturer.initialize(surfaceTextureHelper, getApplicationContext(), videoSource.getCapturerObserver());
        videoCapturer.startCapture(480, 640, 30);

        SurfaceViewRenderer localView = findViewById(R.id.localView);
        localView.setMirror(true);
        localView.init(eglBaseContext, null);

        // create VideoTrack
        VideoTrack videoTrack = peerConnectionFactory.createVideoTrack("101", videoSource);
        // display in localView
        videoTrack.addSink(localView);
    }

    private VideoCapturer createCameraCapturer() {
        Camera1Enumerator enumerator = new Camera1Enumerator(false);
        final String[] deviceNames = enumerator.getDeviceNames();

        // First, try to find front facing camera
        for (String deviceName : deviceNames) {
            if (enumerator.isFrontFacing(deviceName)) {
                VideoCapturer videoCapturer = enumerator.createCapturer(deviceName, null);

                if (videoCapturer != null) {
                    return videoCapturer;
                }
            }
        }

        // Front facing camera not found, try something else
        for (String deviceName : deviceNames) {
            if (!enumerator.isFrontFacing(deviceName)) {
                VideoCapturer videoCapturer = enumerator.createCapturer(deviceName, null);

                if (videoCapturer != null) {
                    return videoCapturer;
                }
            }
        }

        return null;
    }

}

附錄

https://webrtc.org/start/
https://codelabs.developers.google.com/codelabs/webrtc-web/#0
https://developer.mozilla.org/en-US/docs/Web/API/MediaStream
https://vivekc.xyz/getting-started-with-webrtc-for-android-daab1e268ff4
https://www.html5rocks.com/en/tutorials/webrtc/basics/

本項目GitHub地址/step1camera

下一篇: Android WebRTC完整入門教程02: 本地回環(huán)

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末在刺,一起剝皮案震驚了整個濱河市逆害,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌蚣驼,老刑警劉巖魄幕,帶你破解...
    沈念sama閱讀 206,968評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異颖杏,居然都是意外死亡纯陨,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評論 2 382
  • 文/潘曉璐 我一進店門输玷,熙熙樓的掌柜王于貴愁眉苦臉地迎上來队丝,“玉大人,你說我怎么就攤上這事欲鹏』茫” “怎么了?”我有些...
    開封第一講書人閱讀 153,220評論 0 344
  • 文/不壞的土叔 我叫張陵赔嚎,是天一觀的道長颈嚼。 經(jīng)常有香客問我嘴拢,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,416評論 1 279
  • 正文 為了忘掉前任奉件,我火速辦了婚禮,結果婚禮上褪测,老公的妹妹穿的比我還像新娘砌溺。我一直安慰自己,他們只是感情好尤勋,可當我...
    茶點故事閱讀 64,425評論 5 374
  • 文/花漫 我一把揭開白布喘落。 她就那樣靜靜地躺著茵宪,像睡著了一般。 火紅的嫁衣襯著肌膚如雪瘦棋。 梳的紋絲不亂的頭發(fā)上稀火,一...
    開封第一講書人閱讀 49,144評論 1 285
  • 那天,我揣著相機與錄音赌朋,去河邊找鬼凰狞。 笑死,一個胖子當著我的面吹牛沛慢,可吹牛的內容都是我干的赡若。 我是一名探鬼主播,決...
    沈念sama閱讀 38,432評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼颠焦,長吁一口氣:“原來是場噩夢啊……” “哼斩熊!你這毒婦竟也來了?” 一聲冷哼從身側響起伐庭,我...
    開封第一講書人閱讀 37,088評論 0 261
  • 序言:老撾萬榮一對情侶失蹤粉渠,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后圾另,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體霸株,經(jīng)...
    沈念sama閱讀 43,586評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,028評論 2 325
  • 正文 我和宋清朗相戀三年集乔,在試婚紗的時候發(fā)現(xiàn)自己被綠了去件。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,137評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡扰路,死狀恐怖尤溜,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情汗唱,我是刑警寧澤宫莱,帶...
    沈念sama閱讀 33,783評論 4 324
  • 正文 年R本政府宣布,位于F島的核電站哩罪,受9級特大地震影響授霸,放射性物質發(fā)生泄漏。R本人自食惡果不足惜际插,卻給世界環(huán)境...
    茶點故事閱讀 39,343評論 3 307
  • 文/蒙蒙 一碘耳、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧框弛,春花似錦辛辨、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽绞蹦。三九已至力奋,卻和暖如春榜旦,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背景殷。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評論 1 262
  • 我被黑心中介騙來泰國打工溅呢, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人猿挚。 一個月前我還...
    沈念sama閱讀 45,595評論 2 355
  • 正文 我出身青樓咐旧,卻偏偏與公主長得像,于是被迫代替她去往敵國和親绩蜻。 傳聞我的和親對象是個殘疾皇子铣墨,可洞房花燭夜當晚...
    茶點故事閱讀 42,901評論 2 345

推薦閱讀更多精彩內容

  • afinalAfinal是一個android的ioc,orm框架 https://github.com/yangf...
    passiontim閱讀 15,401評論 2 45
  • 發(fā)現(xiàn) 關注 消息 iOS 第三方庫办绝、插件伊约、知名博客總結 作者大灰狼的小綿羊哥哥關注 2017.06.26 09:4...
    肇東周閱讀 12,033評論 4 62
  • 1.集群的目的實現(xiàn)高可用性。那么集群是如何實現(xiàn)高可用性的呢主要原因有兩個1)當主節(jié)點掛掉的時候孕蝉,如果此時任何一個s...
    何甜甜在嗎閱讀 804評論 0 1
  • 一 14年那會兒屡律,我還沒認識麗姐。 我先認識的降淮,是麗姐的爸爸超埋,親爸的那種。 麗姐的親爸呢佳鳖,在麗姐很小的時候霍殴,就和她...
    黑色深呼吸閱讀 514評論 0 0