mirror of
https://github.com/owntone/owntone-server.git
synced 2025-01-14 08:15:02 -05:00
[player/dacp/mpd/jsonapi] Combine seek commands into one with mode param
Also changes relative seeking behavior: - seeking behind the the current track only switches to the previous track, if we are not more than 3 seconds into the current track, otherwise starts current track from the beginning - seeking beyond the current track will start the next track from the beginning
This commit is contained in:
parent
d69f328973
commit
6e5d6791ff
@ -549,7 +549,7 @@ seek_timer_cb(int fd, short what, void *arg)
|
|||||||
|
|
||||||
DPRINTF(E_DBG, L_DACP, "Seek timer expired, target %d ms\n", seek_target);
|
DPRINTF(E_DBG, L_DACP, "Seek timer expired, target %d ms\n", seek_target);
|
||||||
|
|
||||||
ret = player_playback_seek(seek_target);
|
ret = player_playback_seek(seek_target, PLAYER_SEEK_POSITION);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
{
|
{
|
||||||
DPRINTF(E_LOG, L_DACP, "Player failed to seek to %d ms\n", seek_target);
|
DPRINTF(E_LOG, L_DACP, "Player failed to seek to %d ms\n", seek_target);
|
||||||
|
@ -1865,7 +1865,7 @@ jsonapi_reply_player_seek(struct httpd_request *hreq)
|
|||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return HTTP_BADREQUEST;
|
return HTTP_BADREQUEST;
|
||||||
|
|
||||||
ret = player_playback_seek(position_ms);
|
ret = player_playback_seek(position_ms, PLAYER_SEEK_POSITION);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -1873,7 +1873,7 @@ jsonapi_reply_player_seek(struct httpd_request *hreq)
|
|||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return HTTP_BADREQUEST;
|
return HTTP_BADREQUEST;
|
||||||
|
|
||||||
ret = player_playback_seek_rel(seek_ms);
|
ret = player_playback_seek(seek_ms, PLAYER_SEEK_RELATIVE);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
|
@ -1521,7 +1521,7 @@ mpd_command_seek(struct evbuffer *evbuf, int argc, char **argv, char **errmsg, s
|
|||||||
seek_target_sec = strtof(argv[2], NULL);
|
seek_target_sec = strtof(argv[2], NULL);
|
||||||
seek_target_msec = seek_target_sec * 1000;
|
seek_target_msec = seek_target_sec * 1000;
|
||||||
|
|
||||||
ret = player_playback_seek(seek_target_msec);
|
ret = player_playback_seek(seek_target_msec, PLAYER_SEEK_POSITION);
|
||||||
|
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
{
|
{
|
||||||
@ -1571,7 +1571,7 @@ mpd_command_seekid(struct evbuffer *evbuf, int argc, char **argv, char **errmsg,
|
|||||||
seek_target_sec = strtof(argv[2], NULL);
|
seek_target_sec = strtof(argv[2], NULL);
|
||||||
seek_target_msec = seek_target_sec * 1000;
|
seek_target_msec = seek_target_sec * 1000;
|
||||||
|
|
||||||
ret = player_playback_seek(seek_target_msec);
|
ret = player_playback_seek(seek_target_msec, PLAYER_SEEK_POSITION);
|
||||||
|
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
{
|
{
|
||||||
@ -1604,7 +1604,7 @@ mpd_command_seekcur(struct evbuffer *evbuf, int argc, char **argv, char **errmsg
|
|||||||
seek_target_msec = seek_target_sec * 1000;
|
seek_target_msec = seek_target_sec * 1000;
|
||||||
|
|
||||||
// TODO If prefixed by '+' or '-', then the time is relative to the current playing position.
|
// TODO If prefixed by '+' or '-', then the time is relative to the current playing position.
|
||||||
ret = player_playback_seek(seek_target_msec);
|
ret = player_playback_seek(seek_target_msec, PLAYER_SEEK_POSITION);
|
||||||
|
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
{
|
{
|
||||||
|
212
src/player.c
212
src/player.c
@ -146,6 +146,12 @@ struct speaker_auth_param
|
|||||||
char pin[5];
|
char pin[5];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct player_seek_param
|
||||||
|
{
|
||||||
|
int ms;
|
||||||
|
enum player_seek_mode mode;
|
||||||
|
};
|
||||||
|
|
||||||
union player_arg
|
union player_arg
|
||||||
{
|
{
|
||||||
struct output_device *device;
|
struct output_device *device;
|
||||||
@ -2167,103 +2173,110 @@ playback_next_bh(void *arg, int *retval)
|
|||||||
return COMMAND_END;
|
return COMMAND_END;
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum command_state
|
static int
|
||||||
playback_seek_bh(void *arg, int *retval)
|
seek_calc_position_ms(struct player_seek_param *seek_param, struct db_queue_item **queue_item, int *position_ms)
|
||||||
{
|
{
|
||||||
struct db_queue_item *queue_item;
|
struct db_queue_item *seek_queue_item = NULL;
|
||||||
union player_arg *cmdarg = arg;
|
int seek_ms = 0;
|
||||||
int ret;
|
|
||||||
|
|
||||||
// outputs_flush() in playback_pause() may have a caused a failure callback
|
// Initialize out parameters
|
||||||
// from the output, which in streaming_cb() can cause pb_abort()
|
*queue_item = NULL;
|
||||||
if (player_state == PLAY_STOPPED)
|
*position_ms = 0;
|
||||||
{
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
queue_item = db_queue_fetch_byitemid(pb_session.playing_now->item_id);
|
// Calculate seek position
|
||||||
if (!queue_item)
|
if (seek_param->mode == PLAYER_SEEK_POSITION)
|
||||||
{
|
seek_ms = seek_param->ms;
|
||||||
DPRINTF(E_DBG, L_PLAYER, "Error seeking in source, queue item has disappeared\n");
|
else
|
||||||
goto error;
|
seek_ms = pb_session.playing_now->pos_ms + seek_param->ms;
|
||||||
}
|
|
||||||
|
|
||||||
ret = pb_session_start(queue_item, cmdarg->intval);
|
|
||||||
free_queue_item(queue_item, 0);
|
|
||||||
if (ret < 0)
|
|
||||||
{
|
|
||||||
DPRINTF(E_DBG, L_PLAYER, "Error seeking to %d, aborting playback\n", cmdarg->intval);
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Silent status change - playback_start() sends the real status update
|
|
||||||
player_state = PLAY_PAUSED;
|
|
||||||
|
|
||||||
*retval = 0;
|
|
||||||
return COMMAND_END;
|
|
||||||
|
|
||||||
error:
|
|
||||||
pb_abort();
|
|
||||||
*retval = -1;
|
|
||||||
return COMMAND_END;
|
|
||||||
}
|
|
||||||
|
|
||||||
static enum command_state
|
|
||||||
playback_seek_rel_bh(void *arg, int *retval)
|
|
||||||
{
|
|
||||||
struct db_queue_item *queue_item;
|
|
||||||
union player_arg *cmdarg = arg;
|
|
||||||
int seek_ms;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
|
|
||||||
// outputs_flush() in playback_pause() may have a caused a failure callback
|
|
||||||
// from the output, which in streaming_cb() can cause pb_abort()
|
|
||||||
if (player_state == PLAY_STOPPED)
|
|
||||||
{
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
seek_ms = pb_session.playing_now->pos_ms + cmdarg->intval;
|
|
||||||
|
|
||||||
|
// Check if we need to switch to a previous track, this will be done if we are in the first 3 seconds
|
||||||
|
// of a track and we have a seek request for more than 3 seconds
|
||||||
if (seek_ms < 0)
|
if (seek_ms < 0)
|
||||||
{
|
{
|
||||||
// Seeking behind the start of the current queue item
|
if (pb_session.playing_now->pos_ms < 3000)
|
||||||
queue_item = queue_item_prev(pb_session.playing_now->item_id);
|
|
||||||
if (queue_item)
|
|
||||||
{
|
{
|
||||||
seek_ms = queue_item->song_length + seek_ms;
|
// We are in the first 3 seconds of the track, switch to the previous track and recalculate the absolute seek position
|
||||||
|
seek_queue_item = queue_item_prev(pb_session.playing_now->item_id);
|
||||||
|
|
||||||
|
if (seek_queue_item)
|
||||||
|
{
|
||||||
|
seek_ms = seek_queue_item->song_length + seek_ms;
|
||||||
|
// Make sure to not try to seek behind the previous track (this is also the case if song_length is zero)
|
||||||
|
seek_ms = (seek_ms < 0) ? 0 : seek_ms;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// There is no previous queue item, seek to the start of the current item
|
||||||
|
seek_ms = 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// There is no previous queue item, seek to the start of the current item
|
// We are more than 3 seconds into the playing track, seek to beginning of current track
|
||||||
queue_item = db_queue_fetch_byitemid(pb_session.playing_now->item_id);
|
|
||||||
seek_ms = 0;
|
seek_ms = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (seek_ms > pb_session.playing_now->len_ms)
|
else if (seek_ms > 0 && seek_ms > pb_session.playing_now->len_ms)
|
||||||
{
|
{
|
||||||
// Seeking beyond the end of the current queue item
|
// We are seeking beyond the current track, play the next track from the beginning
|
||||||
queue_item = queue_item_next(pb_session.playing_now->item_id);
|
seek_queue_item = queue_item_next(pb_session.playing_now->item_id);
|
||||||
seek_ms = seek_ms - pb_session.playing_now->len_ms;
|
if (seek_queue_item)
|
||||||
}
|
{
|
||||||
else
|
seek_ms = 0;
|
||||||
{
|
}
|
||||||
// Seeking in the current queue item
|
|
||||||
queue_item = db_queue_fetch_byitemid(pb_session.playing_now->item_id);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!queue_item)
|
if (!seek_queue_item)
|
||||||
|
{
|
||||||
|
// Seeking in the current queue item
|
||||||
|
seek_queue_item = db_queue_fetch_byitemid(pb_session.playing_now->item_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!seek_queue_item)
|
||||||
|
{
|
||||||
|
DPRINTF(E_LOG, L_PLAYER, "Error fetching queue item for seek command (seek_ms=%d, seek_mode=%d)\n", seek_param->ms, seek_param->mode);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (seek_ms < 0)
|
||||||
|
{
|
||||||
|
DPRINTF(E_LOG, L_PLAYER, "Error calculating new seek position for seek command (seek_ms=%d, seek_mode=%d)\n", seek_param->ms, seek_param->mode);
|
||||||
|
free(seek_queue_item);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
*queue_item = seek_queue_item;
|
||||||
|
*position_ms = seek_ms;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static enum command_state
|
||||||
|
playback_seek_bh(void *arg, int *retval)
|
||||||
|
{
|
||||||
|
struct player_seek_param *seek_param = arg;
|
||||||
|
struct db_queue_item *queue_item;
|
||||||
|
int position_ms;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
// outputs_flush() in playback_pause() may have a caused a failure callback
|
||||||
|
// from the output, which in streaming_cb() can cause pb_abort()
|
||||||
|
if (player_state == PLAY_STOPPED)
|
||||||
{
|
{
|
||||||
DPRINTF(E_DBG, L_PLAYER, "Error seeking in source, queue item has disappeared\n");
|
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = pb_session_start(queue_item, seek_ms);
|
ret = seek_calc_position_ms(seek_param, &queue_item, &position_ms);
|
||||||
|
if (ret < 0)
|
||||||
|
{
|
||||||
|
DPRINTF(E_LOG, L_PLAYER, "Error calculating new seek position\n");
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = pb_session_start(queue_item, position_ms);
|
||||||
free_queue_item(queue_item, 0);
|
free_queue_item(queue_item, 0);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
{
|
{
|
||||||
DPRINTF(E_DBG, L_PLAYER, "Error seeking to %d, aborting playback\n", cmdarg->intval);
|
DPRINTF(E_LOG, L_PLAYER, "Error seeking to %d, aborting playback\n", position_ms);
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2348,6 +2361,21 @@ playback_pause(void *arg, int *retval)
|
|||||||
return COMMAND_END;
|
return COMMAND_END;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static enum command_state
|
||||||
|
playback_seek(void *arg, int *retval)
|
||||||
|
{
|
||||||
|
// Only check if the current playing track is seekable, other checks will be done in playback_pause()
|
||||||
|
if (pb_session.playing_now && pb_session.playing_now->len_ms <= 0)
|
||||||
|
{
|
||||||
|
DPRINTF(E_WARN, L_PLAYER, "Failed to seek, track is not seekable\n");
|
||||||
|
|
||||||
|
*retval = -1;
|
||||||
|
return COMMAND_END;
|
||||||
|
}
|
||||||
|
|
||||||
|
return playback_pause(arg, retval);
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
device_to_speaker_info(struct player_speaker_info *spk, struct output_device *device)
|
device_to_speaker_info(struct player_speaker_info *spk, struct output_device *device)
|
||||||
{
|
{
|
||||||
@ -2986,27 +3014,31 @@ player_playback_pause(void)
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Seeks to the position "seek_ms", depending on the given "seek_mode" seek_ms is
|
||||||
|
* either the new position in the current track (seek_mode == PLAYER_SEEK_POSITION)
|
||||||
|
* or a relative amount of milliseconds from the current playing position
|
||||||
|
* (seek_mode == PLAYER_SEEK_RELATIVE).
|
||||||
|
*
|
||||||
|
* Relative seeking switches tracks, if:
|
||||||
|
* - seeking behind the the current track and current playing position is not more than 3 seconds
|
||||||
|
* - seeking beyond the current track
|
||||||
|
*
|
||||||
|
* @param seek_ms Position or relative amount of milliseconds to seek to
|
||||||
|
* @param seek_mode If PLAYER_SEEK_POSITION seek_ms is a position in milliseconds,
|
||||||
|
* if PLAYER_SEEK_RELATIVE seek_ms is the relative amount of milliseconds
|
||||||
|
* @return Returns 0 on success and a negative value on error
|
||||||
|
*/
|
||||||
int
|
int
|
||||||
player_playback_seek(int ms)
|
player_playback_seek(int seek_ms, enum player_seek_mode seek_mode)
|
||||||
{
|
{
|
||||||
union player_arg cmdarg;
|
struct player_seek_param seek_param;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
cmdarg.intval = ms;
|
seek_param.ms = seek_ms;
|
||||||
|
seek_param.mode = seek_mode;
|
||||||
|
|
||||||
ret = commands_exec_sync(cmdbase, playback_pause, playback_seek_bh, &cmdarg);
|
ret = commands_exec_sync(cmdbase, playback_seek, playback_seek_bh, &seek_param);
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
int
|
|
||||||
player_playback_seek_rel(int ms)
|
|
||||||
{
|
|
||||||
union player_arg cmdarg;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
cmdarg.intval = ms;
|
|
||||||
|
|
||||||
ret = commands_exec_sync(cmdbase, playback_pause, playback_seek_rel_bh, &cmdarg);
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
10
src/player.h
10
src/player.h
@ -22,6 +22,11 @@ enum repeat_mode {
|
|||||||
REPEAT_ALL = 2,
|
REPEAT_ALL = 2,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum player_seek_mode {
|
||||||
|
PLAYER_SEEK_POSITION = 1,
|
||||||
|
PLAYER_SEEK_RELATIVE = 2,
|
||||||
|
};
|
||||||
|
|
||||||
struct player_speaker_info {
|
struct player_speaker_info {
|
||||||
uint64_t id;
|
uint64_t id;
|
||||||
char name[255];
|
char name[255];
|
||||||
@ -109,10 +114,7 @@ int
|
|||||||
player_playback_pause(void);
|
player_playback_pause(void);
|
||||||
|
|
||||||
int
|
int
|
||||||
player_playback_seek(int ms);
|
player_playback_seek(int seek_ms, enum player_seek_mode seek_mode);
|
||||||
|
|
||||||
int
|
|
||||||
player_playback_seek_rel(int ms);
|
|
||||||
|
|
||||||
int
|
int
|
||||||
player_playback_next(void);
|
player_playback_next(void);
|
||||||
|
Loading…
Reference in New Issue
Block a user