2017-2-8

基本架构

1、使用SurfaceView在UI层面展示视频
2、通过JNI调用C代码控制视频的播放、停止

基本功能

1、从服务器获取正在直播的主播的列表信息

2、进入直播界面

3、可获取的主播信息


Activity_主列表界面

public class MainActivity extends Activity {
    public static final String SERVER_URL = "api.95xiu.com";//"tapi.95xiu.com"
    public static final String RQ_LIVE = "/show/anchor_list_v3.php";
    private List<LiveBean> mList;
    private GridView gridView;
    private MyBaseAdapter adapter;
    private DisplayImageOptions options;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);//取消标题栏
        ImageLoader.getInstance().init(new ImageLoaderConfiguration.Builder(this).build());
        options = new DisplayImageOptions.Builder().cacheInMemory(true).cacheOnDisk(true)//将图片缓存到内存和硬盘中
                .showImageOnLoading(R.drawable.live_icon_default).showImageOnFail(R.drawable.live_icon_default)//加载中、加载错误时显示的图片
                .bitmapConfig(Bitmap.Config.RGB_565).build();
        setContentView(R.layout.layout_main);
        mList = new ArrayList<LiveBean>();
        loadingData();
        gridView = (GridView) findViewById(R.id.gv);
        adapter = new MyBaseAdapter();
        gridView.setAdapter(adapter);
    }

    private void loadingData() {
        RequestParams requestParams = new RequestParams();
        requestParams.put("page_index", "0");//非必须
        requestParams.put("version", "1");//非必须
        AsyncHttpHelper.get(SERVER_URL,RQ_LIVE, requestParams, new OnHttpListener<JSONObject>() {//http://api.95xiu.com/show/anchor_list_v3.php
            @Override
            public void onHttpListener(boolean httpSuccessed, JSONObject obj) {
                if (httpSuccessed) {
                    Log.i("bqt", obj.toString());
                    if (obj.optInt("result") == 1) {
                        JSONArray arr = obj.optJSONArray("user_info");
                        if (arr != null && arr.length() > 0) {
                            for (int i = 0; i < arr.length(); i++) {
                                JSONObject jObject = arr.optJSONObject(i);
                                mList.add(new LiveBean(jObject.optString("anchor_image"), jObject.optString("nickname"), jObject.optString("live_num")//
                                        , jObject.optString("uid")));
                            }
                            Log.i("bqt", "数据个数:" + mList.size());
                        }
                        adapter.notifyDataSetChanged();
                    }
                }
            }
        });
    }

    public class MyBaseAdapter extends BaseAdapter {
        private ViewHolder mViewHolder;
        @Override
        public int getCount() {
            return mList == null ? 0 : mList.size();
        }
        @Override
        public Object getItem(int position) {
            return mList == null ? null : mList.get(position);
        }
        @Override
        public long getItemId(int position) {
            return position;
        }
        @Override
        public View getView(final int position, View convertView, ViewGroup parent) {
            if (convertView != null) mViewHolder = (ViewHolder) convertView.getTag();
            else {
                convertView = LayoutInflater.from(MainActivity.this).inflate(R.layout.item, null);
                mViewHolder = new ViewHolder();
                mViewHolder.iv_head = (ImageView) convertView.findViewById(R.id.iv_anchor_img);
                mViewHolder.tv_name = (TextView) convertView.findViewById(R.id.tv_anchor_name);
                mViewHolder.tv_audience_num = (TextView) convertView.findViewById(R.id.tv_audience_num);
                mViewHolder.iv_rankingLev = (ImageView) convertView.findViewById(R.id.iv_rankingLev);
                mViewHolder.iv_moods = (ImageView) convertView.findViewById(R.id.iv_moods);
                mViewHolder.iv_coverage = (ImageView) convertView.findViewById(R.id.iv_coverage);
                convertView.setTag(mViewHolder);
            }
            if (mList != null) {
                ImageLoader.getInstance().displayImage(mList.get(position).anchor_image, mViewHolder.iv_head, options);
                mViewHolder.tv_name.setText(mList.get(position).nickname);
                mViewHolder.tv_audience_num.setText(mList.get(position).live_num);
                if (position == 0) mViewHolder.iv_rankingLev.setVisibility(View.VISIBLE);
                else mViewHolder.iv_rankingLev.setVisibility(View.INVISIBLE);
                if (position == 0 || position == 1 || position == 3 || position == 6) mViewHolder.iv_moods.setVisibility(View.VISIBLE);
                else mViewHolder.iv_moods.setVisibility(View.INVISIBLE);
                mViewHolder.iv_coverage.setOnClickListener(new OnClickListener() {
                    @Override
                    public void onClick(View v) {
                        Intent intent = new Intent(MainActivity.this, LiveActivity.class);
                        intent.putExtra("anchorId", mList.get(position).anchorId);
                        Log.i("bqt", "主播ID:" + mList.get(position).anchorId);
                        startActivity(intent);
                    }
                });
            }
            return convertView;
        }
    }

    public static class ViewHolder {
        public ImageView iv_head;//主播头像
        public TextView tv_name;//主播名字
        public TextView tv_audience_num;//直播间人数
        public ImageView iv_rankingLev;//排名
        public ImageView iv_moods;//人气等标签
        public ImageView iv_coverage;//背景框
    }
}

