3 // negative = SVQC autochannels
4 // positive = one per entity
7 const int CH_WEAPON_A = -1;
8 const int CH_WEAPON_B = -1;
9 const int CH_WEAPON_SINGLE = 1;
10 const int CH_VOICE = -2;
11 // const int CH_VOICE_SINGLE = 2;
12 const int CH_TRIGGER = -3;
13 const int CH_TRIGGER_SINGLE = 3;
14 const int CH_SHOTS = -4;
15 const int CH_SHOTS_SINGLE = 4;
16 // const int CH_TUBA = -5;
17 const int CH_TUBA_SINGLE = 5;
18 const int CH_PAIN = -6;
19 const int CH_PAIN_SINGLE = 6;
20 const int CH_PLAYER = -7;
21 const int CH_PLAYER_SINGLE = 7;
22 // const int CH_BGM_SINGLE = -8;
23 const int CH_BGM_SINGLE = 8;
24 const int CH_AMBIENT = -9;
25 const int CH_AMBIENT_SINGLE = 9;
27 const float ATTEN_NONE = 0;
28 const float ATTEN_MIN = 0.015625;
29 const float ATTEN_LOW = 0.2;
30 const float ATTEN_NORM = 0.5;
31 const float ATTEN_LARGE = 1;
32 const float ATTEN_IDLE = 2;
33 const float ATTEN_STATIC = 3;
34 const float ATTEN_MAX = 3.984375;
36 const float VOL_BASE = 0.7;
37 const float VOL_BASEVOICE = 1.0;
38 const float VOL_MUFFLED = 0.35;
40 // Play all sounds via sound7, for access to the extra channels.
41 // Otherwise, channels 8 to 15 would be blocked for a weird QW feature.
43 #define _sound(e, c, s, v, a) \
46 if (sound_allowed(MSG_BROADCAST, __e)) \
47 sound7(__e, c, s, v, a, 0, 0); \
50 #define _sound(e, c, s, v, a) sound7(e, c, s, v, a, 0, 0)
52 #define sound(e, c, s, v, a) _sound(e, c, Sound_fixpath(s), v, a)
55 * because sound7 didn't have origin
57 * @param e sound owner
58 * @param o sound origin
59 * @param chan sound channel
60 * @param samp sound filename
61 * @param vol sound volume
62 * @param atten sound attenuation
66 #define sound8(e, o, chan, samp, vol, atten, speed, sf) \
70 string __samp = samp; \
72 if (__chan > 0) __e = e; \
76 __chan = fabs(__chan); \
77 entity tmp = __e = new(csqc_autochannel); \
78 setthink(tmp, SUB_Remove); \
79 tmp.nextthink = time + soundlength(__samp); \
81 vector old_origin = __e.origin; \
82 vector old_mins = __e.mins; \
83 vector old_maxs = __e.maxs; \
85 setsize(__e, '0 0 0', '0 0 0'); \
86 sound7(__e, __chan, __samp, vol, atten, speed, sf); \
89 setorigin(__e, old_origin); \
90 setsize(__e, old_mins, old_maxs); \
94 string _Sound_fixpath(string base)
96 if (base == "") return string_null;
98 return strcat(base, ".wav"); // let the client engine decide
100 #define extensions(x) \
105 #define tryext(ext) { \
106 string s = strcat(base, "." #ext); \
107 if (fexists(strcat("sound/", s))) { \
112 LOG_WARNF("Missing sound: \"%s\"", strcat("sound/", base));
120 ATTRIB(Sound, m_id, int, 0);
121 ATTRIB(Sound, sound_str, string());
122 ATTRIB(Sound, sound_str_, string);
123 CONSTRUCTOR(Sound, string() path)
126 this.sound_str = path;
128 METHOD(Sound, sound_precache, void(Sound this))
131 string s = _Sound_fixpath(this.sound_str());
133 //profile(sprintf("precache_sound(\"%s\")", s));
135 strcpy(this.sound_str_, s);
139 entity _Sound_fixpath_this;
140 string _Sound_fixpath_cached;
141 #define Sound_fixpath(this) ( \
142 _Sound_fixpath_this = (this), \
143 _Sound_fixpath_cached = _Sound_fixpath_this.sound_str_, \
144 _Sound_fixpath_cached ? _Sound_fixpath_cached : _Sound_fixpath(_Sound_fixpath_this.sound_str()) \