Rune Central http://runequake.com/forums/ |
|
Colored Holograms and Bodies in GL http://runequake.com/forums/viewtopic.php?f=2&t=826 |
Page 1 of 1 |
Author: | Baker [ Sun Aug 07, 2005 5:06 pm ] |
Post subject: | Colored Holograms and Bodies in GL |
I had never seen colored dead bodies in GL Quake before but some various FrogBot mods do it. Source code of one Frogbot mod with colored bodies in GL: http://www.parboil.quakeworld.ru/parboil/src101.zip I noticed that the Quakeworld server, qw.shmack.net, has colored holograms. |
Author: | Baker [ Tue Sep 20, 2005 8:11 am ] |
Post subject: | |
http://www.inside3d.com/qip/q1/qcwa.htm#gl_skins Quote: Workaround for GLQuake "Invalid skin #<number>" message and dead bodies colors by Robert Field
This workaround should work with any existing mod, it was originally for the Frogbot mod. It was put in a form that can be easily interfaced to other mods. But this workaround is not a final solution, for the following reasons: * You have to set "maxplayers" to the number of players plus 4 to see colored bodies * If you change your colors during gameplay, your corpses will not change * The "Invalid skin #<number> workaround will only work if the clients have a "player.mdl" file with the skin number in the player model (eyes and head models are used with skin 0). When players get the eyes or use a player.mdl which don't have heads (besides skin 0), then the invalid skin error is now avoided. * Unfortunately, the shirt color is not accessable from QuakeC, though if shirt colors are set by your mod code then it is easy to change WriteByte(MSG_ALL, (ent.team - 1) * 17); in "skinfix.qc" to WriteByte(MSG_ALL, (ent.shirt * 16 + ent.team - 1)); to set proper shirt colors. Code: First create a new file called "SkinFix.qc" which will contain some new and replacement functions for "InitBodyQue()" and "CopyToBodyQue()" in "World.qc". "SkinFix.qc" /* These functions give dead bodies color and avoid the "Invalid skin" message under GLQuake (currently v0.97) The first maxplayers entities (ie. the entities besides world that exist at the start of worldspawn) are used by clients when they connect. The client entities are the only ones that have colors for player models in GLQuake. Spare client entities can be used for dead bodies. The approach of this workaround is to use client entities for player models where possible and use normal entities for head gibs and eyes. Hence head gibs and eyes always have skin 0, thus avoiding the invalid player skin # error. If there is not 4 spare client entities then some bodies use normal entities and hence don't have color in GLQuake. */ // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field // file added /* ============ PostThink Displays alternative entities for players or bodies if necessary each frame. (the actual player or body entity in these cases are not rendered) P.S.: don't mix up with PlayerPostThink() in client.qc ============ */ void() PostThink = { self.nextthink = 0.001; // call PostThink every frame // loop through players linked list and display their eye or head gib model if necessary self = find(world, classname_, "player"); while (self) { if (self.display_ent.modelindex) { // self.display_ent is a non-client entity setorigin(self.display_ent, self.origin); self.display_ent.angles = self.angles; self.display_ent.effects = self.effects; } self = find(self, classname_, "player"); } // loop through bodies linked list and display their player model if necessary self = first_bodyque; while (self) { // self.display_ent is a spare client entity or world // self.display_ent.modelindex == 0 if a non-player model is being used, and self.display_ent is a spare client // self.display_ent.modelindex == 1 if self.display_ent == world if (self.display_ent.modelindex > 1) setorigin(self.display_ent, self.origin); self = self.owner; } }; /* ============ FindClientBodyQue Assigns a spare client entity to bodyque_entry. If none spare then assign world. Start search for client entity at bodyque_client. ============ */ void(entity bodyque_entry) FindClientBodyQue = { // post_think is the first entity after the client entities while (bodyque_client != post_think) { // check if bodyque_client is currently used by a client or body if (bodyque_client.classname_ != "player" && bodyque_client.classname_ != "body") { // bodyque_client is spare bodyque_entry.display_ent = bodyque_client; bodyque_client.classname_ = "body"; bodyque_client.score_pos = score_pos_; return; } // get next bodyque_client and increment its index score_pos_ = score_pos_ + 1; bodyque_client = nextent(bodyque_client); } bodyque_entry.display_ent = world; // no spare client entities }; /* ============ AssignBodies Assigns spare client entities to the 4 body entities. If none spare then assign world. This function is called whenever there is a change in the connected client entities list. ============ */ void() AssignBodies = { local entity bodyque_entry; // get first bodyque_client (with index 0) score_pos_ = 0; bodyque_client = nextent(world); // loop through bodies linked list and assign client entities if possible bodyque_entry = first_bodyque; while (bodyque_entry) { if (bodyque_entry.classname_ != "body") FindClientBodyQue(bodyque_entry); bodyque_entry = bodyque_entry.owner; } }; /* ============ InitBodyQue The bodies form a linked list (in the original id code they form a ring). Called at the start of worldspawn so that post_think is the first entity after the client entities. ============ */ void() InitBodyQue = { post_think = spawn(); post_think.nextthink = 0.001; post_think.think = PostThink; bodyque_head = first_bodyque = spawn(); bodyque_head.owner = spawn(); bodyque_head.owner.owner = spawn(); bodyque_head.owner.owner.owner = spawn(); }; /* ============ CopyToBodyQue Called whenever the original id CopyToBodyQue was called. ============ */ void(entity ent) CopyToBodyQue = { local entity bodyque_entry; // the entity that renders the body if (ent.modelindex) { // ent is using the player model if (bodyque_head.display_ent) { // bodyque_head is using a spare client entity (ie. bodyque_entry) bodyque_entry = bodyque_head.display_ent; bodyque_head.modelindex = 0; // don't render bodyque_head // set body colors for GLQuake (ignored by normal Quake since it takes notice of .colormap below) WriteByte(MSG_ALL, MSG_UPDATECOLORS); WriteByte(MSG_ALL, bodyque_entry.score_pos); // the index of the client entity bodyque_entry WriteByte(MSG_ALL, (ent.team - 1) * 17); // GLQuake color (ent.team - 1) // if ent has a shirt color ent.shirt then replace this last line by // WriteByte(MSG_ALL, ent.shirt * 16 + (ent.team - 1)); setorigin (bodyque_entry, ent.origin); // just in case PostThink has already been called this frame } else { // bodyque_head doesn't have a client entity so use itself (its color won't work in GLQuake though) bodyque_entry = bodyque_head; } bodyque_entry.skin = ent.skin; bodyque_entry.modelindex = ent.modelindex; } else { // ent is not using the player model if (bodyque_head.display_ent) bodyque_head.display_ent.modelindex = 0; // don't render the client entity bodyque_head.display_ent bodyque_entry = bodyque_head; bodyque_entry.skin = 0; bodyque_entry.modelindex = ent.display_ent.modelindex; } // set bodyque_entry.model to != "" bodyque_entry.model = "/"; // set body colors for normal Quake (ignored by GLQuake) bodyque_entry.colormap = ent.colormap; // copy info display info of ent bodyque_entry.angles = ent.angles; bodyque_entry.frame = ent.frame; // copy physics info of ent bodyque_head.velocity = ent.velocity; bodyque_head.flags = 0; setorigin (bodyque_head, ent.origin); setsize (bodyque_head, ent.mins, ent.maxs); bodyque_head.movetype = MOVETYPE_TOSS; // move along bodyque_head (note: using a linked list not a ring) bodyque_head = bodyque_head.owner; if (!bodyque_head) bodyque_head = first_bodyque; }; /* ============ SkinFixConnect Called when client connects. Assigns client an entity which is used for its head gib or eye model. ============ */ void(entity client) SkinFixConnect = { client.classname_ = "player"; client.display_ent = spawn(); // set client.display_ent to != "" client.display_ent.model = "/"; }; /* ============ SkinFixDisConnect Called when client disconnects. Does garbage collection for client.display_ent. ============ */ void(entity client) SkinFixDisConnect = { // since removing client.display_ent need to copy it if it is being rendered if (client.display_ent.modelindex) CopyToBodyQue(client); // must garbage collect client.display_ent since a new client will have client.display_ent == world remove(client.display_ent); }; /* ============ Set_modelindex_0 Called when client is not being rendered (eg. intermission or observer mode) ============ */ void(entity client) Set_modelindex_0 = { client.display_ent.modelindex = 0; }; /* ============ Set_modelindex_player Called after client has been given a player model. (client.modelindex must have been set) ============ */ void(entity client) Set_modelindex_player = { if (client.display_ent.modelindex) { // client.display_ent was being rendered client.display_ent.modelindex = 0; // don't render client.display_ent // switch to client viewport msg_entity = client; WriteByte(MSG_ONE, SVC_SETVIEWPORT); WriteEntity(MSG_ONE, client); } }; /* ============ Set_modelindex_non_player Called after client has been given a non-player model. (client.modelindex must have been set) ============ */ void(entity client) Set_modelindex_non_player = { if (client.display_ent.modelindex != client.modelindex) { // client.modelindex is the desired non-player model which client.display_ent will be rendering client.display_ent.modelindex = client.modelindex; // render client.display_ent // switch to client.display_ent viewport msg_entity = client; WriteByte(MSG_ONE, SVC_SETVIEWPORT); WriteEntity(MSG_ONE, client.display_ent); } client.modelindex = 0; // don't render client }; Now you have to inform the compiler about the new file by putting the following line after "Defs.qc" in "Progs.src". "Progs.src" skinfix.qc // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field Some new variables will be needed to be defined in "Defs.qc". "Defs.src" // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field start // // skinfix.qc // // protocol bytes float SVC_SETVIEWPORT = 5; float MSG_UPDATECOLORS = 17; entity post_think; // used to call PostThink every frame (after player physics), // and is the first entity after the client entities entity bodyque_head; // the next bodyque entity to be used entity first_bodyque; // the start of the bodyque linked list entity bodyque_client; // used to find a spare client entity to be used as a body float score_pos_; // the entity index of bodyque_client .entity display_ent; // the alternative display entity of a client or body .float score_pos; // the entity index .string classname_; // used to avoid conflict with .classname, but not strictly necessary // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field end After this you have to add the new functionality to the code: * Whenever a player entity is made blank include a call to "Set_modelindex_0()" after it * Whenever a player entity is set to a player model include a call to "Set_modelindex_player()" after it * Whenever a player entity is set to a non-player model include a call to "Set_modelindex_non_player()" after it * At the end of "ClientConnect()" in "Client.qc" include "SkinFixConnect(self);" and "AssignBodies();" * At the end of "ClientDisconnect()" in "Client.qc" include "SkinFixDisConnect(self);" * Comment out "InitBodyQue()" and "CopyToBodyQue()" in "World.qc" "Client.qc" void() execute_changelevel = { ... while (other != world) { ... other.modelindex = 0; Set_modelindex_0(other); // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field setorigin (other, pos.origin); other = find (other, classname, "player"); } WriteByte (MSG_ALL, SVC_INTERMISSION); }; ... void() ClientKill = { ... self.modelindex = modelindex_player; Set_modelindex_player(self); // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field ... }; ... void() CheckPowerups = { ... // invisibility if (self.invisible_finished) { ... // use the eyes self.frame = 0; self.modelindex = modelindex_eyes; Set_modelindex_non_player(self); // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field } else { // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field self.modelindex = modelindex_player; // don't use eyes // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field start Set_modelindex_player(self); } // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field end // invincibility ... }; ... void() ClientConnect = { ... // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field start SkinFixConnect(self); AssignBodies(); // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field end }; ... void() ClientDisconnect = { ... SkinFixDisConnect(self); // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field }; "Player.qc" void(string gibname, float dm) ThrowHead = { setmodel (self, gibname); // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field start if (self.classname == "player") Set_modelindex_non_player(self); // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field end ... }; ... void() PlayerDie = { ... self.modelindex = modelindex_player; // don't use eyes Set_modelindex_player(self); // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field ... }; "World.qc" // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field start /* void() InitBodyQue = { ... }; // make a body que entry for the given ent so the ent can be // respawned elsewhere void(entity ent) CopyToBodyQue = { ... }; */ // 1998-06-21 Invalid skin and GLQuake body color workaround by Robert Field end |
Page 1 of 1 | All times are UTC - 5 hours [ DST ] |
Powered by phpBB® Forum Software © phpBB Group https://www.phpbb.com/ |