基于Rtsp推流,实现局域网无纸化会议直播

注:没有实现音频推送

录屏到直播逻辑过程

  • 推流端:得到录屏权限,开始录屏
  • 推流端:开启线程,解码得到裸流
  • 推流端:通过Rtsp建立服务器推流
  • 播放端:得到rtsp://ip 通过vlc播放

什么是Rtsp?

RTSP协议以客户服务器方式工作,,如:暂停/继续、后退、前进等。它是一个多媒体播放控制协议,用来使用户在播放从因特网下载的实时数据时能够进行控制,
因此 RTSP 又称为“因特网录像机??匦椤薄?a target="_blank" rel="nofollow">RTSP协议详解

如何录屏?

运用Android 5.0开放接口MediaProjection,通过MediaProjectionManage申请录屏权限,用户允许后开始录制屏幕;

步骤:

1.获取MediaProjectionManager实例

mMediaProjectionManager = (MediaProjectionManager) getSystemService(MEDIA_PROJECTION_SERVICE);

2.申请权限

 Intent captureIntent = mMediaProjectionManager.createScreenCaptureIntent();
 startActivityForResult(captureIntent, REQUEST_CODE);

3.在onActivityResult获取授权结果

public void onActivityResult(int requestCode, int resultCode, Intent data) {
        try {
            MediaProjection mediaProjection = mMediaProjectionManager.getMediaProjection(resultCode, data);
            if(mediaProjection == null){
                T.showShort(ServerActivity.this, "程序发生错误:MediaProjection@1");
                RunState.getInstance().setRun(false);
                return;
            }
           
        }
        catch (Exception e){

        }


    }

4.创建录屏线程

public class ScreenRecordThread extends Thread {

    private final static String TAG = "ScreenRecord";

    private Surface mSurface;
    private Context mContext;
    private VirtualDisplay mVirtualDisplay;
    private MediaProjection mMediaProjection;

    private VideoMediaCodec mVideoMediaCodec;

    private WindowManager wm;
    private int windowWidth;
    private int windowHeight;
    private int screenDensity;


    public ScreenRecordThread(Context context, MediaProjection mp, H264DataCollecter mH264Collecter){
        this.mContext = context;
        this.mMediaProjection = mp;
        wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
        windowWidth = DisplayUtils.getDisplayW(context);
        windowHeight = DisplayUtils.getDisplayH(context);
        DisplayMetrics displayMetrics = new DisplayMetrics();
        wm.getDefaultDisplay().getMetrics(displayMetrics);
        screenDensity = displayMetrics.densityDpi;
        mVideoMediaCodec = new VideoMediaCodec(wm, context, mH264Collecter);
    }

    @Override
    public void run() {

        DisplayMetrics displayMetrics = new DisplayMetrics();
        wm.getDefaultDisplay().getMetrics(displayMetrics);
        screenDensity = displayMetrics.densityDpi;
        Log.e("wm-dpi", windowWidth + "-" + windowHeight + "-" + screenDensity);
        mVideoMediaCodec.prepare();
        mSurface =  mVideoMediaCodec.getSurface();
        mVirtualDisplay =mMediaProjection.createVirtualDisplay(TAG + "-display", windowWidth, windowHeight, Constant.VIDEO_DPI,
                DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR,
                mSurface, null, null);
        mVideoMediaCodec.isRun(true);
        mVideoMediaCodec.getBuffer();
    }

    /**
     * 停止
     * **/
    public void release(){
        mVideoMediaCodec.release();
    }




}

5.解码得到裸流