class LiveBean {
    public String anchor_image;
    public String nickname;
    public String live_num;
    public String anchorId;
    public LiveBean(String anchor_image, String nickname, String live_num, String anchorId) {
        super();
        this.anchor_image = anchor_image;
        this.nickname = nickname;
        this.live_num = live_num;
        this.anchorId = anchorId;
    }
}

两个布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:baselineAligned="false"
    android:background="#fff"
    android:orientation="horizontal" >
    <GridView
        android:id="@+id/gv"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:cacheColorHint="#00000000"
        android:listSelector="#00000000"
        android:verticalSpacing="5dp"
        android:horizontalSpacing="5dp"
        android:descendantFocusability ="beforeDescendants"
        android:numColumns="2" />
</LinearLayout>

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/rl_item"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="#fbf7ed" >
    <ImageView
        android:id="@+id/iv_anchor_img"
        android:layout_width="match_parent"
        android:layout_height="130dp"
        android:layout_alignParentTop="true"
        android:scaleType="fitXY"
        android:src="@drawable/live_icon_default" />
    <ImageView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignBottom="@id/iv_anchor_img"
        android:background="@drawable/main_live_head_bg" />
    <TextView
        android:id="@+id/tv_audience_num"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignBottom="@id/iv_anchor_img"
        android:layout_alignParentLeft="true"
        android:layout_marginBottom="4dp"
        android:drawableLeft="@drawable/living_true"
        android:drawablePadding="4dp"
        android:ellipsize="end"
        android:gravity="center"
        android:paddingLeft="8dp"
        android:singleLine="true"
        android:text="111"
        android:textColor="#00f"
        android:textSize="12sp" />
    <TextView
        android:id="@+id/tv_anchor_name"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignBottom="@id/tv_audience_num"
        android:layout_alignParentRight="true"
        android:layout_marginLeft="5dp"
        android:layout_toRightOf="@id/tv_audience_num"
        android:ellipsize="end"
        android:gravity="right"
        android:paddingRight="8dp"
        android:singleLine="true"
        android:text="这是主播主播主播"
        android:textColor="#fff"
        android:textSize="12sp" />
    <ImageView
        android:id="@+id/iv_rankingLev"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_alignParentTop="true"
        android:layout_marginLeft="4dp"
        android:layout_marginTop="8dp"
        android:src="@drawable/live_ranking1" />
    <ImageView
        android:id="@+id/iv_moods"
        android:layout_width="40dp"
        android:layout_height="17dp"
        android:layout_alignParentRight="true"
        android:layout_alignParentTop="true"
        android:layout_marginTop="8dp"
        android:src="@drawable/live_moods" />
    <ImageView
        android:id="@+id/iv_coverage"
        android:layout_width="match_parent"
        android:layout_height="130dp"
        android:background="@drawable/live_main_item_bg"
        android:clickable="true"
        android:focusable="true" />
