3 #include <server/defs.qh>
4 #include <server/miscfunctions.qh>
5 #include <server/items.qh>
6 #include <server/resources.qh>
7 #include <common/t_items.qh>
8 #include <common/mapobjects/triggers.qh>
9 #include <common/mapobjects/trigger/counter.qh>
10 #include <common/mutators/mutator/buffs/buffs.qh>
11 #include <common/notifications/all.qh>
12 #include <common/weapons/_all.qh>
14 //***********************
15 //QUAKE 3 ENTITIES - So people can play quake3 maps with the xonotic weapons
16 //***********************
18 // NOTE: for best experience, you need to swap MGs with SGs in the map or it won't have a MG
21 SPAWNFUNC_ITEM(ammo_shells, ITEM_Shells)
24 SPAWNFUNC_ITEM(ammo_bullets, ITEM_Bullets)
27 SPAWNFUNC_ITEM(ammo_grenades, ITEM_Rockets)
30 SPAWNFUNC_WEAPON(weapon_prox_launcher, WEP_MINE_LAYER)
31 SPAWNFUNC_ITEM(ammo_mines, ITEM_Rockets)
33 // Q3TA chaingun -> HLAC
34 SPAWNFUNC_WEAPON(weapon_chaingun, WEP_HLAC)
35 SPAWNFUNC_ITEM(ammo_belt, ITEM_Cells)
37 // Q3TA nails -> cells
38 SPAWNFUNC_ITEM(ammo_nails, ITEM_Cells)
41 SPAWNFUNC_WEAPON(weapon_lightning, WEP_ELECTRO)
42 SPAWNFUNC_ITEM(ammo_lightning, ITEM_Cells)
45 SPAWNFUNC_WEAPON(weapon_plasmagun, WEP_HAGAR)
46 SPAWNFUNC_ITEM(ammo_cells, ITEM_Rockets)
49 SPAWNFUNC_WEAPON(weapon_railgun, WEP_VORTEX)
50 SPAWNFUNC_ITEM(ammo_slugs, ITEM_Cells)
53 SPAWNFUNC_WEAPON(weapon_bfg, WEP_CRYLINK)
54 SPAWNFUNC_ITEM(ammo_bfg, ITEM_Cells)
56 // grappling hook -> hook
57 SPAWNFUNC_WEAPON(weapon_grapplinghook, WEP_HOOK)
60 SPAWNFUNC_ITEM(ammo_rockets, ITEM_Rockets)
63 SPAWNFUNC_ITEM(item_armor_body, ITEM_ArmorMega)
64 SPAWNFUNC_ITEM(item_armor_combat, ITEM_ArmorBig)
65 SPAWNFUNC_ITEM(item_armor_shard, ITEM_ArmorSmall)
66 SPAWNFUNC_ITEM(item_enviro, ITEM_Shield)
68 // medkit -> armor (we have no holdables)
69 SPAWNFUNC_ITEM(holdable_medkit, ITEM_ArmorBig)
74 // weapon remove ent from df
75 void target_init_verify(entity this)
78 for(trigger = NULL; (trigger = find(trigger, classname, "trigger_multiple")); )
79 for(targ = NULL; (targ = find(targ, targetname, trigger.target)); )
80 if (targ.classname == "target_init" || targ.classname == "target_give" || targ.classname == "target_items")
87 //setsize(targ, trigger.mins, trigger.maxs);
88 //setorigin(targ, trigger.origin);
93 void target_init_use(entity this, entity actor, entity trigger)
95 if (!(this.spawnflags & 1))
97 SetResource(actor, RES_ARMOR, start_armorvalue);
98 actor.pauserotarmor_finished = time + autocvar_g_balance_pause_armor_rot;
101 if (!(this.spawnflags & 2))
103 SetResource(actor, RES_HEALTH, start_health);
104 actor.pauserothealth_finished = time + autocvar_g_balance_pause_health_rot;
105 actor.pauseregen_finished = time + autocvar_g_balance_pause_health_regen;
108 if (!(this.spawnflags & 4))
110 if(this.spawnflags & 32) // spawn with only melee
112 SetResource(actor, RES_SHELLS, 0);
113 SetResource(actor, RES_BULLETS, 0);
114 SetResource(actor, RES_ROCKETS, 0);
115 SetResource(actor, RES_CELLS, 0);
116 SetResource(actor, RES_PLASMA, 0);
117 SetResource(actor, RES_FUEL, 0);
119 STAT(WEAPONS, actor) = WEPSET(SHOTGUN);
123 SetResource(actor, RES_SHELLS, start_ammo_shells);
124 SetResource(actor, RES_BULLETS, start_ammo_nails);
125 SetResource(actor, RES_ROCKETS, start_ammo_rockets);
126 SetResource(actor, RES_CELLS, start_ammo_cells);
127 SetResource(actor, RES_PLASMA, start_ammo_plasma);
128 SetResource(actor, RES_FUEL, start_ammo_fuel);
130 STAT(WEAPONS, actor) = start_weapons;
134 if (!(this.spawnflags & 8))
136 STAT(STRENGTH_FINISHED, actor) = 0;
137 STAT(INVINCIBLE_FINISHED, actor) = 0;
138 if(STAT(BUFFS, actor)) // TODO: make a dropbuffs function to handle this
140 int buffid = buff_FirstFromFlags(STAT(BUFFS, actor)).m_id;
141 Send_Notification(NOTIF_ONE, actor, MSG_MULTI, ITEM_BUFF_DROP, buffid);
142 sound(actor, CH_TRIGGER, SND_BUFF_LOST, VOL_BASE, ATTN_NORM);
143 if(!IS_INDEPENDENT_PLAYER(actor))
144 Send_Notification(NOTIF_ALL_EXCEPT, actor, MSG_INFO, INFO_ITEM_BUFF_LOST, actor.netname, buffid);
145 STAT(BUFFS, actor) = 0;
146 STAT(BUFF_TIME, actor) = 0;
150 if (!(this.spawnflags & 16))
152 // We don't have holdables.
155 SUB_UseTargets(this, actor, trigger);
158 spawnfunc(target_init)
160 this.use = target_init_use;
161 InitializeEntity(this, target_init_verify, INITPRIO_FINDTARGET);
164 // weapon give ent from defrag
165 void target_give_init(entity this)
167 IL_EACH(g_items, it.targetname == this.target,
169 if (it.classname == "weapon_devastator") {
170 SetResourceExplicit(this, RES_ROCKETS, GetResource(this, RES_ROCKETS) + it.count * WEP_CVAR_PRI(devastator, ammo)); // WEAPONTODO
171 this.netname = cons(this.netname, "devastator");
173 else if (it.classname == "weapon_vortex") {
174 SetResourceExplicit(this, RES_CELLS, GetResource(this, RES_CELLS) + it.count * WEP_CVAR_PRI(vortex, ammo)); // WEAPONTODO
175 this.netname = cons(this.netname, "vortex");
177 else if (it.classname == "weapon_electro") {
178 SetResourceExplicit(this, RES_CELLS, GetResource(this, RES_CELLS) + it.count * WEP_CVAR_PRI(electro, ammo)); // WEAPONTODO
179 this.netname = cons(this.netname, "electro");
181 else if (it.classname == "weapon_hagar") {
182 SetResourceExplicit(this, RES_ROCKETS, GetResource(this, RES_ROCKETS) + it.count * WEP_CVAR_PRI(hagar, ammo)); // WEAPONTODO
183 this.netname = cons(this.netname, "hagar");
185 else if (it.classname == "weapon_crylink") {
186 SetResourceExplicit(this, RES_CELLS, GetResource(this, RES_CELLS) + it.count * WEP_CVAR_PRI(crylink, ammo)); // WEAPONTODO
187 this.netname = cons(this.netname, "crylink");
189 else if (it.classname == "weapon_mortar") {
190 SetResourceExplicit(this, RES_ROCKETS, GetResource(this, RES_ROCKETS) + it.count * WEP_CVAR_PRI(mortar, ammo)); // WEAPONTODO
191 this.netname = cons(this.netname, "mortar");
193 else if (it.classname == "weapon_shotgun") {
194 SetResourceExplicit(this, RES_SHELLS, GetResource(this, RES_SHELLS) + it.count * WEP_CVAR_PRI(shotgun, ammo)); // WEAPONTODO
195 this.netname = cons(this.netname, "shotgun");
197 else if (it.classname == "item_armor_mega")
198 SetResourceExplicit(this, RES_ARMOR, 100);
199 else if (it.classname == "item_health_mega")
200 SetResourceExplicit(this, RES_HEALTH, 200);
201 else if (it.classname == "item_buff") {
202 entity buff = buff_FirstFromFlags(STAT(BUFFS, it));
203 this.netname = cons(this.netname, buff.netname);
204 STAT(BUFF_TIME, this) = it.count;
207 //remove(it); // removing ents in init functions causes havoc, workaround:
208 setthink(it, SUB_Remove);
212 this.spawnfunc_checked = true;
213 spawnfunc_target_items(this);
214 InitializeEntity(this, target_init_verify, INITPRIO_FINDTARGET);
217 spawnfunc(target_give)
219 InitializeEntity(this, target_give_init, INITPRIO_FINDTARGET);
222 void score_use(entity this, entity actor, entity trigger)
224 if(!IS_PLAYER(actor))
226 actor.fragsfilter_cnt += this.count;
228 spawnfunc(target_score)
230 if(!g_cts) { delete(this); return; }
234 this.use = score_use;
237 void fragsfilter_use(entity this, entity actor, entity trigger)
239 if(!IS_PLAYER(actor))
241 if(actor.fragsfilter_cnt >= this.frags)
242 SUB_UseTargets(this, actor, trigger);
244 spawnfunc(target_fragsFilter)
246 if(!g_cts) { delete(this); return; }
250 this.use = fragsfilter_use;
253 //spawnfunc(item_flight) /* handled by buffs mutator */
254 //spawnfunc(item_doubler) /* handled by buffs mutator */
255 //spawnfunc(item_haste) /* handled by buffs mutator */
256 //spawnfunc(item_health) /* handled in t_quake.qc */
257 //spawnfunc(item_health_large) /* handled in t_items.qc */
258 //spawnfunc(item_health_small) /* handled in t_items.qc */
259 //spawnfunc(item_health_mega) /* handled in t_items.qc */
260 //spawnfunc(item_invis) /* handled by buffs mutator */
261 //spawnfunc(item_regen) /* handled by buffs mutator */
263 // CTF spawnfuncs handled in mutators/gamemode_ctf.qc now
272 bool DoesQ3ARemoveThisEntity(entity this)
274 // Q3 style filters (DO NOT USE, THIS IS COMPAT ONLY)
276 // DeFRaG mappers use "notcpm" or "notvq3" to disable an entity in CPM or VQ3 physics
277 // Xonotic is usually played with a CPM-based physics so we default to CPM mode
278 if(cvar_string("g_mod_physics") == "Q3") {
279 if(this.notvq3) return true;
281 else if(this.notcpm) return true;
283 // Q3 mappers use "notq3a" or "notta" to disable an entity in Q3A or Q3TA
284 // Xonotic has ~equivalent features to Team Arena
303 // static char *gametypeNames[] = {"ffa", "tournament", "single", "team", "ctf", "oneflag", "obelisk", "harvester", "teamtournament"}
304 gametypename = "ffa";
306 gametypename = "team";
308 gametypename = "ctf";
310 gametypename = "tournament";
312 gametypename = "single";
313 // we do not have the other types (oneflag, obelisk, harvester, teamtournament)
314 if(strstrofs(this.gametype, gametypename, 0) < 0)