public void getBuffer(){

        try
        {
            while(isRun){
                if(mEncoder == null)
                    break;
                if (startTime == 0) {
                    startTime = mBufferInfo.presentationTimeUs * 1000;
                }

                if (System.currentTimeMillis() - timeStamp >= 1000) {
                    timeStamp = System.currentTimeMillis();
                    Bundle params = new Bundle();
                    params.putInt(MediaCodec.PARAMETER_KEY_REQUEST_SYNC_FRAME, 0);
                    mEncoder.setParameters(params);
                }
                int outputBufferIndex  = mEncoder.dequeueOutputBuffer(mBufferInfo, TIMEOUT_USEC);

                if (outputBufferIndex == MediaCodec.INFO_OUTPUT_FORMAT_CHANGED){
                    MediaFormat outputFormat = mEncoder.getOutputFormat();
                    byte[] AVCDecoderConfigurationRecord = Packager.H264Packager.generateAVCDecoderConfigurationRecord(outputFormat);
                    int packetLen = Packager.FLVPackager.FLV_VIDEO_TAG_LENGTH +
                            AVCDecoderConfigurationRecord.length;
                    byte[] finalBuff = new byte[packetLen];
                    Packager.FLVPackager.fillFlvVideoTag(finalBuff,
                            0,
                            true,
                            true,
                            AVCDecoderConfigurationRecord.length);
                    System.arraycopy(AVCDecoderConfigurationRecord, 0,
                            finalBuff, Packager.FLVPackager.FLV_VIDEO_TAG_LENGTH, AVCDecoderConfigurationRecord.length);

                    H264Data data = new H264Data(finalBuff, 1, 10);
                    if (mH264Collecter != null){
                        mH264Collecter.collect(data);
                    }
                }

                while (outputBufferIndex >= 0){
                    ByteBuffer outputBuffer = mEncoder.getOutputBuffer(outputBufferIndex);

//                    MediaFormat bufferFormat = mEncoder.getOutputFormat(outputBufferIndex);

                    byte[] outData = new byte[mBufferInfo.size];
                    outputBuffer.get(outData);
                    if(mBufferInfo.flags == 2){
                        configbyte = new byte[mBufferInfo.size];
                        configbyte = outData;
                    }else if(mBufferInfo.flags == MediaCodec.BUFFER_FLAG_KEY_FRAME){
                        byte[] keyframe = new byte[mBufferInfo.size + configbyte.length];
                        System.arraycopy(configbyte, 0, keyframe, 0, configbyte.length);
                        System.arraycopy(outData, 0, keyframe, configbyte.length, outData.length);
                        H264Data data = new H264Data(keyframe, 1, mBufferInfo.presentationTimeUs*1000);
                        if (mH264Collecter != null){
                            mH264Collecter.collect(data);
                        }
                    }else{
                        H264Data data = new H264Data(outData, 2, mBufferInfo.presentationTimeUs*1000);
                        if (mH264Collecter != null){
                            mH264Collecter.collect(data);
                        }
                    }
                    mEncoder.releaseOutputBuffer(outputBufferIndex, true);
                    outputBufferIndex = mEncoder.dequeueOutputBuffer(mBufferInfo, TIMEOUT_USEC);
                }
            }
        }
        catch (Exception e){

        }
        try {
            mEncoder.stop();
            mEncoder.release();
            mEncoder = null;
        } catch (Exception e){
            e.printStackTrace();
        }
    }

自此录屏得到裸流过程完成

如何推流?

1.依赖rtsp lib

rtsp开源项目
libstreaming
步骤:

1.bindService And rtsp.start()

bindService(new Intent(this,RtspServer.class), mRtspServiceConnection, Context.BIND_AUTO_CREATE);

private ServiceConnection mRtspServiceConnection = new ServiceConnection() {

        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            mRtspServer = ((RtspServer.LocalBinder)service).getService();
            mRtspServer.addCallbackListener(mRtspCallbackListener);
            mRtspServer.start();
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {}
    };

2.encode后send出去

@Override
    public void collect(H264Data data) {
        DataUtil.getInstance().putData(data);
    }

