简单 介绍
每天都能看到关于VR的新闻,但是在日常生活中,关于VR的相关却并不多见,虽说今年是VR爆发的元年,然而对普通消费者来说却毫无知觉,VR似乎离我们这么近,又那么的远。
目前VR设备主要分为三大类:VR头盔、VR一体机、VR盒子(类别名称还没有统一叫法)。
目前VR内容的平台虽然不多,但也足够体验用,而且资源也算丰富和全面。下面推荐的内容平台均为APP应用,基本大部分都支持iOS和安卓端使用,比较热门的有:UtoVR、3D%E6%92%AD%E6%92%AD/’ target=’_blank’>3D播播、暴风魔镜、橙子VR、VR热播等(具体见下截图)。
VR开发基础—全景图
搭建VR图片开发的环境
导入从github搜索下载的 google vr sdk (点击查看) 里面的引用库
1、导入谷歌官方提供的库:
commo
commonwidget
panowidget(全景图)
videowidget(全景视频)
2、出现类未定义错误的缺少库
compile 'com.google.protobuf.nano:protobuf-javanano:3.0.0-alpha-7'
compile project(':common')
compile project(':commonwidget')
compile project(':panowidget')
3、准备全景图片用来测试代码 放在assets目录下面,例assets/a.jpg
4、对当前应用进行内存设置,希望应用可使用最大内存 避免OOM
<application android:largeHeap="true"
例如:
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
加载全景图片到内存中成为Bitmap(bitmap是图片在内存中的表示对象),展示在全景图片控件
布局全景图片控件
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android=""
xmlns:tools=""
android:id="@+id/activity_main"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="com.itheima.a001vrpanodemo.MainActivity">
<com.google.vr.sdk.widgets.pano.VrPanoramaView
android:id="@+id/vr_pano"
android:layout_width="match_parent"
android:layout_height="match_parent"></com.google.vr.sdk.widgets.pano.VrPanoramaView>
</RelativeLayout>
查找控件
vrPanoramaView = (VrPanoramaView) findViewById(R.id.vr_pano)
因为图片比较大,希望在异步线程里面加载,防止占用主线程
imageTask = new ImageTask();
imageTask.execute("a.jpg");
private class ImageTask extends AsyncTask<String,Void,Bitmap>
{
@Override
protected Bitmap doInBackground(String... params) {
try {
InputStream inputStream = getAssets().open(params[0]);
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;
VrPanoramaEventListener 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);
}
}
}
处理全景控件展示细节
@Override
protected void onPause() {
super.onPause();
if (vrPanoramaView != null) {
vrPanoramaView.pauseRendering();
}
}
@Override
protected void onResume() {
super.onResume();
if (vrPanoramaView != null) {
vrPanoramaView.resumeRendering();
}
vrPanoramaView.setInfoButtonEnabled(false);
vrPanoramaView.setFullscreenButtonEnabled(false);
vrPanoramaView.setDisplayMode(VrWidgetView.DisplayMode.FULLSCREEN_STEREO);
}
@Override
protected void onDestroy() {
super.onDestroy();
if (vrPanoramaView != null) {
vrPanoramaView.shutdown();
}
if (imageTask != null && !imageTask.isCancelled()) {
imageTask.cancel(true);
imageTask = null;
}
}
完整代码Demo1
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);
imageTask = new ImageTask();
imageTask.execute("a.jpg");
}
private class ImageTask extends AsyncTask<String,Void,Bitmap>
{
@Override
protected Bitmap doInBackground(String... params) {
try {
InputStream inputStream = getAssets().open(params[0]);
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;
VrPanoramaEventListener 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);
}
}
}
@Override
protected void onPause() {
super.onPause();
if (vrPanoramaView != null) {
vrPanoramaView.pauseRendering();
}
}
@Override
protected void onResume() {
super.onResume();
if (vrPanoramaView != null) {
vrPanoramaView.resumeRendering();
}
vrPanoramaView.setInfoButtonEnabled(false);
vrPanoramaView.setFullscreenButtonEnabled(false);
vrPanoramaView.setDisplayMode(VrWidgetView.DisplayMode.FULLSCREEN_STEREO);
}
@Override
protected void onDestroy() {
super.onDestroy();
if (vrPanoramaView != null) {
vrPanoramaView.shutdown();
}
if (imageTask != null && !imageTask.isCancelled()) {
imageTask.cancel(true);
imageTask = null;
}
}
}
全景视频展示
搭建全景视频开发环境
导入github搜索google vr sdk 的开发库 common,commonwidget,videowidget
导入API跟全景图片一样,特别是开发全景视频就导入videowidget(全景视频)、如果是开发全景图片就导入panowidget(全景图),根据开发的需要。
依赖 当前三个库缺少的api
compile 'com.google.android.exoplayer:exoplayer:r1.5.10'
compile 'com.google.protobuf.nano:protobuf-javanano:3.0.0-alpha-7'
compile project(':common')
compile project(':commonwidget')
compile project(':videowidget')
准备全景视频来测试程序 ,放到assets下面 例 assets/b.mp4
设置应用的内存选项,在内存不足时可以使用最大内存.
在线程加载全景视频,显示在全景视频控件上
界面布局
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android=""
xmlns:tools=""
android:id="@+id/activity_main"
android:layout_width="match_parent"
android:layout_height="match_parent">
<com.google.vr.sdk.widgets.video.VrVideoView
android:id="@+id/vr_video_view"
android:layout_width="match_parent"
android:layout_height="match_parent">
</com.google.vr.sdk.widgets.video.VrVideoView>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical">
<SeekBar
android:id="@+id/seekbar"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
<TextView
android:id="@+id/time"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="#AEAEAE"
android:gravity="center"
android:text="00/100s"
android:textColor="#FFFFFF" />
</LinearLayout>
<ProgressBar
android:id="@+id/loading"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true" />
</RelativeLayout>
布局控件并查找
vrVideoView = (VrVideoView) findViewById(R.id.vr_video_view)
seekBar = (SeekBar) findViewById(R.id.seekbar)
timeView = (TextView) findViewById(R.id.time)
progressBar = (ProgressBar) findViewById(R.id.loading)
创建子线程加载资源
videoTask = new VideoTask();
videoTask.execute("b.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_MONO;
options.inputFormat = VrVideoView.Options.FORMAT_DEFAULT;
VrVideoEventListener listener = new VrVideoEventListener() {
@Override
public void onLoadSuccess() {
super.onLoadSuccess();
progressBar.setVisibility(View.GONE);
}
@Override
public void onLoadError(String errorMessage) {
super.onLoadError(errorMessage);
progressBar.setVisibility(View.GONE);
Toast.makeText(MainActivity.this, errorMessage, Toast.LENGTH_SHORT).show();
}
@Override
public void onNewFrame() {
super.onNewFrame();
seekBar.setMax((int) vrVideoView.getDuration());
seekBar.setProgress((int) vrVideoView.getCurrentPosition());
String curr = String.format("%.2f", vrVideoView.getCurrentPosition() / 1000f);
String total = String.format("%.2f", vrVideoView.getDuration() / 1000f);
timeView.setText(curr + "/" + total + "s");
}
@Override
public void onCompletion() {
super.onCompletion();
vrVideoView.seekTo(0);
vrVideoView.pauseVideo();
isPasuse = true;
Toast.makeText(MainActivity.this, "播放完成,是否重播", Toast.LENGTH_SHORT).show();
}
private boolean isPasuse = false;
@Override
public void onClick() {
super.onClick();
if (isPasuse) {
vrVideoView.playVideo();
isPasuse = false;
} else {
vrVideoView.pauseVideo();
isPasuse = true;
}
}
};
try {
vrVideoView.setEventListener(listener);
vrVideoView.loadVideo(Uri.parse(params[0]),options);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
}
根据用户的操作来调整显示(性能优化,防止黑屏)
@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(true);
videoTask = null;
}
}
界面设置
在onCreate
vrVideoView.setInfoButtonEnabled(false);
vrVideoView.setFullscreenButtonEnabled(false);
vrVideoView.setDisplayMode(VrWidgetView.DisplayMode.EMBEDDED);
完整代码Demo2
public class MainActivity extends AppCompatActivity {
private VrVideoView vrVideoView;
private VideoTask videoTask;
private ProgressBar progressBar;
private TextView timeView;
private SeekBar seekBar;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
vrVideoView = (VrVideoView) findViewById(R.id.vr_video_view);
videoTask = new VideoTask();
videoTask.execute("");
seekBar = (SeekBar) findViewById(R.id.seekbar);
timeView = (TextView) findViewById(R.id.time);
progressBar = (ProgressBar) findViewById(R.id.loading);
vrVideoView.setInfoButtonEnabled(false);
vrVideoView.setFullscreenButtonEnabled(false);
vrVideoView.setDisplayMode(VrWidgetView.DisplayMode.EMBEDDED);
}
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_MONO;
options.inputFormat = VrVideoView.Options.FORMAT_DEFAULT;
VrVideoEventListener listener = new VrVideoEventListener() {
@Override
public void onLoadSuccess() {
super.onLoadSuccess();
progressBar.setVisibility(View.GONE);
}
@Override
public void onLoadError(String errorMessage) {
super.onLoadError(errorMessage);
progressBar.setVisibility(View.GONE);
Toast.makeText(MainActivity.this, errorMessage, Toast.LENGTH_SHORT).show();
}
@Override
public void onNewFrame() {
super.onNewFrame();
seekBar.setMax((int) vrVideoView.getDuration());
seekBar.setProgress((int) vrVideoView.getCurrentPosition());
String curr = String.format("%.2f", vrVideoView.getCurrentPosition() / 1000f);
String total = String.format("%.2f", vrVideoView.getDuration() / 1000f);
timeView.setText(curr + "/" + total + "s");
}
@Override
public void onCompletion() {
super.onCompletion();
vrVideoView.seekTo(0);
vrVideoView.pauseVideo();
isPasuse = true;
Toast.makeText(MainActivity.this, "播放完成,是否重播", Toast.LENGTH_SHORT).show();
}
private boolean isPasuse = false;
@Override
public void onClick() {
super.onClick();
if (isPasuse) {
vrVideoView.playVideo();
isPasuse = false;
} else {
vrVideoView.pauseVideo();
isPasuse = true;
}
}
};
try {
vrVideoView.setEventListener(listener);
vrVideoView.loadVideo(Uri.parse(params[0]),options);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
}
@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(true);
videoTask = null;
}
}
}
获取网络资源的全景图片和全景视频
1.要求获取网络图片,获取网络流是差别
vrPanoramaView = new VrPanoramaView(this);
setContentView(vrPanoramaView);
imageTask = new ImageTask();
imageTask.execute(item.url);
}
private class ImageTask extends AsyncTask<String,Void,Bitmap>
{
@Override
protected Bitmap doInBackground(String... params) {
try {
URL url=new URL(params[0]);
HttpURLConnection connection= (HttpURLConnection) url.openConnection();
connection.setRequestMethod("GET");
connection.setConnectTimeout(10000);
InputStream inputStream =connection.getInputStream();
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_MONO;
VrPanoramaEventListener listener=new VrPanoramaEventListener(){
@Override
public void onLoadError(String errorMessage) {
super.onLoadError(errorMessage);
Toast.makeText(VrImagePlayActivity.this, errorMessage, Toast.LENGTH_SHORT).show();} /
/3.1.2.加载bitmap成功
@Override
public void onLoadSuccess() {
super.onLoadSuccess();
Toast.makeText(VrImagePlayActivity.this, "进入VR图片显示...", Toast.LENGTH_SHORT).show();
}
};
vrPanoramaView.setEventListener(listener);
vrPanoramaView.loadImageFromBitmap(bitmap,options);
}
}
}
2.全景的网络视频
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_MONO;
options.inputFormat = VrVideoView.Options.FORMAT_DEFAULT;
VrVideoEventListener listener = new VrVideoEventListener() {
@Override
public void onLoadSuccess() {
super.onLoadSuccess();
progressBar.setVisibility(View.GONE);
}
@Override
public void onLoadError(String errorMessage) {
super.onLoadError(errorMessage);
progressBar.setVisibility(View.GONE);
Toast.makeText(MainActivity.this, errorMessage, Toast.LENGTH_SHORT).show();
}
@Override
public void onNewFrame() {
super.onNewFrame();
seekBar.setMax((int) vrVideoView.getDuration());
seekBar.setProgress((int) vrVideoView.getCurrentPosition());
String curr = String.format("%.2f", vrVideoView.getCurrentPosition() / 1000f);
String total = String.format("%.2f", vrVideoView.getDuration() / 1000f);
timeView.setText(curr + "/" + total + "s");
}
@Override
public void onCompletion() {
super.onCompletion();
vrVideoView.seekTo(0);
vrVideoView.pauseVideo();
isPasuse = true;
Toast.makeText(MainActivity.this, "播放完成,是否重播", Toast.LENGTH_SHORT).show();
}
private boolean isPasuse = false;
@Override
public void onClick() {
super.onClick();
if (isPasuse) {
vrVideoView.playVideo();
isPasuse = false;
} else {
vrVideoView.pauseVideo();
isPasuse = true;
}
}
};
try {
vrVideoView.setEventListener(listener);
vrVideoView.loadVideo(Uri.parse(params[0]),options);
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
}
vrVideoView.loadVideo(Uri.parse(params[0]),options);
3.注意网络问题
权限
<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
手机联网
服务器开启