在JavaFX中将WebView静音,这可能吗?
非常简单。是否可以将 JavaFX WebView 的音量静音或控制?我在谷歌上搜索了一段时间,但我找不到任何提及。我查看了 和 的代码,似乎没有任何关于控制音量的内容。WebView
WebEngine
我仍然需要同一应用程序中的其他 s 来工作和产生声音,因此,我无法将整个应用程序静音。MediaPlayer
非常简单。是否可以将 JavaFX WebView 的音量静音或控制?我在谷歌上搜索了一段时间,但我找不到任何提及。我查看了 和 的代码,似乎没有任何关于控制音量的内容。WebView
WebEngine
我仍然需要同一应用程序中的其他 s 来工作和产生声音,因此,我无法将整个应用程序静音。MediaPlayer
目前,在WebView或WebEngine Control中没有用于声音系统的API。
但是你可以注入JavaScript来静音/取消静音和控制声音。此方法仅适用于 html5 元素,<视频>或<音频>,因为 Flash 或 JS 初始化音频等项目通常无法限制。我为您创建了示例程序:
import javafx.application.Application;
import javafx.event.ActionEvent;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.layout.VBox;
import javafx.scene.web.WebEngine;
import javafx.scene.web.WebView;
import javafx.stage.Stage;
public class Main extends Application {
@Override
public void start(Stage primaryStage) throws Exception{
WebView myWebView = new WebView();
WebEngine engine = myWebView.getEngine();
//dirty code xD
Button btn = new Button("Load Youtube");
btn.setOnAction((ActionEvent event) -> engine.load("http://www.youtube.com/embed/A6hrw6KGdtM?autoplay=1"));
Button btn2 = new Button("Mute");
btn2.setOnAction((ActionEvent event) -> engine.executeScript(getJSAudioVideo(true)));
Button btn3 = new Button("Unmute");
btn3.setOnAction((ActionEvent event) -> engine.executeScript(getJSAudioVideo(false)));
Button btn4 = new Button("+");
btn4.setOnAction((ActionEvent event) -> engine.executeScript(getJSSoundVolume(0.9)));
Button btn5 = new Button("-");
btn5.setOnAction((ActionEvent event) -> engine.executeScript(getJSSoundVolume(0.1)));
VBox root = new VBox();
root.getChildren().addAll(myWebView, btn, btn2, btn3, btn4, btn5);
Scene scene = new Scene(root, 800, 500);
primaryStage.setScene(scene);
primaryStage.setTitle("Mute a WebView in JavaFX, is it possible?");
primaryStage.show();
}
/**
* @param mute
* @return JS code for mute/unmute
*/
public String getJSAudioVideo(boolean mute){
return "var videos = document.querySelectorAll('video'),\n" +
" audios = document.querySelectorAll('listen');\n" +
" [].forEach.call(videos, function(video) { video.muted = "+String.valueOf(mute)+"; });\n" +
" [].forEach.call(audios, function(audio) { audio.muted = "+String.valueOf(mute)+"; });";
}
/**
* @param volume
* @return JS code for setting volume sound
*/
public String getJSSoundVolume(double volume){
//max 1.0, min 0.0 Default: 1.0
double _volume = (volume > 1.0 || volume < 0.0) ? 1.0 : volume;
return "var videos = document.querySelectorAll('video'),\n" +
" audios = document.querySelectorAll('listen');\n" +
" [].forEach.call(videos, function(video) { video.volume = "+String.valueOf(_volume)+"; });\n" +
" [].forEach.call(audios, function(audio) { audio.volume = "+String.valueOf(_volume)+"; });";
}
public static void main(String[] args) {
launch(args);
}
}
我们都知道它是API中缺失的一部分,那么您能做些什么来自己实现它呢?让我们想想WebView到底是什么:
嵌入式浏览器组件基于WebKit [...]默认情况下,WebKit 不支持网页呈现。为了呈现和显示HTML内容,Oracle必须使用Java Graphics 2D API编写自己的渲染器。
这意味着引擎必须具有实现。WebKit 的实现位于包中(相当一部分类只是本机调用的包装器)。当然,在大多数情况下,你不想使用类,但目前你正在使用JavaFX,所以这并不重要。com.sun.webkit
com.sun.*
如果我们在太阳的来源中跳一点,我们可以找到一些抽象的音频方法,例如:WCMediaPlayer.class
protected abstract void setRate(float rate);
protected abstract void setVolume(float volume);
protected abstract void setMute(boolean mute);
protected abstract void setSize(int w, int h);
protected abstract void setPreservesPitch(boolean preserve);
C'mon Java,让我叫你:
volumeMethod = WCMediaPlayer.class.getDeclaredMethod("setVolume", float.class);
volumeMethod.setAccessible(true);
感谢您的帮助,但是如何获取实例?我们必须查看对 的引用。咔嚓! 通过方法创建MediaPlayer,毕竟,它将指针和实例放在:WCMediaPlayer
new WCMediaPlayerImpl()
WCGraphicsManager.class
fwkCreateMediaPlayer()
refMap
Field refMapField = WCGraphicsManager.class.getDeclaredField("refMap");
refMapField.setAccessible(true);
幸运的是,管理器公开了获取实例的方法:getGraphicsManager()
WCGraphicsManager graphicsManager = WCGraphicsManager.getGraphicsManager();
refMap = (Map<Integer, Ref>) refMapField.get(graphicsManager);
捕获的映射包含实例(还有其他实例),因此您必须过滤它们:Ref
WC*
Collection<WCMediaPlayer> mediaPlayers = refMap.values().stream()
.filter(ref -> ref instanceof WCMediaPlayer)
.map(ref -> (WCMediaPlayer) ref)
.collect(Collectors.toList());
您可能期望工作示例,因此这是我使用的代码:
public class WebEngineTest extends Application {
private Map<Integer, Ref> refMap;
private Method volumeMethod;
@Override
@SuppressWarnings("unchecked")
public void start(Stage primaryStage) throws Exception {
WebView webView = new WebView();
WebEngine engine = webView.getEngine();
engine.load("https://www.youtube.com/watch?v=hRAZBSoAsgs");
Field refMapField = WCGraphicsManager.class.getDeclaredField("refMap");
refMapField.setAccessible(true);
volumeMethod = WCMediaPlayer.class.getDeclaredMethod("setVolume", float.class);
volumeMethod.setAccessible(true);
WCGraphicsManager graphicsManager = WCGraphicsManager.getGraphicsManager();
refMap = (Map<Integer, Ref>) refMapField.get(graphicsManager);
Button button = new Button("Volume");
button.setOnAction(event -> setVolume(0.1f));
Group group = new Group();
group.getChildren().addAll(webView, button);
Scene scene = new Scene(group, 625, 625);
primaryStage.setScene(scene);
primaryStage.show();
}
public void setVolume(float volume) {
Collection<WCMediaPlayer> mediaPlayers = this.getMediaPlayers();
mediaPlayers.forEach(mediaPlayer -> setVolumeMethod(mediaPlayer, volume));
}
private void setVolumeMethod(Object instance, Object... args) {
try {
volumeMethod.invoke(instance, args);
} catch (Exception e) {
e.printStackTrace();
}
}
private Collection<WCMediaPlayer> getMediaPlayers() {
return refMap.values().stream()
.filter(ref -> ref instanceof WCMediaPlayer)
.map(ref -> (WCMediaPlayer) ref)
.collect(Collectors.toList());
}
}
最后,请记住这些调用的顺序。例如,在引擎状态为 not 之前不包含全部,或者如果根本没有创建图形元素,则返回。refMap
Refs
SUCCEEDED
WCGraphicsManager.getGraphicsManager()
null
也许,最好的方法是将这些解决方案结合起来。如果没有 JavaFX 提供的场景和糟糕的 API,就很难支持 Web 技术组合。您也可以尝试嵌入其他浏览器,如Chromium。