【发布时间】:2012-09-07 05:37:59
【问题描述】:
我有一个Activity,它是一个WebView。我里面有一个WebChromeClient。在其中,有几个回调旨在返回处理视频位的MediaPlayer。例如:
@Override
public void onPrepared(MediaPlayer mp) {
Log.i(TAG, " -------------> onPrepared");
}
当我使用 HTML <video> 标签(通过注入)将 MP4 流加载到 WebView 时,这些无法触发。
当我finish() 活动时,logcat 会报告:
09-13 23:55:24.590: E/MediaPlayer(7949): mOnBufferingUpdateListener is null. Failed to send MEDIA_BUFFERING_UPDATE message.
09-13 23:55:24.680: E/MediaPlayer(7949): mOnBufferingUpdateListener is null. Failed to send MEDIA_BUFFERING_UPDATE message.
09-13 23:55:24.680: E/MediaPlayer(7949): mOnVideoSizeChangedListener is null. Failed to send MEDIA_SET_VIDEO_SIZE message.
09-13 23:55:25.675: E/MediaPlayer(7949): mOnBufferingUpdateListener is null. Failed to send MEDIA_BUFFERING_UPDATE message.
09-13 23:55:26.735: E/MediaPlayer(7949): mOnBufferingUpdateListener is null. Failed to send MEDIA_BUFFERING_UPDATE message.
09-13 23:55:27.755: E/MediaPlayer(7949): mOnBufferingUpdateListener is null. Failed to send MEDIA_BUFFERING_UPDATE message.
09-13 23:55:28.705: E/MediaPlayer(7949): mOnBufferingUpdateListener is null. Failed to send MEDIA_BUFFERING_UPDATE message.
尽我所能,我无法让它停止,即使WebView 被清除,然后被销毁。使用<video> 标签加载视频时,我不知道有什么方法可以强制WebChromeClient 使用我为它创建的特定MediaPlayer。似乎决心使用一些隐藏的,它报告了上述内容。有没有办法找到通过WebView 上的<video> 标签创建的MediaPlayer?
--更新
这里是初始化WebView的代码:
mWebView = new WebView(mContext);
mWebView.getSettings().setJavaScriptEnabled(true);
mWebView.getSettings().setPluginState(PluginState.OFF);
mWebView.setVisibility(View.INVISIBLE);
mWebView.getSettings().setDefaultZoom(WebSettings.ZoomDensity.FAR);
mWebView.getSettings().setBuiltInZoomControls(false);
mWebView.getSettings().setCacheMode(WebSettings.LOAD_NO_CACHE);
mWebView.getSettings().setLoadWithOverviewMode(true);
mWebView.getSettings().setUseWideViewPort(true);
mWebView.clearHistory();
mWebView.clearFormData();
mWebView.clearCache(true);
mWebView.getSettings().setAllowFileAccess(true);
mWebView.getSettings().setUserAgentString("Android Mozilla/5.0 AppleWebKit/534.30 (KHTML, like Gecko) Version/4.0 Mobile Safari/534.30");
chrome = new MyWebChromeClient();
mWebView.setWebChromeClient(chrome);
wvc = new MyWebViewClient();
mWebView.setWebViewClient(wvc);
mDomain = "http://foo.bar.com";
mWebView.requestFocus(View.FOCUS_DOWN);
String meat = genMainHTML(R.raw.frame);
mWebView.loadDataWithBaseURL(mDomain, meat, "text/html", "utf-8", null);
“帧”代码是一个 iframe,用于启动一些视频(至少 Vimeo 和 YouTube 似乎都采用这种方法)。我对它进行了一些修剪以避免杂乱无章:
<!DOCTYPE html>
<html>
<head>
</head>
<body>
<div>
<iframe src="-target.url.with.params-" frameborder="0" webkitAllowFullScreen mozallowfullscreen allowFullScreen>
</iframe>
</div>
</body>
</html>
在那个 WebView 类中,存在这个类:
private class MyWebViewClient extends WebViewClient {
@Override
public void onPageFinished(WebView view, String url) {
super.onPageFinished(view, url);
String injection = injectPageMonitor();
if(injection != null && !injectionComplete) {
// Log.i(TAG, " ---------------> Page Loaded . . .");
view.loadUrl(injection);
injectionComplete = true;
}
}
@Override
public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {
Log.d(TAG, "*** Error ["+description+"] ["+failingUrl+"]");
Toast.makeText(mContext, description, Toast.LENGTH_SHORT).show();
}
}
还有这个类:
private class MyWebChromeClient extends WebChromeClient implements MediaPlayer.OnInfoListener, MediaPlayer.OnSeekCompleteListener, MediaPlayer.OnErrorListener, MediaPlayer.OnVideoSizeChangedListener, MediaPlayer.OnCompletionListener, MediaPlayer.OnPreparedListener, MediaPlayer.OnBufferingUpdateListener {
@Override
public void onProgressChanged(WebView view, int progress) {
//Log.e(TAG, " -------------> Progress Changed . . . . ["+progress+"] mWebView ["+mWebView+"] ["+view+"]");
if(progress == 100) {
// Do something really interesting
} else {
updateBuffering(UPDATE_PERCENT_LAUNCH_EXTRACTOR + (progress / 2));
}
}
@Override
public boolean onConsoleMessage(ConsoleMessage cm) {
// Spit out lots of console messages here
return(true);
}
}
我已经覆盖了所有常见的嫌疑人,以防万一我幸运并从 MediaPlayer 中得到一些响应:
@Override
public void onBufferingUpdate(MediaPlayer mp, int percent) {
// Jump up and down because we got a mp!
// never happens, so no jumping
}
作为旁注,我确实使用反射进行了挖掘并通过这种方式获得了 MediaPlayer,但在做那种事情之后我总是觉得有点古怪。即使我有 MediaPlayer 对象,我也不愿意发布它。对 MP 源代码的一点研究表明不会发生任何不好的事情,但是 . . .
这并不是在每台设备上都会发生的,我注意到了。例如,HTC 的表现似乎比三星好(“相同的”操作系统/API 级别——我并没有足够的野心来比较两个源代码树)。
下面提到的行为很烦人,但仅限于我(正在观看 logcat 输出的人。对于用户或应用程序来说,它似乎没有任何效果。只是我真的不喜欢我正在启动 MediaPlayer 的想法例如,将其设置为繁忙的缓存,然后(如果/当用户终止我的应用程序时)将那个丁香莓挂在那里。此时我有很多钱用于治疗,这似乎没有帮助.
提前致谢。
【问题讨论】:
-
似乎是一个类似的问题。请看这个帖子:stackoverflow.com/questions/12156162/…
-
马克 - 非常感谢。正如您所建议的,我已将我的 cmets 移至您提出的问题。在那个帖子中,'Tony' 准确地提出了我喜欢 采用的方法。但是我似乎无法在没有反射的情况下找到
MediaPlayer,这总是让我想去拜访绅士们洗漱。 -
你说“尽我所能,我无法让它停止,即使 WebView 被清除,然后被销毁” - 你怎么知道它被销毁了?取消对 WebView 的引用可能还不够,我认为您可能需要取消设置 WebViewClient 和任何其他引用,然后才能将其销毁/垃圾收集?
-
好点。我会试试的。
标签: java android webview android-mediaplayer webchromeclient