/* * Copyright 2008-2013, ETH Zürich, Samuel Welten, Michael Kuhn, Tobias Langner, * Sandro Affentranger, Lukas Bossard, Michael Grob, Rahul Jain, * Dominic Langenegger, Sonia Mayor Alonso, Roger Odermatt, Tobias Schlueter, * Yannick Stucki, Sebastian Wendland, Samuel Zehnder, Samuel Zihlmann, * Samuel Zweifel * * This file is part of Jukefox. * * Jukefox is free software: you can redistribute it and/or modify it under the * terms of the GNU General Public License as published by the Free Software * Foundation, either version 3 of the License, or any later version. Jukefox is * distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A * PARTICULAR PURPOSE. See the GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along with * Jukefox. If not, see <http://www.gnu.org/licenses/>. */ package ch.ethz.dcg.jukefox.playmode; import ch.ethz.dcg.jukefox.commons.utils.Log; import ch.ethz.dcg.jukefox.commons.utils.RandomProvider; import ch.ethz.dcg.jukefox.model.AbstractCollectionModelManager; import ch.ethz.dcg.jukefox.model.AbstractPlayerModelManager; import ch.ethz.dcg.jukefox.model.collection.BaseAlbum; import ch.ethz.dcg.jukefox.model.collection.BaseArtist; import ch.ethz.dcg.jukefox.model.collection.IReadOnlyPlaylist; import ch.ethz.dcg.jukefox.model.collection.PlaylistSong; import ch.ethz.dcg.jukefox.model.commons.NoNextSongException; import ch.ethz.dcg.jukefox.model.commons.PlaylistPositionOutOfRangeException; import ch.ethz.dcg.jukefox.model.player.PlayModeType; import ch.ethz.dcg.jukefox.model.player.PlayerAction; /** * Repeats the playlist by going to the first song once the last one finished. If you're at the first song and try to go * back, you'll end up at the last song. */ public class ShufflePlaylistPlayMode extends BasePlayMode { private static final String TAG = ShufflePlaylistPlayMode.class.getSimpleName(); protected int songAvoidanceNumber; public ShufflePlaylistPlayMode(AbstractCollectionModelManager collectionModel, AbstractPlayerModelManager playerModel, int songAvoidanceNumber) { super(collectionModel, playerModel); this.songAvoidanceNumber = songAvoidanceNumber; } private int getNextRandomSongPos(IReadOnlyPlaylist playlist) throws NoNextSongException, PlaylistPositionOutOfRangeException { PlaylistSong<BaseArtist, BaseAlbum> currentSong = null; try { currentSong = playlist.getSongAtPosition(playlist.getPositionInList()); } catch (PlaylistPositionOutOfRangeException e) { Log.w(TAG, e); } int randomPos = RandomProvider.getRandom().nextInt(playlist.getSize()); PlaylistSong<BaseArtist, BaseAlbum> tempSong = playlist.getSongAtPosition(randomPos); int avoidance = Math.min(songAvoidanceNumber, playlist.getSize() - 2); if (avoidance < 0) { avoidance = 0; } while (playerModel.getPlayLog().isSongInRecentHistory(tempSong, avoidance) || isEqualToCurrentSong(currentSong, tempSong)) { randomPos = RandomProvider.getRandom().nextInt(playlist.getSize()); tempSong = playlist.getSongAtPosition(randomPos); } return randomPos; } private boolean isEqualToCurrentSong(PlaylistSong<BaseArtist, BaseAlbum> currentSong, PlaylistSong<BaseArtist, BaseAlbum> tempSong) { if (currentSong == null || tempSong == null) { return false; } return currentSong.getId() == tempSong.getId(); } @Override public PlayerControllerCommands next(IReadOnlyPlaylist currentPlaylist) throws NoNextSongException { PlayerControllerCommands commands = new PlayerControllerCommands(); if (currentPlaylist.isPlaylistEmpty()) { return commands; } try { commands.setListPos(getNextRandomSongPos(currentPlaylist)); commands.playerAction(PlayerAction.PLAY); } catch (PlaylistPositionOutOfRangeException e) { Log.w(TAG, e); } return commands; } @Override public PlayModeType getPlayModeType() { return PlayModeType.SHUFFLE_PLAYLIST; } }