关键词:
【中文标题】如何使用适用于 Android 的 vlc lib 在视频流中选择音频通道和字幕?【英文标题】:How to select audio channels and subtitles in a video stream, using vlc lib for Android? 【发布时间】:2015-11-26 11:23:25 【问题描述】:我正在使用 Android 中的流式视频播放器,需要显示此流的字幕(如果存在),还需要为用户提供一种选择流中存在的音频通道的方法。 我正在从 Android MediaPlayer 切换到 VlcLib 以便能够做到这一点。 但我找不到任何关于如何做到这一点的例子。
有人已经这样做了吗?你能分享一些例子或提供线索吗?
非常感谢您的帮助。
我的代码是基于互联网上的一个示例:
package com.wass08.vlcsimpleplayer;
import com.wass08.vlcsimpleplayer.util.SystemUiHider;
import android.annotation.TargetApi;
import android.app.Activity;
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.content.pm.ActivityInfo;
import android.content.res.Configuration;
import android.graphics.PixelFormat;
import android.os.Build;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.util.DisplayMetrics;
import android.util.Log;
import android.view.Gravity;
import android.view.MotionEvent;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.view.ViewGroup;
import android.view.Window;
import android.view.WindowManager;
import android.widget.FrameLayout;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.ScrollView;
import android.widget.SeekBar;
import android.widget.TextView;
import android.widget.Toast;
import org.videolan.libvlc.EventHandler;
import org.videolan.libvlc.IVideoPlayer;
import org.videolan.libvlc.LibVLC;
import org.videolan.libvlc.Media;
import org.videolan.libvlc.MediaList;
import java.lang.ref.WeakReference;
import java.util.ArrayList;
import java.util.Map;
/**
* An example full-screen activity that shows and hides the system UI (i.e.
* status bar and navigation/system bar) with user interaction.
*
* @see SystemUiHider
*/
public class FullscreenVlcPlayer extends Activity implements SurfaceHolder.Callback, IVideoPlayer
private String urlToStream;
// Display Surface
private LinearLayout vlcContainer;
private SurfaceView mSurface;
private SurfaceView mSubtitlesSurface;
private SurfaceHolder holder;
private SurfaceHolder mSubtitlesSurfaceHolder;
// Overlay / Controls
private FrameLayout vlcOverlay;
private ImageView vlcButtonPlayPause;
private Handler handlerOverlay;
private Runnable runnableOverlay;
private Handler handlerSeekbar;
private Runnable runnableSeekbar;
private SeekBar vlcSeekbar;
private TextView vlcDuration;
private TextView overlayTitle;
// media player
private LibVLC libvlc;
private int mVideoWidth;
private int mVideoHeight;
private final static int VideoSizeChanged = -1;
@Override
protected void onCreate(Bundle savedInstanceState)
super.onCreate(savedInstanceState);
// Retrieve our url
Bundle b = getIntent().getExtras();
urlToStream = b.getString("url", null);
// HIDE THE ACTION BAR
getActionBar().hide();
// SETUP THE UI
setContentView(R.layout.activity_fullscreen_vlc_player);
// VLC
vlcContainer = (LinearLayout) findViewById(R.id.vlc_container);
mSurface = (SurfaceView) findViewById(R.id.vlc_surface);
mSubtitlesSurface = (SurfaceView) findViewById(R.id.subtitles_surface);
// OVERLAY / CONTROLS
vlcOverlay = (FrameLayout) findViewById(R.id.vlc_overlay);
vlcButtonPlayPause = (ImageView) findViewById(R.id.vlc_button_play_pause);
vlcSeekbar = (SeekBar) findViewById(R.id.vlc_seekbar);
vlcDuration = (TextView) findViewById(R.id.vlc_duration);
overlayTitle = (TextView) findViewById(R.id.vlc_overlay_title);
overlayTitle.setText(urlToStream);
// AUTOSTART
playMovie();
private void setSubtitles()
/* Only show the subtitles surface when using "Full Acceleration" mode */
//if (libvlc.getHardwareAcceleration() == 2)
//
mSubtitlesSurface.setVisibility(View.VISIBLE);
//
Map<Integer, String> mSubtitleTracksList;
Map<Integer, String> mAudioTracksList;
int numSubtitleTracks = libvlc.getSpuTracksCount();
int numAudioTracks = libvlc.getAudioTracksCount();
mSubtitleTracksList = libvlc.getSpuTrackDescription();
mAudioTracksList = libvlc.getAudioTrackDescription();
if (mSubtitleTracksList != null && mSubtitleTracksList.size() > 0)
if (mAudioTracksList != null && mAudioTracksList.size() > 0)
Log.d("LIBVLC_PLAYER", "Number of Subtitles Tracks = " + numSubtitleTracks);
Log.d("LIBVLC_PLAYER", "Number of Audio Tracks = " + numAudioTracks);
int i = 0;
for (Map.Entry<Integer, String> entry : mSubtitleTracksList.entrySet())
Log.d("LIBVLC_PLAYER", "Subtitle Track " + i + " = " + entry.getValue().toString());
i++;
libvlc.setSpuTrack(0);
i = 0;
for (Map.Entry<Integer, String> entry2 : mAudioTracksList.entrySet())
Log.d("LIBVLC_PLAYER", "Audio Track " + i + " = " + entry2.getValue().toString());
i++;
libvlc.setAudioTrack(0);
private final SurfaceHolder.Callback mSubtitlesSurfaceCallback = new SurfaceHolder.Callback()
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height)
if (libvlc != null)
libvlc.attachSubtitlesSurface(holder.getSurface());
Log.d("LIBVLC_PLAYER", "SurfaceHolder.Callback: width: " + width + " height: " + height);
@Override
public void surfaceCreated(SurfaceHolder holder)
Log.d("LIBVLC_PLAYER", "SurfaceHolder.Callback -> surfaceCreated");
@Override
public void surfaceDestroyed(SurfaceHolder holder)
libvlc.detachSubtitlesSurface();
Log.d("LIBVLC_PLAYER", "SurfaceHolder.Callback -> surfaceDestroyed");
;
private void showOverlay()
vlcOverlay.setVisibility(View.VISIBLE);
private void hideOverlay()
vlcOverlay.setVisibility(View.GONE);
private void setupControls()
getActionBar().hide();
// PLAY PAUSE
vlcButtonPlayPause.setOnClickListener(
new View.OnClickListener()
@Override
public void onClick(View view)
if (libvlc.isPlaying())
libvlc.pause();
vlcButtonPlayPause.setImageDrawable(getResources().getDrawable(R.drawable.ic_action_play_over_video));
else
libvlc.play();
vlcButtonPlayPause.setImageDrawable(getResources().getDrawable(R.drawable.ic_action_pause_over_video));
);
// SEEKBAR
handlerSeekbar = new Handler();
runnableSeekbar = new Runnable()
@Override
public void run()
if (libvlc != null)
long curTime = libvlc.getTime();
long totalTime = (long) (curTime / libvlc.getPosition());
int minutes = (int) (curTime / (60 * 1000));
int seconds = (int) ((curTime / 1000) % 60);
int endMinutes = (int) (totalTime / (60 * 1000));
int endSeconds = (int) ((totalTime / 1000) % 60);
String duration = String.format("%02d:%02d / %02d:%02d", minutes, seconds, endMinutes, endSeconds);
vlcSeekbar.setProgress((int) (libvlc.getPosition() * 100));
vlcDuration.setText(duration);
handlerSeekbar.postDelayed(runnableSeekbar, 1000);
;
runnableSeekbar.run();
vlcSeekbar.setOnSeekBarChangeListener(
new SeekBar.OnSeekBarChangeListener()
@Override
public void onProgressChanged(SeekBar seekBar, int i, boolean b)
Log.v("NEW POS", "pos is : " + i);
//if (i != 0)
// libvlc.setPosition(((float) i / 100.0f));
@Override
public void onStartTrackingTouch(SeekBar seekBar)
@Override
public void onStopTrackingTouch(SeekBar seekBar)
);
// OVERLAY
handlerOverlay = new Handler();
runnableOverlay = new Runnable()
@Override
public void run()
vlcOverlay.setVisibility(View.GONE);
toggleFullscreen(true);
;
final long timeToDisappear = 3000;
handlerOverlay.postDelayed(runnableOverlay, timeToDisappear);
vlcContainer.setOnClickListener(
new View.OnClickListener()
@Override
public void onClick(View view)
vlcOverlay.setVisibility(View.VISIBLE);
handlerOverlay.removeCallbacks(runnableOverlay);
handlerOverlay.postDelayed(runnableOverlay, timeToDisappear);
);
public void playMovie()
if (libvlc != null && libvlc.isPlaying())
return;
vlcContainer.setVisibility(View.VISIBLE);
holder = mSurface.getHolder();
holder.addCallback(this);
createPlayer(urlToStream);
private void toggleFullscreen(boolean fullscreen)
WindowManager.LayoutParams attrs = getWindow().getAttributes();
if (fullscreen)
attrs.flags |= WindowManager.LayoutParams.FLAG_FULLSCREEN;
vlcContainer.setSystemUiVisibility(
View.SYSTEM_UI_FLAG_LAYOUT_STABLE
| View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION
| View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
| View.SYSTEM_UI_FLAG_HIDE_NAVIGATION
| View.SYSTEM_UI_FLAG_FULLSCREEN
| View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY
);
else
attrs.flags &= ~WindowManager.LayoutParams.FLAG_FULLSCREEN;
getWindow().setAttributes(attrs);
@Override
public void onConfigurationChanged(Configuration newConfig)
super.onConfigurationChanged(newConfig);
setSize(mVideoWidth, mVideoHeight);
@Override
protected void onResume()
super.onResume();
@Override
protected void onPause()
super.onPause();
//releasePlayer();
@Override
protected void onDestroy()
super.onDestroy();
releasePlayer();
/**
* **********
* Surface
* ***********
*/
public void surfaceCreated(SurfaceHolder holder)
public void surfaceChanged(SurfaceHolder surfaceholder, int format,
int width, int height)
if (libvlc != null)
libvlc.attachSurface(surfaceholder.getSurface(), this);
public void surfaceDestroyed(SurfaceHolder surfaceholder)
private void setSize(int width, int height)
mVideoWidth = width;
mVideoHeight = height;
if (mVideoWidth * mVideoHeight <= 1)
return;
// get screen size
int w = getWindow().getDecorView().getWidth();
int h = getWindow().getDecorView().getHeight();
// getWindow().getDecorView() doesn't always take orientation into
// account, we have to correct the values
boolean isPortrait = getResources().getConfiguration().orientation == Configuration.ORIENTATION_PORTRAIT;
if (w > h && isPortrait || w < h && !isPortrait)
int i = w;
w = h;
h = i;
float videoAR = (float) mVideoWidth / (float) mVideoHeight;
float screenAR = (float) w / (float) h;
if (screenAR < videoAR)
h = (int) (w / videoAR);
else
w = (int) (h * videoAR);
// force surface buffer size
if (holder != null)
holder.setFixedSize(mVideoWidth, mVideoHeight);
// set display size
ViewGroup.LayoutParams lp = mSurface.getLayoutParams();
lp.width = w;
lp.height = h;
mSurface.setLayoutParams(lp);
mSurface.invalidate();
@Override
public void setSurfaceSize(int width, int height, int visible_width,
int visible_height, int sar_num, int sar_den)
Message msg = Message.obtain(mHandler, VideoSizeChanged, width, height);
msg.sendToTarget();
/**
* **********
* Player
* ***********
*/
private void createPlayer(String media)
releasePlayer();
setupControls();
try
if (media.length() > 0)
Toast toast = Toast.makeText(this, media, Toast.LENGTH_LONG);
toast.setGravity(
Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL, 0,
0
);
toast.show();
// Create a new media player
libvlc = LibVLC.getInstance();
libvlc.setHardwareAcceleration(LibVLC.HW_ACCELERATION_FULL);
libvlc.eventVideoPlayerActivityCreated(true);
libvlc.setSubtitlesEncoding("");
libvlc.setAout(LibVLC.AOUT_OPENSLES);
libvlc.setTimeStretching(true);
libvlc.setChroma("RV32");
libvlc.setVerboseMode(true);
LibVLC.restart(this);
EventHandler.getInstance().addHandler(mHandler);
holder.setFormat(PixelFormat.RGBX_8888);
holder.setKeepScreenOn(true);
mSubtitlesSurfaceHolder = mSubtitlesSurface.getHolder();
mSubtitlesSurfaceHolder.setFormat(PixelFormat.RGBA_8888);
mSubtitlesSurface.setZOrderMediaOverlay(true);
//mSubtitlesSurface.setZOrderOnTop(true);
mSubtitlesSurfaceHolder.setFormat(PixelFormat.TRANSLUCENT);
mSubtitlesSurfaceHolder.addCallback(mSubtitlesSurfaceCallback);
MediaList list = libvlc.getMediaList();
list.clear();
list.add(new Media(libvlc, LibVLC.PathToURI(media)), false);
libvlc.setVolume(100);
libvlc.playIndex(0);
catch (Exception e)
Toast.makeText(this, "Could not create Vlc Player", Toast.LENGTH_LONG).show();
private void releasePlayer()
if (handlerSeekbar != null && runnableSeekbar != null)
handlerSeekbar.removeCallbacks(runnableSeekbar);
EventHandler.getInstance().removeHandler(mHandler);
if (libvlc == null)
return;
libvlc.stop();
libvlc.detachSurface();
holder = null;
libvlc.closeAout();
mVideoWidth = 0;
mVideoHeight = 0;
/**
* **********
* Events
* ***********
*/
private Handler mHandler = new MyHandler(this);
private static class MyHandler extends Handler
private WeakReference<FullscreenVlcPlayer> mOwner;
public MyHandler(FullscreenVlcPlayer owner)
mOwner = new WeakReference<FullscreenVlcPlayer>(owner);
@Override
public void handleMessage(Message msg)
FullscreenVlcPlayer player = mOwner.get();
// Player events
if (msg.what == VideoSizeChanged)
player.setSize(msg.arg1, msg.arg2);
return;
// Libvlc events
Bundle b = msg.getData();
switch (b.getInt("event"))
case EventHandler.MediaPlayerEndReached:
player.releasePlayer();
break;
case EventHandler.MediaPlayerPlaying:
player.setSubtitles();
break;
case EventHandler.MediaPlayerPaused:
case EventHandler.MediaPlayerStopped:
default:
break;
【问题讨论】:
【参考方案1】:这不适用于 Android,但我认为可以帮助您
/* Create a new item */
media = libvlc_media_new_path (inst, "BlackBerry.3gp");
/* Create a media player playing environement */
mediaplayer = libvlc_media_player_new_from_media (media);
/* play the media_player */
libvlc_media_player_play (mediaplayer);
//*****************************************************************************
//*****************************************************************************
//********* Subtitle section. Must be after libvlc_media_player_play **********
//*****************************************************************************
//*****************************************************************************
int temp = libvlc_video_set_subtitle_file(mediaplayer, "NewSubFile.srt");
printf("sub status=%d\n", temp);
temp = libvlc_video_set_subtitle_file(mediaplayer, "BlackBerry.srt");
printf("sub status=%d\n", temp);
int sub_count = libvlc_video_get_spu_count(mediaplayer);
printf("sub count=%d\n", sub_count);
//*****************************************************************************
//*****************************************************************************
//*****************************************************************************
//*****************************************************************************
//*****************************************************************************
sleep (10); /* Let it play a bit */
/* No need to keep the media now */
libvlc_media_release (media);
/* Stop playing */
libvlc_media_player_stop (mediaplayer);
/* Free the media_player */
libvlc_media_player_release (mediaplayer);
【讨论】:
如何使 Android 的 aapt 和 adb 在没有 ia32-libs 的 64 位 Ubuntu 上工作(适用于版本 12、13 和 14)
】如何使Android的aapt和adb在没有ia32-libs的64位Ubuntu上工作(适用于版本12、13和14)【英文标题】:HowtomakeAndroid\'saaptandadbworkon64-bitUbuntuwithoutia32-libs(worksforversions12,13and14)【发布时间】:2013-10-3115:44:28【问题描述】:由于Ubuntu13.10不... 查看详情
如何使用适用于 Android 的 Fabric Sdk 从 Twitter 注销
】如何使用适用于Android的FabricSdk从Twitter注销【英文标题】:HowtologoutfromtwitterusingFabricSdkforandroid【发布时间】:2015-06-2622:19:15【问题描述】:我用过Twitter.getSessionManager().clearActiveSession();这不起作用,下次我使用twitter登录时,它... 查看详情
如何使用适用于 Android 的 OpenCV 减少实时视频序列中的运动效果?
】如何使用适用于Android的OpenCV减少实时视频序列中的运动效果?【英文标题】:howcanIreducemovementeffectsinarealtimesequenceofvideousingOpenCVforAndroid?【发布时间】:2014-07-3014:53:45【问题描述】:我是Android上OpenCV的新编程人员。我最近几天... 查看详情
适用于 Android 的 Mono - 支持的架构 - 放弃 x86?
】适用于Android的Mono-支持的架构-放弃x86?【英文标题】:MonoforAndroid-SupportedArchitecture-Dropx86?【发布时间】:2012-10-2722:44:12【问题描述】:我们有一个近乎完整的应用程序,并正在努力压缩GooglePlay的50MB下载限制。在检查我们的APK... 查看详情
适用于 Android 的 Azure 通知中心:如何使用后台服务处理数据消息?
】适用于Android的Azure通知中心:如何使用后台服务处理数据消息?【英文标题】:AzureNotificationHubsforAndroid:HowdoIhandleData-MessageswithaBackground-Service?【发布时间】:2021-09-2920:23:06【问题描述】:我正在尝试制作一个能够处理Azure通知... 查看详情
如何使用适用于 Android 的 Adobe AIR 设置 Admob
】如何使用适用于Android的AdobeAIR设置Admob【英文标题】:HowtoSetupAdmobwithAdobeAIRforAndroid【发布时间】:2014-04-0811:55:08【问题描述】:我目前完成了一个项目并将其放在GooglePlay商店中。我使用AdobeFlashCS6和FlashDevelopAS3创建了... 查看详情
如何使用适用于 Android V2 的 Google Maps 处理地图移动端?
】如何使用适用于AndroidV2的GoogleMaps处理地图移动端?【英文标题】:HowcanIhandlemapmoveendusingGoogleMapsforAndroidV2?【发布时间】:2012-11-2200:46:45【问题描述】:我想在地图中心更改后立即对地址进行地理编码。如何使用新的Android版Goog... 查看详情
如何使用适用于 Android 的 Push sharp 库一次发送多个推送通知?
】如何使用适用于Android的Pushsharp库一次发送多个推送通知?【英文标题】:HowtosendmultiplePushnotificationsatatimeUsingPushsharplibraryforAndroid?【发布时间】:2014-09-0506:32:50【问题描述】:我正在使用PushSharp库向Android设备发送推送通知。有... 查看详情
如何使用 Phonegap 插件(适用于 Android)向 API 发出简单的 json 发布请求?
】如何使用Phonegap插件(适用于Android)向API发出简单的json发布请求?【英文标题】:HowtousePhonegapPlugins(ForAndroid)tomakesimplejsonpostrequesttoAPI?【发布时间】:2017-01-2608:50:17【问题描述】:我是Android编程的初学者,所以请与我裸露。... 查看详情
如何使用适用于 Gingerbread 和 Froyo 的 ICS UI 创建 Android 应用程序?
】如何使用适用于Gingerbread和Froyo的ICSUI创建Android应用程序?【英文标题】:HowToCreateaAndroidAppWithICSUIThatWorksWithGingerbreadandFroyo?【发布时间】:2012-03-2709:15:31【问题描述】:HackerNewsbeta之类的应用如何在其应用上使用IceCreamSandwich4.0... 查看详情
如何使用适用于 Android 和 iOS 的 Xamarin 表单在特定频道上的 youtube 上上传视频?
】如何使用适用于Android和iOS的Xamarin表单在特定频道上的youtube上上传视频?【英文标题】:HowtouploadvideoonyoutubeonaspecificchannelusingXamarinformsforAndroidandiOSboth?【发布时间】:2021-10-3015:43:50【问题描述】:有没有什么方法可以使用适用... 查看详情
如何从 Maven 获取适用于 Android 的 JAR
】如何从Maven获取适用于Android的JAR【英文标题】:HowtogetJARfromMavenforAndroid【发布时间】:2015-09-2510:28:24【问题描述】:我们如何从Mavenrepo获取jar文件/库文件。Maven提供AAR文件。我知道使用AndroidStudio是最好的,但我的系统遇到了... 查看详情
适用于 Android 的 WebP [关闭]
】适用于Android的WebP[关闭]【英文标题】:WebPforAndroid[closed]【发布时间】:2011-10-2508:25:52【问题描述】:有没有关于如何在Android上使用WebP的示例?我试图获取webp图像列表并将它们显示在带有图像视图的列表视图中。我知道有一... 查看详情
如何使用适用于 Android 的 WebRTC Native Code 实现 3 路电话会议视频聊天?
】如何使用适用于Android的WebRTCNativeCode实现3路电话会议视频聊天?【英文标题】:Howtoimplement3-wayconferencecallvideochatwithWebRTCNativeCodeforAndroid?【发布时间】:2015-12-3122:41:05【问题描述】:我正在尝试使用WebRTCNativeCodepackageforAndroid(... 查看详情
已安装的适用于 Windows 10 64 位的 OpenCV3.2.0 中缺少许多 .lib 文件
...于opencv库,但是我从网上下载的代码只有opencv_core320.lib。如何安装opencv以便访问VisualStudio2 查看详情
如何使用适用于 android 的 google play 游戏服务实现后端服务器身份验证
】如何使用适用于android的googleplay游戏服务实现后端服务器身份验证【英文标题】:howtoimplementbackendserverauthenticationwithgoogleplaygamesservicesforandroid【发布时间】:2016-08-1921:06:55【问题描述】:也许我忽略了一些东西,但是通过Play游... 查看详情
适用于 Android 的 Firebase 实时(在线)数据库的安全性如何?
】适用于Android的Firebase实时(在线)数据库的安全性如何?【英文标题】:HowSecureisFirebaseReal-time(Online)databaseforAndroid?【发布时间】:2018-03-2322:30:15【问题描述】:我最近刚刚开始使用Google的firebase作为android应用数据的在线集中... 查看详情
GCM 是如何工作的? (适用于 Android 的谷歌云消息传递)
】GCM是如何工作的?(适用于Android的谷歌云消息传递)【英文标题】:HowdoesGCMwork?(googlecloudmessagingforAndroid)【发布时间】:2014-05-2412:51:34【问题描述】:我有一个android应用程序,我使用RESTAPI(djangorest框架)连接到我的服务器这... 查看详情