如何在没有监听器的情况下获取javafx媒体元数据

时间:2013-01-12 18:29:20

标签: listener media javafx

所以我一直在寻找这个一周,并且阅读虽然每个问题都很相似,但似乎没有一个问题与我的问题完全相同(尝试逆向工程其他解决方案类似于我想要的但没有成功。

解释了穴居人风格:我正在尝试使用元数据创建列表。

  1. 我打开多个对话框并选择多个mp3
  2. 我把文件放在ArrayList<File>
  3. 我循环使用增强的for循环文件并使用媒体变量提取元数据
  4. 元数据的信息(如“艺术家”)是我想要保存在ArrayList中的例子
  5. 问题是,只有在增强循环结束后,侦听器才能正常工作 ArrayList<String>有一个对象没有任何内容

    这是一个示例:

    ArrayList<String> al;
    String path;
    public void open(){
        files=chooser.showOpenMultipleDialog(new Stage());
        for( File f:files){             
            path=f.getPath();
            Media media = new Media("file:/"+path.replace("\\", "/").replace(" ", "%20"));
            al= new ArrayList<String>();
            media.getMetadata().addListener(new MapChangeListener<String, Object>() {                 
                public void onChanged(Change<? extends String, ? extends Object> change) {
                    if (change.wasAdded()) {
                        if (change.getKey().equals("artist")) {
                            al.add((String) change.getValueAdded());
                         }
                    }
                }
            });
        }//close for loop
        //then i want to see the size of al like this
        system.out.println(al.size());
        //then it returns 1 no matter how much file i selected
        //when i system out "al" i get an empty string
    

3 个答案:

答案 0 :(得分:3)

通过添加侦听器来读取媒体源元数据的另一种方法是在媒体播放器中提取该信息.setOnReady();这是java控制器类的一个示例部分

public class uiController实现Initializable {

@FXML private Label label;
@FXML private ListView<String> lv;
@FXML private AnchorPane root;
@FXML private Button button;

private ObservableList<String> ol= FXCollections.observableArrayList();
private List<File> selectedFiles;
private final Object obj= new Object();

@Override
public void initialize(URL url, ResourceBundle rb) {
    assert button != null : "fx:id=\"button\" was not injected: check your FXML file 'ui.fxml'.";
    assert label != null : "fx:id=\"label\" was not injected: check your FXML file 'ui.fxml'.";
    assert lv != null : "fx:id=\"lv\" was not injected: check your FXML file 'ui.fxml'.";
    assert root != null : "fx:id=\"root\" was not injected: check your FXML file 'ui.fxml'.";

    // initialize your logic here: all @FXML variables will have been injected
    lv.setItems(ol);
}  

@FXML private void open(ActionEvent event) {
    FileChooser.ExtensionFilter extention= new FileChooser.ExtensionFilter("Music Files", "*.mp3","*.m4a","*.aif","*.wav","*.m3u","*.m3u8");
    FileChooser fc= new FileChooser();
    fc.setInitialDirectory(new File(System.getenv("userprofile")));
    fc.setTitle("Select File(s)");
    fc.getExtensionFilters().add(extention);
    selectedFiles =fc.showOpenMultipleDialog(root.getScene().getWindow());
    if(selectedFiles != null &&!selectedFiles.isEmpty()){
        listFiles();
    }
}
/**
 * Convert each fie selected to its URI
 */
private void listFiles(){ 
    try {
        for (File file : selectedFiles) {
            readMetaData(file.toURI().toString());
            synchronized(obj){
               obj.wait(100);
            }
        }
    } catch (InterruptedException ex) {
    }
    System.gc();
}
/**
 * Read a Media source metadata 
 * Note: Sometimes the was unable to extract the metadata especially when 
 * i have selected large number of files reasons i don't known why
 * @param mediaURI Media file URI
 */
private void readMetaData(String mediaURI){
    final MediaPlayer mp= new MediaPlayer(new Media(mediaURI));
    mp.setOnReady(new Runnable() {

        @Override
        public void run() {
            String artistName=(String) mp.getMedia().getMetadata().get("artist");
            ol.add(artistName);
            synchronized(obj){//this is required since mp.setOnReady creates a new thread and our loopp  in the main thread
                obj.notify();// the loop has to wait unitl we are able to get the media metadata thats why use .wait() and .notify() to synce the two threads(main thread and MediaPlayer thread)
            }
        }
    });
}

}

所做的一些更改是使用ObservableList来存储元数据中的艺术家名称

在代码中你会发现这个                

 synchronized(obj){
                    obj.wait(100);
                } 
我这样做是因为mediaplayer .setOnReady()创建了一个新线程并且循环在主应用程序线程中。循环必须等待一段时间才能创建另一个线程并且我们能够提取元数据,并且.setOnReady()有一个               
 synchronized(obj){
                    obj.notify;
                } 
唤醒主线程因此循环能够移动到下一个项目

我承认这可能不是最好的解决方案,但欢迎任何有更好方法从文件列表中读取JavaFx媒体元数据的人 可以在https://docs.google.com/file/d/0BxDEmOcXqnCLSTFHbTVFcGIzT1E/edit?usp=sharing

找到完整的Netbeans项目

plus使用JavaFX创建了一个小型MediaPlayer应用程序,该应用程序使用了元数据https://docs.google.com/file/d/0BxDEmOcXqnCLR1Z0VGN4ZlJkbUU/edit?usp=sharing

答案 1 :(得分:0)

您可以使用以下函数检索给定Media对象的元数据:

public static void initializeMetaData(Media media) {
    final Ref<Boolean> ready = new Ref<>(false);

    MediaPlayer mediaPlayer = new MediaPlayer(media);
    mediaPlayer.setOnReady(() -> {
        synchronized (ready) {
            ready.set(false);
            ready.notify();
        }
    });

    synchronized (ready) {
        if (!ready.get()) {
            try {
                ready.wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

但是,不要从JavaFX线程调用initializeMetaData,否则线程会陷入死锁。

PS:人们必须建立这样的解决方案真的很荒谬。我希望将来Media会提供一个初始化()方法来完成这项工作。

答案 2 :(得分:0)

我对这个问题的解决方案是:

public class MediaListener implements MapChangeListener<String, Object>
{
    public String title = null;
    public String artist = null;
    public String album = null;

    private final Consumer<MediaListener> handler;
    private boolean handled = false;

    public MediaListener(Consumer<MediaListener> handler)
    {
        this.handler = handler;
    }

    @Override
    public void onChanged(MapChangeListener.Change<? extends String, ?> ch)
    {
        if (ch.wasAdded())
        {
            String key = ch.getKey();
            switch (key)
            {
                case "title":
                    title = (String) ch.getValueAdded();
                    break;
                case "artist":
                    artist = (String) ch.getValueAdded();
                    break;
                case "album":
                    album = (String) ch.getValueAdded();
                    break;
            }

            if (!handled && title != null && artist != null && album != null)
            {
                handler.accept(this);
                handled = true;
            }
        }
    }
}

这可能不是最好的方式,但它比每个文件创建一个新的MediaPlayer更清晰。

使用示例:

Media media = Util.createMedia(path);
media.getMetadata().addListener(new MediaListener((data) ->
{
    // Use the data object to access the media
}));