2014-02-10 01:10:30 +00:00
|
|
|
/*************************************************************************/
|
|
|
|
/* cp_player_data_control.cpp */
|
|
|
|
/*************************************************************************/
|
|
|
|
/* This file is part of: */
|
|
|
|
/* GODOT ENGINE */
|
2017-08-27 12:11:45 +00:00
|
|
|
/* https://godotengine.org */
|
2014-02-10 01:10:30 +00:00
|
|
|
/*************************************************************************/
|
2019-06-04 09:41:49 +00:00
|
|
|
/* Copyright (c) 2007-2019 Juan Linietsky, Ariel Manzur. */
|
|
|
|
/* Copyright (c) 2014-2019 Godot Engine contributors (cf. AUTHORS.md) */
|
2014-02-10 01:10:30 +00:00
|
|
|
/* */
|
|
|
|
/* Permission is hereby granted, free of charge, to any person obtaining */
|
|
|
|
/* a copy of this software and associated documentation files (the */
|
|
|
|
/* "Software"), to deal in the Software without restriction, including */
|
|
|
|
/* without limitation the rights to use, copy, modify, merge, publish, */
|
|
|
|
/* distribute, sublicense, and/or sell copies of the Software, and to */
|
|
|
|
/* permit persons to whom the Software is furnished to do so, subject to */
|
|
|
|
/* the following conditions: */
|
|
|
|
/* */
|
|
|
|
/* The above copyright notice and this permission notice shall be */
|
|
|
|
/* included in all copies or substantial portions of the Software. */
|
|
|
|
/* */
|
|
|
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
|
|
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
|
|
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
|
|
|
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
|
|
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
|
|
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
|
|
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
|
|
|
/*************************************************************************/
|
|
|
|
|
|
|
|
#include "cp_player_data.h"
|
|
|
|
|
|
|
|
void CPPlayer::play_start_pattern(int p_pattern) {
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
play_start(p_pattern, -1, -1);
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CPPlayer::play_start_song() {
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
play_start(-1, -1, -1);
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CPPlayer::play_start_song_from_order(int p_order) {
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
play_start(-1, p_order, -1);
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
void CPPlayer::play_start_song_from_order_and_row(int p_order, int p_row) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
play_start(-1, p_order, p_row);
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
void CPPlayer::play_start(int p_pattern, int p_order, int p_row, bool p_lock) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if (control.play_mode != PLAY_NOTHING) play_stop();
|
2014-02-10 01:10:30 +00:00
|
|
|
|
|
|
|
reset();
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if (p_pattern != -1) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
control.play_mode = PLAY_PATTERN;
|
|
|
|
control.position.current_pattern = p_pattern;
|
|
|
|
control.position.current_row = (p_row != -1) ? p_row : 0;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
|
|
|
} else {
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
control.position.current_order = get_song_next_order_idx(song, (p_order == -1) ? p_order : p_order - 1);
|
|
|
|
if (control.position.current_order != -1) {
|
|
|
|
|
|
|
|
control.play_mode = PLAY_SONG;
|
|
|
|
control.position.current_pattern = song->get_order(control.position.current_order);
|
|
|
|
control.position.current_row = (p_row != -1) ? p_row : 0;
|
|
|
|
}
|
|
|
|
}
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
control.reached_end = (control.play_mode == PLAY_NOTHING);
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CPPlayer::play_stop() {
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
control.play_mode = PLAY_NOTHING;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
for (i = 0; i < control.max_voices; i++) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
|
|
|
voice[i].reset();
|
|
|
|
mixer->stop_voice(i);
|
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
for (i = 0; i < CPPattern::WIDTH; i++) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
|
|
|
control.channel[i].reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
reset();
|
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
void CPPlayer::play_note(int p_channel, CPNote note, bool p_reserve) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if (control.play_mode == PLAY_NOTHING) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
control.ticks_counter = 0;
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*control.channel[p_channel].reset();
|
|
|
|
control.channel[p_channel].channel_volume=song->get_channel_volume(p_channel);
|
|
|
|
control.channel[p_channel].channel_panning=((int)song->get_channel_pan( p_channel)*255/64);*/
|
|
|
|
if (p_reserve) {
|
2017-03-18 23:36:26 +00:00
|
|
|
control.channel[p_channel].mute = false;
|
|
|
|
control.channel[p_channel].reserved = true;
|
2014-02-10 01:10:30 +00:00
|
|
|
} else {
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
control.channel[p_channel].reserved = false;
|
|
|
|
}
|
|
|
|
process_note(p_channel, note);
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CPPlayer::get_voice_volume(int p_voice) {
|
2017-03-18 23:36:26 +00:00
|
|
|
|
2014-02-10 01:10:30 +00:00
|
|
|
return voice[p_voice].display_volume;
|
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
int CPPlayer::get_voice_envelope_pos(int p_voice, CPEnvelope *p_envelope) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
int i, tmp_index = -1;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
i = p_voice;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if ((song->has_instruments()) && (voice[i].instrument_ptr != NULL) && (voice[i].fadeout_volume > 0)) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if ((p_envelope == voice[i].instrument_ptr->get_volume_envelope()) && (voice[i].instrument_ptr->get_volume_envelope()->is_enabled())) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
tmp_index = voice[i].volume_envelope_ctrl.pos_index;
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if ((p_envelope == voice[i].instrument_ptr->get_pan_envelope()) && (voice[i].instrument_ptr->get_pan_envelope()->is_enabled())) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
tmp_index = voice[i].panning_envelope_ctrl.pos_index;
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if ((p_envelope == voice[i].instrument_ptr->get_pitch_filter_envelope()) && (voice[i].instrument_ptr->get_pitch_filter_envelope()->is_enabled())) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
tmp_index = voice[i].pitch_envelope_ctrl.pos_index;
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return tmp_index;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CPPlayer::goto_next_order() {
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if (control.play_mode != PLAY_SONG) return;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
control.position.current_row = 0;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
control.position.current_order = get_song_next_order_idx(song, control.position.current_order);
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if (control.position.current_order == -1) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
reset();
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
control.position.current_pattern = song->get_order(control.position.current_order);
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
void CPPlayer::goto_previous_order() {
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if (control.play_mode != PLAY_SONG) return;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
int next_order, current_order;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
control.position.current_row = 0;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
current_order = control.position.current_order;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
next_order = get_song_next_order_idx(song, current_order);
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
while ((next_order != control.position.current_order) && (next_order != -1)) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
current_order = next_order;
|
|
|
|
next_order = get_song_next_order_idx(song, current_order);
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if (next_order == -1) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
reset();
|
2014-02-10 01:10:30 +00:00
|
|
|
} else {
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
control.position.current_order = current_order;
|
|
|
|
control.position.current_pattern = song->get_order(control.position.current_order);
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int CPPlayer::get_channel_voice(int p_channel) {
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if (control.channel[p_channel].slave_voice == NULL)
|
|
|
|
return -1;
|
|
|
|
else
|
|
|
|
return control.channel[p_channel].slave_voice_index;
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
const char *CPPlayer::get_voice_sample_name(int p_voice) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2015-05-03 00:16:26 +00:00
|
|
|
const char *name = NULL;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if (!voice[p_voice].sample_ptr) name = voice[p_voice].sample_ptr->get_name();
|
2014-02-10 01:10:30 +00:00
|
|
|
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CPPlayer::is_voice_active(int p_voice) {
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
return !(((voice[p_voice].kick == KICK_NOTHING) || (voice[p_voice].kick == KICK_ENVELOPE)) && !mixer->is_voice_active(p_voice));
|
|
|
|
}
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
int CPPlayer::get_voice_envelope_pos(int p_voice, CPInstrument::EnvelopeType p_env_type) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
|
|
|
if (!is_voice_active(p_voice))
|
|
|
|
return -1;
|
2017-03-18 23:36:26 +00:00
|
|
|
|
|
|
|
Voice_Control::Envelope_Control *env = 0;
|
|
|
|
|
2014-02-10 01:10:30 +00:00
|
|
|
switch (p_env_type) {
|
2017-03-18 23:36:26 +00:00
|
|
|
|
|
|
|
case CPInstrument::VOLUME_ENVELOPE: env = &voice[p_voice].volume_envelope_ctrl; break;
|
|
|
|
case CPInstrument::PAN_ENVELOPE: env = &voice[p_voice].panning_envelope_ctrl; break;
|
|
|
|
case CPInstrument::PITCH_ENVELOPE: env = &voice[p_voice].pitch_envelope_ctrl; break;
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
2017-03-18 23:36:26 +00:00
|
|
|
|
2014-02-10 01:10:30 +00:00
|
|
|
if (!env)
|
|
|
|
return -1;
|
2017-03-18 23:36:26 +00:00
|
|
|
|
2014-02-10 01:10:30 +00:00
|
|
|
if (!env->active || env->terminated)
|
|
|
|
return -1;
|
2017-03-18 23:36:26 +00:00
|
|
|
|
2014-02-10 01:10:30 +00:00
|
|
|
return env->pos_index;
|
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
CPEnvelope *CPPlayer::get_voice_envelope(int p_voice, CPInstrument::EnvelopeType p_env_type) {
|
|
|
|
|
|
|
|
CPInstrument *ins = voice[p_voice].instrument_ptr;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
|
|
|
if (!ins)
|
|
|
|
return 0;
|
2017-03-18 23:36:26 +00:00
|
|
|
|
|
|
|
switch (p_env_type) {
|
|
|
|
|
|
|
|
case CPInstrument::VOLUME_ENVELOPE: return ins->get_volume_envelope();
|
|
|
|
case CPInstrument::PAN_ENVELOPE: return ins->get_pan_envelope();
|
2014-02-10 01:10:30 +00:00
|
|
|
case CPInstrument::PITCH_ENVELOPE: return ins->get_pitch_filter_envelope();
|
|
|
|
};
|
2017-03-18 23:36:26 +00:00
|
|
|
|
2014-02-10 01:10:30 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
const char *CPPlayer::get_voice_instrument_name(int p_voice) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2015-05-03 00:16:26 +00:00
|
|
|
const char *name = NULL;
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
if (voice[p_voice].instrument_ptr != NULL) name = voice[p_voice].instrument_ptr->get_name();
|
2014-02-10 01:10:30 +00:00
|
|
|
|
|
|
|
return name;
|
|
|
|
}
|
2017-03-18 23:36:26 +00:00
|
|
|
void CPPlayer::set_filters_enabled(bool p_enable) {
|
2014-02-10 01:10:30 +00:00
|
|
|
|
2017-03-18 23:36:26 +00:00
|
|
|
control.filters = p_enable;
|
2014-02-10 01:10:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CPPlayer::get_voice_sample_index(int p_voice) {
|
|
|
|
|
|
|
|
return voice[p_voice].sample_index;
|
|
|
|
}
|