</RelativeLayout>

Activity_直播界面

public class LiveActivity extends Activity {
    private ViEAndroidGLES20 mSurfaceView;
    private LinearLayout root;
    private String mVideoUrl;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);//取消标题栏
        initViews();
        setContentView(root);
    }
    private void initViews() {
        // 视频比率 4:3 
        DisplayMetrics displayMetrics = new DisplayMetrics();
        getWindowManager().getDefaultDisplay().getMetrics(displayMetrics);
        int width = displayMetrics.widthPixels;
        int h = width * 3 / 4;
        if (h % 2 != 0) h += 1;
        root = new LinearLayout(this);
        root.setOrientation(LinearLayout.VERTICAL);
        mSurfaceView = ViERenderer.CreateRenderer(this, true, null);
        mSurfaceView.setLayoutParams(new FrameLayout.LayoutParams(width, h));//具有父控件的View设置setLayoutParams才有效!
        root.addView(mSurfaceView);
        TextView textView = new TextView(this);
        root.addView(textView);
        Intent intent = getIntent();
        if (intent != null && intent.getStringExtra("anchorId") != null) {
            mVideoUrl = String.format("app/%s?k=1092dc67c9402014144fc19181974172&t=540fb568", intent.getStringExtra("anchorId"));
            textView.setText("视频地址:" + mVideoUrl);
            //livestream.setPlayerStateHandler(mHandler);
            livestream.init(mVideoUrl, mSurfaceView);
            startVideo();
        }
    }

    public void startVideo() {
        try {
            Log.v("vvvv", "startPlay");
            livestream.setVideoPath(mVideoUrl);
            livestream.setSurfaceView(mSurfaceView);
            livestream.playVideo();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    public void stopVideo(boolean isFreeHandler) {
        try {
            Log.v("vvvv", "stopPlay");
            livestream.stopVideo();
            if (isFreeHandler) {
                livestream.setPlayerStateHandler(null);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    public static void freeVideo() {
        try {
            livestream.setPlayerStateHandler(null);
            livestream.free();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Override
    public void onPause() {
        super.onPause();
    }
    @Override
    public void onDestroy() {
        super.onDestroy();
        if (mSurfaceView != null) mSurfaceView.onPause();
        stopVideo(true);
    }
}

清单文件

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.bqt.test"
    android:versionCode="1"
    android:versionName="1.0" >
    <uses-sdk
        android:minSdkVersion="16"
        android:targetSdkVersion="21" />
    <uses-permission android:name="android.permission.WAKE_LOCK" />
    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name" >
        <activity android:name=".MainActivity" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity android:name=".LiveActivity" >
        </activity>
    </application>
</manifest>
 

附件列表

直播【95秀】JNI 基本实现 简洁的更多相关文章

  1. 95秀-自定义对话框 dialog 合集

    普通的确认对话框 NormalDialog.java import android.app.Dialog; import android.content.Context; import android ...

  2. 95秀-ViewPager 使用实例

    Activity的样式     <style name="under_live_indicator" parent="android:Theme.NoTitleBa ...

  3. 95秀-异步http请求完整过程

    最终调用时的代码     private void ansyClearApplyInfor() {         RequestParams params = new RequestParams() ...

  4. 95秀-dialog 进度对话框 实用工具

    工具Util public class DialogUtil {     public static ProgressDialogView progressDialog;     /**      * ...

  5. 95秀-PullToRefreshListView 示例

        正在加载.暂无数据页面 public class RefreshGuideTool {     private RelativeLayout rl_loading_guide;//整个View ...

  6. 95秀-弹窗+listview+动画 示例

    Dialog布局 dialog.xml <?xml version="1.0" encoding="utf-8"?> <RelativeLay ...

  7. 做一款仿映客的直播App

    投稿文章,作者:JIAAIR(GitHub) 一.直播现状简介 1.技术实现层面 技术相对都比较成熟,设备也都支持硬编码.iOS还提供现成的Video ToolBox框架,可以对摄像头和流媒体数据结构 ...

  8. 做一款仿映客的直播App?看我就够了

    来源:JIAAIR 链接:http://www.jianshu.com/p/5b1341e97757 一.直播现状简介   1.技术实现层面: 技术相对都比较成熟,设备也都支持硬编码.IOS还提供现成 ...

  9. 开发一款直播APP系统软件应该有哪些功能,如何开发?

    1.技术实现层面: 技术相对都比较成熟,设备也都支持硬编码.IOS还提供现成的 Video ToolBox框架,可以对摄像头和流媒体数据结构进行处理,但Video ToolBox框架只兼容8.0以上版 ...

随机推荐

  1. Spring技术内幕:设计理念和整体架构概述(转)

    程序员都很崇拜技术大神,很大一部分是因为他们发现和解决问题的能力,特别是线上出现紧急问题时,总是能够快速定位和解决. 一方面,他们有深厚的技术基础,对应用的技术知其所以然,另一方面,在采坑的过程中不断 ...

  2. OptParse选项工具模块

    OptParse是一个从Python2.3版本起引入的一个编写命令行工具模块,示例如下 ######example.py###### import optparse if __name__ == &q ...

  3. CSS3组件化之圆波扩散

    本篇文章主要介绍用CSS3实现的水波扩散涟漪,圆波扩散,光圈扩散,雷达波向外散发动画. 预期效果应该是这样:,其实应该比这个更优美,因为设计师提供的gif出现透明度丢失问题,所以建议用css3实现. ...

  4. C# NPOCO 轻量级ORM框架(入门)

    目前公司使用这个框架,搜不到很详细的中文资料. 只有英文wiki,所以翻译学习一下. 因为博主也是低水平的,可能会有一些理解不到位的地方. 可能会有错误的地方,如果有园友发现可以指出. wiki地址: ...

  5. 【BZOJ 3640】JC的小苹果 (高斯消元,概率DP)

    JC的小苹果 Submit: 432  Solved: 159 Description 让我们继续JC和DZY的故事. “你是我的小丫小苹果,怎么爱你都不嫌多!” “点亮我生命的火,火火火火火!” 话 ...

  6. luoguP4208 [JSOI2008]最小生成树计数 矩阵树定理

    题目大意: 求最小生成树的数量 曾今的我感觉这题十分的不可做 然而今天看了看,好像是个类模板的题.... 我们十分容易知道,记能出现在最小生成树中的边的集合为\(S\) 那么,只要是\(S\)中的边构 ...

  7. CF696B Puzzles 期望

    显然可以树形$dp$ 令$f[i]$表示$i$号节点的期望时间戳 不妨设$fa$有$k$个子节点,对于$i$的子节点$u$,它是第$j(1 \leqslant j \leqslant k)$个被访问的 ...

  8. UOJ.87.mx的仙人掌(圆方树 虚树)(未AC)

    题目链接 本代码10分(感觉速度还行..). 建圆方树,预处理一些东西.对询问建虚树. 对于虚树上的圆点直接做:对于方点特判,枚举其所有儿子,如果子节点不在该方点代表的环中,跳到那个点并更新其val, ...

  9. 一个java高级工程师的进阶之路【转】

    宏观方面 一. JAVA.要想成为JAVA(高级)工程师肯定要学习JAVA.一般的程序员或许只需知道一些JAVA的语法结构就可以应付了.但要成为JAVA(高级) 工程师,您要对JAVA做比较深入的研究 ...

  10. 如何在ubuntu安装phpstorm

    第一步:使用组合键ctrl+alt+t 打开Terminal,cd /home/xxx(当前登录用户名)/downloads(下载目录) 第二步:下载 phpstorm 附截止发文最新版本链接:htt ...