345 lines
8.3 KiB
C++
345 lines
8.3 KiB
C++
|
/*************************************************************************/
|
||
|
/* cp_instrument.cpp */
|
||
|
/*************************************************************************/
|
||
|
/* This file is part of: */
|
||
|
/* GODOT ENGINE */
|
||
|
/* http://www.godotengine.org */
|
||
|
/*************************************************************************/
|
||
|
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
|
||
|
/* */
|
||
|
/* 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_instrument.h"
|
||
|
#include "cp_song.h"
|
||
|
#include "cp_note.h"
|
||
|
|
||
|
|
||
|
|
||
|
const char *CPInstrument::get_name() {
|
||
|
|
||
|
return name;
|
||
|
|
||
|
}
|
||
|
void CPInstrument::set_name(const char *p_name) {
|
||
|
|
||
|
|
||
|
if (p_name==NULL) {
|
||
|
name[0]=0;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
bool done=false;
|
||
|
for (int i=0;i<MAX_NAME_LEN;i++) {
|
||
|
|
||
|
|
||
|
name[i]=done?0:p_name[i];
|
||
|
if (!done && p_name[i]==0)
|
||
|
done=true;
|
||
|
}
|
||
|
|
||
|
name[MAX_NAME_LEN-1]=0; /* just in case */
|
||
|
|
||
|
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_sample_number(uint8_t p_note,uint8_t p_sample_id) {
|
||
|
|
||
|
CP_ERR_COND(p_note>=CPNote::NOTES);
|
||
|
CP_ERR_COND(p_sample_id>CPSong::MAX_SAMPLES && p_sample_id!=CPNote::EMPTY);
|
||
|
data.sample_number[p_note]=p_sample_id;
|
||
|
|
||
|
|
||
|
}
|
||
|
uint8_t CPInstrument::get_sample_number(uint8_t p_note) {
|
||
|
|
||
|
CP_ERR_COND_V(p_note>=CPNote::NOTES,0);
|
||
|
return data.sample_number[p_note];
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_note_number(uint8_t p_note,uint8_t p_note_id) {
|
||
|
|
||
|
CP_ERR_COND(p_note>=CPNote::NOTES);
|
||
|
CP_ERR_COND(p_note_id>=CPNote::NOTES && p_note_id!=CPNote::EMPTY);
|
||
|
data.note_number[p_note]=p_note_id;
|
||
|
|
||
|
}
|
||
|
uint8_t CPInstrument::get_note_number(uint8_t p_note) {
|
||
|
|
||
|
CP_ERR_COND_V(p_note>=CPNote::NOTES,0);
|
||
|
return data.note_number[p_note];
|
||
|
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_NNA_type(NNA_Type p_NNA_type) {
|
||
|
|
||
|
data.NNA_type=p_NNA_type;
|
||
|
}
|
||
|
CPInstrument::NNA_Type CPInstrument::get_NNA_type() {
|
||
|
|
||
|
return data.NNA_type;
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_DC_type(DC_Type p_DC_type) {
|
||
|
|
||
|
data.DC_type=p_DC_type;
|
||
|
}
|
||
|
CPInstrument::DC_Type CPInstrument::get_DC_type() {
|
||
|
|
||
|
return data.DC_type;
|
||
|
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_DC_action(DC_Action p_DC_action) {
|
||
|
|
||
|
data.DC_action=p_DC_action;
|
||
|
}
|
||
|
CPInstrument::DC_Action CPInstrument::get_DC_action() {
|
||
|
|
||
|
return data.DC_action;
|
||
|
}
|
||
|
|
||
|
/* Volume */
|
||
|
|
||
|
void CPInstrument::set_volume_global_amount(uint8_t p_amount) {
|
||
|
|
||
|
CP_ERR_COND(p_amount>MAX_VOLUME);
|
||
|
data.volume.global_amount=p_amount;
|
||
|
|
||
|
}
|
||
|
uint8_t CPInstrument::get_volume_global_amount() {
|
||
|
|
||
|
return data.volume.global_amount;
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_volume_fadeout(uint16_t p_amount) {
|
||
|
CP_ERR_COND(p_amount>MAX_FADEOUT);
|
||
|
data.volume.fadeout=p_amount;
|
||
|
}
|
||
|
uint16_t CPInstrument::get_volume_fadeout() {
|
||
|
|
||
|
return data.volume.fadeout;
|
||
|
}
|
||
|
void CPInstrument::set_volume_random_variation(uint8_t p_amount) {
|
||
|
|
||
|
CP_ERR_COND(p_amount>MAX_VOLUME_RANDOM);
|
||
|
data.volume.random_variation=p_amount;
|
||
|
}
|
||
|
uint8_t CPInstrument::get_volume_random_variation() {
|
||
|
|
||
|
return data.volume.random_variation;
|
||
|
}
|
||
|
|
||
|
/* Panning */
|
||
|
|
||
|
void CPInstrument::set_pan_default_amount(uint8_t p_amount) {
|
||
|
|
||
|
CP_ERR_COND(p_amount>MAX_PAN);
|
||
|
data.pan.default_amount=p_amount;
|
||
|
}
|
||
|
uint8_t CPInstrument::get_pan_default_amount() {
|
||
|
|
||
|
return data.pan.default_amount;
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_pan_default_enabled(bool p_enabled) {
|
||
|
|
||
|
data.pan.use_default=p_enabled;
|
||
|
}
|
||
|
bool CPInstrument::is_pan_default_enabled() {
|
||
|
|
||
|
return data.pan.use_default;
|
||
|
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_pan_pitch_separation(int8_t p_amount) {
|
||
|
|
||
|
CP_ERR_COND(p_amount<-32);
|
||
|
CP_ERR_COND(p_amount>32);
|
||
|
data.pan.pitch_separation=p_amount;
|
||
|
}
|
||
|
int8_t CPInstrument::get_pan_pitch_separation() {
|
||
|
|
||
|
return data.pan.pitch_separation;
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_pan_pitch_center(uint8_t p_amount) {
|
||
|
|
||
|
CP_ERR_COND(p_amount>=CPNote::NOTES);
|
||
|
data.pan.pitch_center=p_amount;
|
||
|
}
|
||
|
uint8_t CPInstrument::get_pan_pitch_center() {
|
||
|
|
||
|
return data.pan.pitch_center;
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_pan_random_variation(uint8_t p_amount) {
|
||
|
|
||
|
CP_ERR_COND(p_amount>MAX_PAN_RANDOM);
|
||
|
data.pan.random_variation=p_amount;
|
||
|
}
|
||
|
uint8_t CPInstrument::get_pan_random_variation() {
|
||
|
|
||
|
return data.pan.random_variation;
|
||
|
}
|
||
|
|
||
|
/* Pitch / Filter */
|
||
|
|
||
|
void CPInstrument::set_pitch_use_as_filter(bool p_enabled) {
|
||
|
|
||
|
data.pitch.use_as_filter=p_enabled;
|
||
|
}
|
||
|
bool CPInstrument::is_pitch_use_as_filter() {
|
||
|
|
||
|
return data.pitch.use_as_filter;
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_filter_use_default_cutoff(bool p_enabled) {
|
||
|
|
||
|
data.pitch.use_default_cutoff=p_enabled;
|
||
|
|
||
|
}
|
||
|
bool CPInstrument::filter_use_default_cutoff() {
|
||
|
|
||
|
return data.pitch.use_default_cutoff;
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_filter_default_cutoff(uint8_t p_amount) {
|
||
|
|
||
|
CP_ERR_COND(p_amount>MAX_FILTER_CUTOFF);
|
||
|
data.pitch.default_cutoff=p_amount;
|
||
|
}
|
||
|
uint8_t CPInstrument::get_filter_default_cutoff() {
|
||
|
|
||
|
return data.pitch.default_cutoff;
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_filter_use_default_resonance(bool p_enabled) {
|
||
|
|
||
|
data.pitch.use_default_resonance=p_enabled;
|
||
|
}
|
||
|
bool CPInstrument::filter_use_default_resonance() {
|
||
|
|
||
|
return data.pitch.use_default_resonance;
|
||
|
}
|
||
|
|
||
|
void CPInstrument::set_filter_default_resonance(uint8_t p_amount) {
|
||
|
|
||
|
CP_ERR_COND(p_amount>MAX_FILTER_RESONANCE);
|
||
|
data.pitch.default_resonance=p_amount;
|
||
|
|
||
|
}
|
||
|
uint8_t CPInstrument::get_filter_default_resonance() {
|
||
|
|
||
|
return data.pitch.default_resonance;
|
||
|
}
|
||
|
|
||
|
/* Envelopes */
|
||
|
|
||
|
|
||
|
CPEnvelope* CPInstrument::get_volume_envelope() {
|
||
|
|
||
|
return &data.volume.envelope;
|
||
|
}
|
||
|
CPEnvelope* CPInstrument::get_pan_envelope() {
|
||
|
|
||
|
return &data.pan.envelope;
|
||
|
}
|
||
|
CPEnvelope* CPInstrument::get_pitch_filter_envelope() {
|
||
|
|
||
|
return &data.pitch.envelope;
|
||
|
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
void CPInstrument::reset() {
|
||
|
|
||
|
name[0]=0;
|
||
|
|
||
|
data.NNA_type=NNA_NOTE_CUT;
|
||
|
data.DC_action=DCA_NOTE_CUT;
|
||
|
data.DC_type=DCT_DISABLED;
|
||
|
|
||
|
for (int i=0;i<CPNote::NOTES;i++) {
|
||
|
|
||
|
data.sample_number[i]=CPNote::EMPTY;
|
||
|
data.note_number[i]=i;
|
||
|
}
|
||
|
|
||
|
data.volume.envelope.reset();
|
||
|
data.volume.envelope.set_max(64);
|
||
|
data.volume.envelope.set_min(0);
|
||
|
data.volume.envelope.add_position(0,64,false);
|
||
|
data.volume.envelope.add_position(30,64,false);
|
||
|
|
||
|
data.volume.global_amount=MAX_VOLUME;
|
||
|
data.volume.fadeout=0;
|
||
|
data.volume.random_variation=0;
|
||
|
|
||
|
data.pan.envelope.reset();
|
||
|
data.pan.envelope.set_max(32);
|
||
|
data.pan.envelope.set_min(-32);
|
||
|
data.pan.envelope.add_position(0,0,false);
|
||
|
data.pan.envelope.add_position(30,0,false);
|
||
|
|
||
|
data.pan.default_amount=32;
|
||
|
data.pan.pitch_center=48;
|
||
|
data.pan.pitch_separation=0;
|
||
|
data.pan.use_default=false;
|
||
|
data.pan.random_variation=0;
|
||
|
|
||
|
|
||
|
data.pitch.envelope.reset();
|
||
|
data.pitch.envelope.set_max(32);
|
||
|
data.pitch.envelope.set_min(-32);
|
||
|
data.pitch.envelope.add_position(0,0,false);
|
||
|
data.pitch.envelope.add_position(30,0,false);
|
||
|
data.pitch.use_as_filter=false;
|
||
|
data.pitch.use_default_cutoff=false;
|
||
|
data.pitch.use_default_resonance=false;
|
||
|
data.pitch.default_cutoff=0;
|
||
|
data.pitch.default_resonance=0;
|
||
|
|
||
|
}
|
||
|
|
||
|
bool CPInstrument::is_empty() {
|
||
|
|
||
|
bool has_sample=false;
|
||
|
|
||
|
for (int i=0;i<CPNote::NOTES;i++) {
|
||
|
|
||
|
if (data.sample_number[i]!=CPNote::EMPTY) {
|
||
|
|
||
|
has_sample=true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return !has_sample;
|
||
|
}
|
||
|
|
||
|
CPInstrument::CPInstrument() {
|
||
|
|
||
|
reset();
|
||
|
|
||
|
}
|
||
|
|