# include <simgear_config.h>
#endif
+#include "xmlsound.hxx"
+
+
#include <simgear/compiler.h>
#include <string.h>
#include <simgear/debug/logstream.hxx>
+#include <simgear/props/props.hxx>
#include <simgear/props/condition.hxx>
-#include <simgear/math/SGMath.hxx>
+#include <simgear/structure/exception.hxx>
+#include <simgear/misc/sg_path.hxx>
+#include "sample_group.hxx"
+#include "sample_openal.hxx"
-#include "xmlsound.hxx"
-
+using std::string;
// static double _snd_lin(double v) { return v; }
static double _snd_inv(double v) { return (v == 0) ? 1e99 : 1/v; }
_dt_play(0.0),
_dt_stop(0.0),
_delay(0.0),
- _stopping(0.0)
+ _stopping(0.0),
+ _initialized(false)
{
}
void
SGXmlSound::init(SGPropertyNode *root, SGPropertyNode *node,
- SGSampleGroup *sgrp, const string &path)
+ SGSampleGroup *sgrp, SGSampleGroup *avionics,
+ const SGPath& currentDir)
{
//
//
_name = node->getStringValue("name", "");
- SG_LOG(SG_GENERAL, SG_DEBUG, "Loading sound information for: " << _name );
+ SG_LOG(SG_SOUND, SG_DEBUG, "Loading sound information for: " << _name );
- const char *mode_str = node->getStringValue("mode", "");
- if ( !strcmp(mode_str, "looped") ) {
+ string mode_str = node->getStringValue("mode", "");
+ if ( mode_str == "looped" ) {
_mode = SGXmlSound::LOOPED;
- } else if ( !strcmp(mode_str, "in-transit") ) {
+ } else if ( mode_str == "in-transit" ) {
_mode = SGXmlSound::IN_TRANSIT;
} else {
_mode = SGXmlSound::ONCE;
}
- _property = root->getNode(node->getStringValue("property", ""), true);
+ bool is_avionics = false;
+ string type_str = node->getStringValue("type", "fx");
+ if ( type_str == "avionics" )
+ is_avionics = true;
+
+ string propval = node->getStringValue("property", "");
+ if (propval != "")
+ _property = root->getNode(propval, true);
+
SGPropertyNode *condition = node->getChild("condition");
if (condition != NULL)
_condition = sgReadCondition(root, condition);
if (!_property && !_condition)
- SG_LOG(SG_GENERAL, SG_WARN,
+ SG_LOG(SG_SOUND, SG_WARN,
" Neither a condition nor a property specified");
_delay = node->getDoubleValue("delay-sec", 0.0);
for (i = 0; (i < kids.size()) && (i < SGXmlSound::MAXPROP); i++) {
_snd_prop volume = {NULL, NULL, NULL, 1.0, 0.0, 0.0, 0.0, false};
- if (strcmp(kids[i]->getStringValue("property"), ""))
- volume.prop = root->getNode(kids[i]->getStringValue("property", ""), true);
+ propval = kids[i]->getStringValue("property", "");
+ if ( propval != "" )
+ volume.prop = root->getNode(propval, true);
- const char *intern_str = kids[i]->getStringValue("internal", "");
- if (!strcmp(intern_str, "dt_play"))
+ string intern_str = kids[i]->getStringValue("internal", "");
+ if (intern_str == "dt_play")
volume.intern = &_dt_play;
- else if (!strcmp(intern_str, "dt_stop"))
+ else if (intern_str == "dt_stop")
volume.intern = &_dt_stop;
if ((volume.factor = kids[i]->getDoubleValue("factor", 1.0)) != 0.0)
volume.subtract = true;
}
- const char *type_str = kids[i]->getStringValue("type", "");
- if ( strcmp(type_str, "") ) {
+ string type_str = kids[i]->getStringValue("type", "");
+ if ( type_str != "" ) {
for (int j=0; __sound_fn[j].fn; j++)
- if ( !strcmp(type_str, __sound_fn[j].name) ) {
+ if ( type_str == __sound_fn[j].name ) {
volume.fn = __sound_fn[j].fn;
break;
}
if (!volume.fn)
- SG_LOG(SG_GENERAL,SG_INFO,
+ SG_LOG(SG_SOUND,SG_INFO,
" Unknown volume type, default to 'lin'");
}
volume.offset = kids[i]->getDoubleValue("offset", 0.0);
if ((volume.min = kids[i]->getDoubleValue("min", 0.0)) < 0.0)
- SG_LOG( SG_GENERAL, SG_WARN,
+ SG_LOG( SG_SOUND, SG_WARN,
"Volume minimum value below 0. Forced to 0.");
volume.max = kids[i]->getDoubleValue("max", 0.0);
if (volume.max && (volume.max < volume.min) )
- SG_LOG(SG_GENERAL,SG_ALERT,
+ SG_LOG(SG_SOUND,SG_ALERT,
" Volume maximum below minimum. Neglected.");
_volume.push_back(volume);
}
- float reference_dist = node->getDoubleValue("reference-dist", 500.0);
- float max_dist = node->getDoubleValue("max-dist", 3000.0);
-
+ // rule of thumb: make reference distance a 100th of the maximum distance.
+ float reference_dist = node->getDoubleValue("reference-dist", 60.0);
+ float max_dist = node->getDoubleValue("max-dist", 6000.0);
+
//
// set pitch properties
//
for (i = 0; (i < kids.size()) && (i < SGXmlSound::MAXPROP); i++) {
_snd_prop pitch = {NULL, NULL, NULL, 1.0, 1.0, 0.0, 0.0, false};
- if (strcmp(kids[i]->getStringValue("property", ""), ""))
- pitch.prop = root->getNode(kids[i]->getStringValue("property", ""), true);
+ propval = kids[i]->getStringValue("property", "");
+ if (propval != "")
+ pitch.prop = root->getNode(propval, true);
- const char *intern_str = kids[i]->getStringValue("internal", "");
- if (!strcmp(intern_str, "dt_play"))
+ string intern_str = kids[i]->getStringValue("internal", "");
+ if (intern_str == "dt_play")
pitch.intern = &_dt_play;
- else if (!strcmp(intern_str, "dt_stop"))
+ else if (intern_str == "dt_stop")
pitch.intern = &_dt_stop;
if ((pitch.factor = kids[i]->getDoubleValue("factor", 1.0)) != 0.0)
pitch.subtract = true;
}
- const char *type_str = kids[i]->getStringValue("type", "");
- if ( strcmp(type_str, "") ) {
+ string type_str = kids[i]->getStringValue("type", "");
+ if ( type_str != "" ) {
for (int j=0; __sound_fn[j].fn; j++)
- if ( !strcmp(type_str, __sound_fn[j].name) ) {
+ if ( type_str == __sound_fn[j].name ) {
pitch.fn = __sound_fn[j].fn;
break;
}
if (!pitch.fn)
- SG_LOG(SG_GENERAL,SG_INFO,
+ SG_LOG(SG_SOUND,SG_INFO,
" Unknown pitch type, default to 'lin'");
}
pitch.offset = kids[i]->getDoubleValue("offset", 1.0);
if ((pitch.min = kids[i]->getDoubleValue("min", 0.0)) < 0.0)
- SG_LOG(SG_GENERAL,SG_WARN,
+ SG_LOG(SG_SOUND,SG_WARN,
" Pitch minimum value below 0. Forced to 0.");
pitch.max = kids[i]->getDoubleValue("max", 0.0);
if (pitch.max && (pitch.max < pitch.min) )
- SG_LOG(SG_GENERAL,SG_ALERT,
+ SG_LOG(SG_SOUND,SG_ALERT,
" Pitch maximum below minimum. Neglected");
_pitch.push_back(pitch);
SGVec3f offset_pos = SGVec3f::zeros();
SGPropertyNode_ptr prop = node->getChild("position");
if ( prop != NULL ) {
- offset_pos[0] = prop->getDoubleValue("y", 0.0);
- offset_pos[1] = prop->getDoubleValue("z", 0.0);
- offset_pos[2] = prop->getDoubleValue("x", 0.0);
+ offset_pos[0] = -prop->getDoubleValue("x", 0.0);
+ offset_pos[1] = -prop->getDoubleValue("y", 0.0);
+ offset_pos[2] = -prop->getDoubleValue("z", 0.0);
}
//
float outer_gain = 0.0;
prop = node->getChild("orientation");
if ( prop != NULL ) {
- dir = SGVec3f(prop->getFloatValue("y", 0.0),
- prop->getFloatValue("z", 0.0),
- prop->getFloatValue("x", 0.0));
+ dir = SGVec3f(-prop->getFloatValue("x", 0.0),
+ -prop->getFloatValue("y", 0.0),
+ -prop->getFloatValue("z", 0.0));
inner = prop->getFloatValue("inner-angle", 360.0);
outer = prop->getFloatValue("outer-angle", 360.0);
outer_gain = prop->getFloatValue("outer-gain", 0.0);
//
// Initialize the sample
//
- _sgrp = sgrp;
- _sample = new SGSoundSample( path.c_str(), node->getStringValue("path", ""));
+ if ((is_avionics)&&(avionics)) {
+ _sgrp = avionics;
+ } else {
+ _sgrp = sgrp;
+ }
+ string soundFileStr = node->getStringValue("path", "");
+ _sample = new SGSoundSample(soundFileStr.c_str(), currentDir);
+ if (!_sample->file_path().exists()) {
+ throw sg_io_exception("XML sound: couldn't find file: '" + soundFileStr + "'");
+ }
+
_sample->set_relative_position( offset_pos );
_sample->set_direction( dir );
_sample->set_audio_cone( inner, outer, outer_gain );
if (_property)
curr_value = _property->getDoubleValue();
+ if (!_initialized)
+ {
+ // update initial value before detecting changes
+ _prev_value = curr_value;
+ _initialized = true;
+ }
+
// If a condition is defined, test whether it is FALSE,
// else
// if a property is defined then test if it's value is FALSE
if ((_mode != SGXmlSound::IN_TRANSIT) || (_stopping > MAX_TRANSIT_TIME))
{
if (_sample->is_playing()) {
- SG_LOG(SG_GENERAL, SG_DEBUG, "Stopping audio after " << _dt_play
+ SG_LOG(SG_SOUND, SG_DEBUG, "Stopping audio after " << _dt_play
<< " sec: " << _name );
_sample->stop();
double vol = volume_offset + volume;
if (vol > 1.0) {
- SG_LOG(SG_GENERAL, SG_DEBUG, "Sound volume too large for '"
+ SG_LOG(SG_SOUND, SG_DEBUG, "Sound volume too large for '"
<< _name << "': " << vol << " -> clipping to 1.0");
vol = 1.0;
}
else
_sample->play(true);
- SG_LOG(SG_GENERAL, SG_DEBUG, "Playing audio after " << _dt_stop
+ SG_LOG(SG_SOUND, SG_DEBUG, "Playing audio after " << _dt_stop
<< " sec: " << _name);
- SG_LOG(SG_GENERAL, SG_DEBUG,
+ SG_LOG(SG_SOUND, SG_DEBUG,
"Playing " << ((_mode == ONCE) ? "once" : "looped"));
_active = true;