//RtspServer
 public void run() {
            Log.i(TAG,"RTSP server listening on port "+mServer.getLocalPort());
            while (!Thread.interrupted()) {
                try {
                    new WorkerThread(mServer.accept()).start();
                } catch (SocketException e) {
                    break;
                } catch (IOException e) {
                    Log.e(TAG,e.getMessage());
                    continue;
                }
            }
            Log.i(TAG,"RTSP server stopped !");
        }

//Response
public void send(OutputStream output) throws IOException {
            int seqid = -1;

            try {
                if (mRequest != null && mRequest.headers != null){
                    if (!TextUtils.isEmpty(mRequest.headers.get("cseq"))){
                        seqid = Integer.parseInt(mRequest.headers.get("cseq").replace(" ",""));
                    }
                }
            } catch (Exception e) {
                Log.e(TAG,"Error parsing CSeq: "+(e.getMessage()!=null?e.getMessage():""));
            }

            String response =   "RTSP/1.0 "+status+"\r\n" +
                    "Server: "+SERVER_NAME+"\r\n" +
                    (seqid>=0?("Cseq: " + seqid + "\r\n"):"") +
                    "Content-Length: " + content.length() + "\r\n" +
                    attributes +
                    "\r\n" +
                    content;


            Log.d(TAG,response.replace("\r", ""));

            output.write(response.getBytes());
        }
    }

具体过程请看libstreaming,自此解码推流过程结束

如何播放?

简单,vlc开源项目,有人已经帮你编译好了直接拿来用
步骤:

1.布局

 <SurfaceView
        android:id="@+id/player_surface"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

2.实例化和配置

    private LibVLC libVLC = null;
    private MediaPlayer mediaPlayer;
    private SurfaceHolder mSurfaceHolder;
    SurfaceView mPlayView;
//播放配置
        ArrayList<String> options = new ArrayList<>();
        libVLC = new LibVLC(getApplication(), options);
        mediaPlayer = new MediaPlayer(libVLC);
        mSurfaceHolder = mPlayView.getHolder();
        mSurfaceHolder.setFixedSize(DisplayUtil.getDisplayW(this), DisplayUtil.getDisplayH(this));
        mSurfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
        mediaPlayer.getVLCVout().setVideoSurface(mPlayView.getHolder().getSurface(), mSurfaceHolder);
        mediaPlayer.getVLCVout().attachViews();
        Media media = new Media(libVLC, Uri.parse(playUrl));
        mediaPlayer.setMedia(media);
       
 @Override
    protected void onPause() {
        super.onPause();
        if (mediaPlayer != null) {
            mediaPlayer.pause();
        }
    }

    @Override
    protected void onResume() {
        super.onResume();
        if (mediaPlayer != null) {
            mediaPlayer.play();
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        if (mediaPlayer != null) {
            mediaPlayer.release();
            mediaPlayer = null;
        }
    }

3.播放

mediaPlayer.play();

体验Apk

推流和播放端一体化 52.37 MB,因为vlc库太大的原因

apk下载地址

Thanks

下一篇文章:
三步实现无纸化会议封装

最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容

  • 1、通过CocoaPods安装项目名称项目信息 AFNetworking网络请求组件 FMDB本地数据库组件 SD...
    阳明先生_X自主阅读 15,975评论 3 119
  • 1)翻译下面的句子: 政府逐渐加强了对网约车的管制和处罚。 The government has graduall...
    Maglight阅读 216评论 0 0
  • 新学期,某位小朋友大有长进呵! 首先是英语听写,可以跟上老师的节奏,能对大部分了,这是一大长进。上个学期,单词默写...
    凤姐姐啊阅读 435评论 0 0
  • 我对别人诚实吗?对我自己呢?我在评估自己的技能和才能方面有多客观?我有多聪明?我在学校有多勤奋?我和其他人扮演多少...
    柳涛虹阅读 238评论 0 0
  • 四年前,我因为一段感情的失意,也源于自己的矫情,从太原这座充满回忆的城市出走到一座举目无亲的城市——南京。 是的,...
    聿靈隨筆阅读 129评论 0 0