1. VR开发概述
时下关于“谷歌、Android与VR”的各种言论纷飞。VR群里有人在争论Android VR是不是一体机,是不是类似Android Wear、为VR打造的全新平台,是不是改良后的Android N。
随着vr设备的流行开来,各大招聘平台上也发布了不少关于andorid vr开发相关的Android工程师岗位,从这点来说掌握vr在项目中的实际应用要点,有助于大家为自己的开发经验上增加前沿技术的积累。
经过研发市面上的主流vr app 的功能,抽取并整合项目中的vr开发知识点,希望大家掌握后,在企业相关vr app游刃有余。
1.1 下载google vr sdk 并搭建开发环境
带大家去github上搜索并下载google vr sdk介绍sdk的组成部分与应用范围搭建一个基本android vr app的开发环境
1.2 CardBorad应用核心功能
带大家查找本地vr 全景图片资源介绍vr全景图与普通图片的不同点使用rv列表进行展示使用VrPanoramaView控件进行本地全景图片的展示
1.3 UtoVR应用核心功能
带大家通过网络请求获取vr视频的json数据使用 Gson解析得到javaBean数据使用VrVideoView控件进行网络全景视图的展示
以上这些功能是现流行的在线vr视频,vr图片相关app的核心功能。例如.vr管家应用,3d播播,discovery VR ,看房 vr等等热门应用。
1.4 知识点
http网络请求技术Gson解析技术与gsonformat插件RecyclerView与cardViewGlider流行图片加载框架VrPanoramaViewVrVideoView
1.5 好玩好用的VR
成本其实很便宜!教你用手机体验VR魅力
17块钱!把手机改造为VR眼镜好玩好用的VR APP推荐
2. 全景图片显示
2.1 搭建vr全景图片的开发环境
VR开发需要gvr-android-sdk,GitHub下载地址
VR开发Google官方技术文档
2.1.1 导入全景图相关的三个开发库
common,commonwidget,panowidget PS:最新的SDK已经没有这三个文件夹了,使用下一步的依赖库即可
2.1.2 依赖该库
compile
'com.google.vr:sdk-panowidget:1.30.0'
2.1.3 准备全景图片测试资源
放在assets目录 例:assets/a.jpg(全景图与普通图片的不同 大,立体)
2.1.4 功能清单配置
android:largeHeap=”true” 全景图片比较耗资源
<application
android:largeHeap="true">
</application>
2.2 布局全景控件显示加载后的全景图片
<com.google.vr.sdk.widgets.pano.VrPanoramaView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/vr_pv" />
public class MainActivity extends AppCompatActivity {
private VrPanoramaView vrPanoramaView;
private ImageTask imageTask;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
2.2.1 布局查找控件
vrPanoramaView = (VrPanoramaView) findViewById(R.id.vr_pv);
2.2.2 设置初始化参数
vrPanoramaView.setDisplayMode(VrWidgetView.DisplayMode.FULLSCREEN_STEREO);
vrPanoramaView.setInfoButtonEnabled(
false);
vrPanoramaView.setFullscreenButtonEnabled(
false);
2.2.3 创建异步任务加载图片
Bitmap是图片在内存中的表示对象,全景图也可加载成bitmap
imageTask =
new ImageTask();
imageTask.execute();
}
private class ImageTask extends AsyncTask<Void, Void, Bitmap> {
@Override
protected Bitmap
doInBackground(Void... params) {
try {
InputStream inputStream = getAssets().open(
"a.jpg");
Bitmap bitmap = BitmapFactory.decodeStream(inputStream);
return bitmap;
}
catch (IOException e) {
e.printStackTrace();
}
return null;
}
@Override
protected void onPostExecute(Bitmap bitmap) {
super.onPostExecute(bitmap);
if (bitmap !=
null) {
VrPanoramaView.Options options =
new VrPanoramaView.Options();
options.inputType = VrPanoramaView.Options.TYPE_STEREO_OVER_UNDER;
if (listener ==
null) {
listener =
new VrPanoramaEventListener() {
@Override
public void onLoadError(String errorMessage) {
super.onLoadError(errorMessage);
Toast.makeText(MainActivity.
this,
"错误消息:" + errorMessage, Toast.LENGTH_SHORT).show();
}
@Override
public void onLoadSuccess() {
super.onLoadSuccess();
Toast.makeText(MainActivity.
this,
"进入vr:", Toast.LENGTH_SHORT).show();
}
};
vrPanoramaView.setEventListener(listener);
}
vrPanoramaView.loadImageFromBitmap(bitmap, options);
}
}
}
private VrPanoramaEventListener listener;
2.3 VrPanoramaView控件退到后台,回到屏幕,销毁处理细节
@Override
protected void onPause() {
super.onPause();
if (vrPanoramaView !=
null) {
vrPanoramaView.pauseRendering();
}
}
@Override
protected void onResume() {
super.onResume();
if (vrPanoramaView !=
null) {
vrPanoramaView.resumeRendering();
}
}
@Override
protected void onDestroy() {
if (vrPanoramaView !=
null) {
vrPanoramaView.shutdown();
}
if (imageTask !=
null && !imageTask.isCancelled()) {
imageTask.cancel(
true);
imageTask =
null;
}
super.onDestroy();
}
}
package com.itheima.demovrimagevideo2;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.AsyncTask;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.Toast;
import com.google.vr.sdk.widgets.common.VrWidgetView;
import com.google.vr.sdk.widgets.pano.VrPanoramaEventListener;
import com.google.vr.sdk.widgets.pano.VrPanoramaView;
import java.io.IOException;
import java.io.InputStream;
public class MainActivity extends AppCompatActivity {
private VrPanoramaView vrPanoramaView;
private ImageTask imageTask;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
vrPanoramaView = (VrPanoramaView) findViewById(R.id.vr_pano);
vrPanoramaView.setInfoButtonEnabled(
false);
vrPanoramaView.setFullscreenButtonEnabled(
false);
imageTask =
new ImageTask();
imageTask.execute();
}
private class ImageTask extends AsyncTask<Void, Void, Bitmap> {
@Override
protected Bitmap
doInBackground(Void... params) {
try {
InputStream inputStream = getAssets().open(
"a.jpg");
Bitmap bitmap = BitmapFactory.decodeStream(inputStream);
return bitmap;
}
catch (IOException e) {
e.printStackTrace();
}
return null;
}
@Override
protected void onPostExecute(Bitmap bitmap) {
super.onPostExecute(bitmap);
if (bitmap !=
null) {
VrPanoramaView.Options option =
new VrPanoramaView.Options();
option.inputType = VrPanoramaView.Options.TYPE_STEREO_OVER_UNDER;
VrPanoramaEventListener listener=
new VrPanoramaEventListener(){
@Override
public void onLoadSuccess() {
super.onLoadSuccess();
Toast.makeText(MainActivity.
this,
"进入vr图片", Toast.LENGTH_SHORT).show();
}
@Override
public void onLoadError(String errorMessage) {
super.onLoadError(errorMessage);
Toast.makeText(MainActivity.
this,
"E:"+errorMessage, Toast.LENGTH_SHORT).show();
}
};
vrPanoramaView.setEventListener(listener);
vrPanoramaView.setDisplayMode(VrWidgetView.DisplayMode.FULLSCREEN_MONO);
vrPanoramaView.loadImageFromBitmap(bitmap, option);
}
}
}
@Override
protected void onPause() {
super.onPause();
if(vrPanoramaView!=
null)
{
vrPanoramaView.pauseRendering();
}
}
@Override
protected void onResume() {
super.onResume();
if(vrPanoramaView!=
null)
{
vrPanoramaView.resumeRendering();
}
}
@Override
protected void onDestroy() {
super.onDestroy();
if (vrPanoramaView !=
null) {
vrPanoramaView.shutdown();
}
if (imageTask !=
null && !imageTask.isCancelled()) {
imageTask.cancel(
true);
imageTask =
null;
}
}
}
3. 全景视频显示开发
3.1 vr视频环境搭建
导入需要的三个库 common,comonwidget.videowiget依赖这三个库准备显示使用到全景视频 assets目录下面 例:assets/b.mp4配置大内存选项 android:largeHeap=”true” 可以使用最大内存
3.1.1 导入vr sdk 中的相关库
common,commonwidget,videowidget PS:最新的SDK已经没有这三个文件夹了,使用下一步的依赖库即可
3.1.2 依赖以上三个库
compile
'com.google.vr:sdk-videowidget:1.30.0'
3.1.3 打开内存设置 android:largeHeap=”true”
<application
android:largeHeap="true">
</application>
3.1.4 准备测试使用的全景视频
放置在assets目录 例:assets/congo_2048.mp4
3.2 布局视频控件,并加载视频内容
public class MainActivity extends AppCompatActivity {
private VrVideoView vrVideoView;
private VideoTask videoTask;
private SeekBar seekBar;
private TextView text;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
vrVideoView = (VrVideoView) findViewById(R.id.vr_vv);
videoTask =
new VideoTask();
videoTask.execute(
"congo_2048.mp4");
}
private class VideoTask extends AsyncTask<String, Void, Void> {
@Override
protected Void
doInBackground(String... params) {
VrVideoView.Options options =
new VrVideoView.Options();
options.inputType = VrVideoView.Options.TYPE_STEREO_OVER_UNDER;
options.inputFormat=VrVideoView.Options.FORMAT_HLS;
try {
if (listener ==
null) {
listener =
new VrVideoEventListener() {
@Override
public void onLoadSuccess() {
super.onLoadSuccess();
long max=vrVideoView.getDuration();
seekBar.setMax((
int) max);
seekBar.setProgress(
0);
Toast.makeText(MainActivity.
this,
"准备播放vr", Toast.LENGTH_SHORT).show();
}
@Override
public void onNewFrame() {
super.onNewFrame();
long currentPosition = vrVideoView.getCurrentPosition();
seekBar.setProgress((
int) currentPosition);
String total=String.format(
"%.2f",vrVideoView.getDuration()/
1000f);
String curr=String.format(
"%.2f",vrVideoView.getCurrentPosition()/
1000f);
text.setText(
"播放进度"+curr+
":"+total);
}
private boolean isPause=
true;
@Override
public void onCompletion() {
super.onCompletion();
seekBar.setProgress(
0);
vrVideoView.seekTo(
0);
vrVideoView.pauseVideo();
isPause=
true;
}
@Override
public void onClick() {
super.onClick();
if (isPause) {
vrVideoView.playVideo();
isPause=
false;
}
else {
vrVideoView.pauseVideo();
isPause=
true;
}
}
@Override
public void onLoadError(String errorMessage) {
super.onLoadError(errorMessage);
Toast.makeText(MainActivity.
this,
"加载失败", Toast.LENGTH_SHORT).show();
}
};
vrVideoView.setEventListener(listener);
}
String url=
"http://youkesvideo.oss-cn-hangzhou.aliyuncs.com/movie2/2016/10/11/湄公河行动.Operation.Mekong.2016.TC720P.X264.AAC.Mandarin.CHS.Mp4Ba.mp4";
vrVideoView.loadVideo(Uri.parse(url),options);
}
catch (IOException e) {
e.printStackTrace();
}
return null;
}
}
private VrVideoEventListener listener;
3.3 处理页面退到后台,回到屏幕,页面销毁
@Override
protected void onPause() {
super.onPause();
if (vrVideoView !=
null) {
vrVideoView.pauseRendering();
}
}
@Override
protected void onResume() {
super.onResume();
if (vrVideoView !=
null) {
vrVideoView.resumeRendering();
}
}
@Override
protected void onDestroy() {
super.onDestroy();
if (vrVideoView !=
null) {
vrVideoView.shutdown();
}
if (videoTask !=
null && !videoTask.isCancelled()) {
videoTask.cancel(
false);
videoTask =
null;
}
}
}
3.4 添加进度条相关事件
3.4.1布局查找出控件
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical">
<SeekBar
android:id="@+id/seek_bar"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
<TextView
android:id="@+id/text"
android:background="#AEAEAE"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:gravity="center"
android:textColor="#FFFFFF"
android:textSize="22sp"
android:text="00:00" />
</LinearLayout>
seekBar = (SeekBar) findViewById(R.id.seek_bar);
text = (TextView) findViewById(R.id.text);
3.4.2.加载成功设置最大值
在VrVideoEventListener中的onLoadSuccess处理
3.4.3.在播放过程中不断更新
进度值 onNewFrame 每播放一个画面就调用该方法一次
3.4.4.同步理新文本时间值
在VrVideoEventListener中的onNewFrame处理
3.4.5.播放完成重新播放
在VrVideoEventListener中的onCompletion处理
package com.itheima.appvideo;
import android.os.AsyncTask;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.SeekBar;
import android.widget.TextView;
import android.widget.Toast;
import com.google.vr.sdk.widgets.common.VrWidgetView;
import com.google.vr.sdk.widgets.video.VrVideoEventListener;
import com.google.vr.sdk.widgets.video.VrVideoView;
import java.io.IOException;
public class MainActivity extends AppCompatActivity {
private VrVideoView vrVideoView;
private VideoTask task;
private SeekBar seekBar;
private TextView timeText;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
vrVideoView = (VrVideoView) findViewById(R.id.vr_video_view);
task =
new VideoTask();
task.execute(
"b.mp4");
seekBar = (SeekBar) findViewById(R.id.seekbar);
timeText = (TextView) findViewById(R.id.time);
}
private class VideoTask extends AsyncTask<String, Void, Void> {
@Override
protected Void
doInBackground(String... params) {
VrVideoView.Options options =
new VrVideoView.Options();
options.inputType = VrVideoView.Options.TYPE_STEREO_OVER_UNDER;
options.inputFormat = VrVideoView.Options.FORMAT_DEFAULT;
try {
VrVideoEventListener listener =
new VrVideoEventListener() {
@Override
public void onLoadSuccess() {
super.onLoadSuccess();
Toast.makeText(MainActivity.
this,
"准备放3d视频", Toast.LENGTH_SHORT).show();
upProgress();
}
@Override
public void onLoadError(String errorMessage) {
super.onLoadError(errorMessage);
Toast.makeText(MainActivity.
this,
"视频加载失败" + errorMessage, Toast.LENGTH_SHORT).show();
}
@Override
public void onNewFrame() {
super.onNewFrame();
upProgress();
}
private boolean isPause =
false;
@Override
public void onCompletion() {
super.onCompletion();
vrVideoView.seekTo(
0);
vrVideoView.pauseVideo();
isPause =
true;
upProgress();
}
@Override
public void onClick() {
super.onClick();
if (isPause) {
isPause =
false;
vrVideoView.playVideo();
}
else {
isPause =
true;
vrVideoView.pauseVideo();
}
}
};
vrVideoView.setEventListener(listener);
vrVideoView.loadVideoFromAsset(params[
0], options);
}
catch (IOException e) {
e.printStackTrace();
}
return null;
}
private void upProgress() {
long max = vrVideoView.getDuration();
long currPosition = vrVideoView.getCurrentPosition();
seekBar.setMax((
int) max);
seekBar.setProgress((
int) currPosition);
timeText.setText(String.format(
"%.2f", currPosition /
1000f) +
"/" + String.format(
"%.2f", max /
1000f));
}
}
@Override
protected void onPause() {
super.onPause();
if (vrVideoView !=
null) {
vrVideoView.pauseRendering();
}
}
@Override
protected void onResume() {
super.onResume();
if (vrVideoView !=
null) {
vrVideoView.resumeRendering();
}
}
@Override
protected void onDestroy() {
super.onDestroy();
if (vrVideoView !=
null) {
vrVideoView.shutdown();
}
if (task !=
null && !task.isCancelled()) {
task.cancel(
true);
task =
null;
}
}
}
GitHub:https://github.com/JackChen1999/GoogleVR