|
Posted by metlslime on 2007/08/08 04:57:56 |
This is a counterpart to the "Mapping Help" thread. If you need help with QuakeC coding, or questions about how to do some engine modification, this is the place for you! We've got a few coders here on the forum and hopefully someone knows the answer. |
|
|
Alternative Universe
#1279 posted by sock on 2014/01/25 14:44:12
Could be worth looking at heirarchical pathfinding, where you have seperate graphs at different scales.
This is essentially what I am doing with my node network except I am using a terminology which is probably misleading. I essentially create a series of large brush entities and when the client starts, I store their volume bounds, reduce them down to a point entity and make sure they are non interactive with the world.
I generally create a loop of nodes in a room linking all the doorways. I then place special volume test entities on top of the nodes and link them together. When an AI arrives at a node it does a simple bounds/volume test on the final destination and if correct goes through the doorway on to the next set of nodes.
It is really easy to cut up a level into 5-10 volumes and being as they are visual in the editor it is easy to place them as a LD. I even added and/or/not logic to the volumes so AI can test several volumes together for a much better test of destination.
Once you've got a handle on things you start to make ai changes that work in tandem with what's in the original game.
The problem is the original game AI design is terrible, they essential run in a straight line bouncing off architecture in order to find their way. It is extremely easy to create architecture that breaks this and essentially the player is just shooting fish in a barrel as the AI run endlessly trapped on ledges or round in circles in lower areas of rooms.
Most Level Designers for Quake compensate for this AI flaw by using range/flying units which evens out the playing field for the monsters.
Cool Stuff
#1280 posted by Kinn on 2014/01/25 16:00:28
I did some last-ditch bodges in my (released) quake maps where I identified some areas in the game where monsters got stuck, like in spiral stairwells and stuff, and added some path corners leading out of there and logic that said things like "if player is above you follow path forwards, else follow path backwards".
Stepping back for a bit, my opinions on pathfinding in Quake have been a bit back and forth over the years. I've always leaned towards just leaving the AI alone. I think one of the things with Quake is that monsters are so aggressive that it actually works to the game's advantage that they are a bit shit at chasing the player. Typically when you have an encounter, you want the option to back out of it and not have the monster chomping at your heels the whole time. Sometimes they chase you, sometimes they don't, and I like that.
#1281 posted by sock on 2014/01/25 16:09:53
I've always leaned towards just leaving the AI alone. I think one of the things with Quake is that monsters are so aggressive that it actually works to the game's advantage that they are a bit shit at chasing the player
My mod needed the AI to hunt the player down, it would have been a very poor stealth game if the AI could not even exit a room or get back to what they were doing before. I can see the attraction of Quake AI being left alone, they are easy to break, but when a monster does something (eg monster_jump) surprising like hunting you down, then it gets more exciting! (well for me)
Same
#1282 posted by ijed on 2014/01/25 19:23:07
I like them to do cleverer stuff, although it's tricky to do it without also giving them super powers.
Like the aiming ogres that added speed to their grenades in order to hit the player.
#1283 posted by necros on 2014/01/25 20:25:44
The problem I found with implementing AI changes is you can end up replacing so much of the underlying QC that it is often difficult to integrate into existing progs.
Yes, this is the really unfortunate part. I've been trying to write a simple tutorial where I could provide the main A* component and then someone could just go in and hook it into their mod, but there are so many little bits that have to be added and so much that is flat out replaced that it's not simple at all to write the tutorial. :\
Sock: Your method is very cool, but I find it is really fiddly with adding in the nodes because you have to tweak it for each area. The results are better but it's more work and I am super lazy: I prefer just throwing down nodes and letting the code sort itself out!
Kinn: I have considered doing the coarse graph -> fine graph approach. For some reason, using a large brush and auto-linking them the way Sock is doing never occurred to me and I always assumed I'd have to do a lot of manual linking which is why I never bothered... This is worth some more thought, I think.
Also, yes I have some utility functions that make linked lists, queues and stacks out of entities. There is a queue of entities that tracks which monster is next to get a path generated, for example.
If you're using fitzquake or a variant, it's less of a problem since you have up to 32000(?) entities to play with and since they have no models attached to them, they don't actually cause overflows or anything.
Containification
#1284 posted by Preach on 2014/01/26 00:23:26
Also, yes I have some utility functions that make linked lists, queues and stacks out of entities.
This is something I was also considering for a series of tutorials - I have a nice way to make generic "container" functions using a trick with field pointers. Maybe I'll start on that next...
#1285 posted by Spike on 2014/01/26 03:14:59
If you want to do routing properly, you really ought to do it on another thread - that means some engine extension to do it.
That said, doing it in the engine has certain issues, including not knowing what a teleporter is.
#1286 posted by necros on 2014/01/26 03:35:52
Teleporters aren't too bad. If your system allows for manual linking, you simply link nodes between both sides and sit them inside the teleport trigger.
Doing it in the engine though, that'd be pretty nice, but if you're going that far, you may as well get compiler support and make AAS files. I was really impressed with the D3 AAS generation.
Less Is More
#1287 posted by sock on 2014/01/26 13:58:06
Your method is very cool, but I find it is really fiddly with adding in the nodes because you have to tweak it for each area. The results are better but it's more work and I am super lazy: I prefer just throwing down nodes and letting the code sort itself out!
Yeah that is the downside to manual node layouts, but it is not really that much work because you can added/updated the nodes using -onlyent compiles. I also found that less nodes often produces better results than trying to saturate an area with complex setups.
Here is the numbers for my two test maps:
S1M1 - 123 Nodes, 61 Vol Nodes, 23 Volumes
S1M2 - 129 Nodes, 67 Vol Nodes, 15 Volumes
The volume nodes are linked to the nodes, they are the high pass route choices. The volumes are brushes defining zones to the map for nodes.
I am still a firm believer that manually designed routes produce more robust results and with the right debug options you can easily find problems and fix them. Which is why I spent so long creating them. You can drop down start and destination markers, work out which nodes to go to first (pretend you are the AI) and then follow the path through the node network visually.
If you want to do routing properly, you really ought to do it on another thread - that means some engine extension to do it.
That said, doing it in the engine has certain issues, including not knowing what a teleporter is.
That would indeed be awesome but based on my experience of Quake engine devs there would probably be 3 different formats not compatible with each other and LDs wondering which format to work with! ;)
The hardest part of node networks for me is lift/platforms because the AI has to wait for platforms, get on them correctly, wait on the platform and then get off. Pausing AI and switching them between stationary and mobile is awkward with the current QC structure.
Cannot Agree More...
"I am still a firm believer that manually designed routes produce more robust results"
Coming from a UT mapping background originally I can honestly say once you got a nice path network going and you followed a bot that could navigate your map properly it was a great feeling of satisfaction.
I envy what Polge did for UT, it's a shame that he didn't flesh out his Reaper Bots for quake a little more. :)
Is There
#1289 posted by ijed on 2014/01/28 22:49:20
A maximum number of entities that can be searched through with the find function?
I'm using it to search through randomiser entities and it seems to be provoking a crash now, as the map and number of entities has grown in size.
#1290 posted by necros on 2014/01/28 23:08:05
it's possibly you're hitting the 100,000(?) operations per frame limit.
In a given frame, you can do 100000 operations before the engine thinks you're stuck in an infinite loop. This includes all operations done before you hit your loop and all the operations done after your loop, so you can get through your loop fine only to crash on subsequent operations done after.
the solution to this is to defer the search after n loops to the next frame by saving the last entity and starting from there instead of world.
Sounds About Right
#1291 posted by ijed on 2014/01/29 00:01:06
Just talking with Sock about it.
I threw in some delays and it certainly seems to be the case. The size of the map is basically breaking this limit.
What I'm doing is randomising monster position, and the code I'm using is searching the entire entity list to do so.
If I chained the entities it'd work much better, but that's a massive amount of work.
Not using randomisation would have the benefit of more precise monster placement and not having to rewrite the code and map.
Just delete a load of entities from the map.
Thanks, seems like the way ahead is clear!
I could rewrite, but it'd take a lot of time. In future projects I can take this limitation into mind and use stuff which has nextthink = time + random(); or similar.
So
#1292 posted by ijed on 2014/01/29 15:11:48
I'm removing all the randomiser entities and replacing the monsters traditionally.
This'll give me a playable level.
I've got in mind a randomiser system that would be supported by the single thread engines that Quake uses.
Basically it flagging monsters as RANDOMISED (additional to the SPAWN_SILENT, SPAWNED and SPAWN_ANGRY flags).
Each monster would have keys to denote their randomiser grouping. These keys would be rname, rnext and pool.
rname and rnext would copy the door code to produce chain grouping. This would make things much lighter when using the find function. Erroneously configured chains would have a failsafe to make sure they'd always be a loop.
All monsters in the same group would randomise their positions between the others in the group, any monster ending up in another position would also inherit the target of that position (to point at the relevant path_corner or trigger_counter) if a monster didn't have a target then this would clean any target held by the arriving monster. The difficulty flags and targetnames would remain unchanged.
Monsters with pool set to 1 would never add the position etc. to the randomisation group, but could take the place of other monsters. Possibly there'd be other functionality for pool, which is why it should be a key and not a flag.
Basically this means that the monsters not set to pool 1 define the positions used for spawning and how many monsters will be spawned.
Monsters would only be randomised when spawned, so quick loading after dying would re-randomise the monster positions as they would be retriggered.
This would produce a system optimal for the engine and simple for the mapper, even though it involves a flag and three separate keys.
Probably won't do it in this project though. Maybe next time.
Pool 2
#1293 posted by ijed on 2014/01/29 15:33:17
This would make a monster not spawn. The utility would be placing three shamblers for example, but only one appears.
Uh
#1294 posted by ijed on 2014/01/29 17:12:46
I mean, pool 2 would make a monster part of the group, but not increase the amount of spawns for that group.
QuakeC IDE
#1295 posted by Kinn on 2014/02/08 00:27:06
So apparently you can use "Code::Blocks" as an IDE for QuakeC, with autocomplete and all that shizzle.
Anyone use this? I followed the instructions here: http://ouns.nexuizninjaz.com/dev:programming_introduction#working_with_the_code
But I can't get the pissing thing to work properly - my symbols tab is empty and it can't find any of my functions or do autocomplete or any of the things which would make it better than, you know, notepad.
Has anyone else got it working for QuakeC?
Hmm
#1296 posted by Preach on 2014/02/08 01:07:41
Gave it a try, but I don't know the first thing to do with codeblocks, so didn't get very far.
Did notice that the lexer_qc.xml file has a bunch of typos in the keywords though, so it's not going to be highlighting .vector, continue or default any time soon...
Yeah
#1297 posted by Kinn on 2014/02/08 01:52:56
I noticed all those typos too. Doesn't fill me with much confidence to be honest.
Dyslexer_qc.xml
#1298 posted by Kinn on 2014/02/08 01:55:59
more like lol
Some C++ Help?
#1299 posted by necros on 2014/02/08 02:58:52
Coming from Java, this C++ stuff is confusing to me...
I have this class which is meant to be an Interface:
class PhysObject
{
public:
virtual void evaluatePhysics() = 0;
protected:
Vec2 velocity; //physicalized objects have velocity
PhysObject() :
velocity(Vec2())
{}
};
Then I have this other class that implements PhysObject:
class Particle : public PhysObject
{
public:
Particle(Vec2 pos) :
position(pos),
size(Vec2(4, 4)),
releaseTime(Game::Time() + 1),
PhysObject()
{}
void evaluatePhysics()
{
this->position = this->position + (this->velocity * Game::FrameTime());
}
...
};
But when I call this->evaluatePhysics() on the Particle object, it crashes with access violation 0xCDCDCDCD which the internet tells me is caused by dereferencing a null pointer. So it looks like my evaluatePhysics() implementation in Particle never actually took, and it's still trying to call the Interface's evaluatePhysics() = 0 pointer.
Any hints?
#1300 posted by ericw on 2014/02/08 03:45:43
I think you just need to add 'virtual' before 'void evaluatePhysics()' in Particle.
Also, sounds like there's a new c++11 feature where you can stick the "override" keyword after the function args in Particle, like this:
virtual void evaluatePhysics() override
{
which will make the compiler check that this evaluatePhysics() is actually overriding something from the parent class.
#1301 posted by necros on 2014/02/08 05:23:57
no dice, neither from adding virtual or override. :(
From what I've read, once you put virtual on a method, everything from that point forward will always be virtual, ie: behave as expected of a polymorphic object and properly call child class methods if held in a base class container.
#1302 posted by ericw on 2014/02/08 06:18:30
ah, you're right - sorry for the bad advice! the setup of evaluatePhysics() looks fine to me then.
all I can think of is generic debugging advice.. like, stick a:
printf("inside Particle::evaluatePhysics()\n");
at the first line of evaluatePhysics(). double check the particle object you're calling ->evaluatePhysics() on is not NULL. is it possible Game::FrameTime() is dereferencing a null pointer?
it should be possible to get a debugger to break when a null pointer dereference happens, so you can look at the call stack and ideally see exactly where the problem is. which IDE/environment are you using?
Test Case
#1303 posted by Preach on 2014/02/08 10:04:30
Here's my very reduced test case
http://ideone.com/u1scWv
I actually put an implementation in my base class but it's easy to check that it works when it's pure virtual. Anyway, it's all got the same syntax and setup as your code above, so there's a detail wrong somewhere.
Have you tried putting a dummy evaluatePhysics function in PhysObject for debugging purposes. Just a quick one-liner that maybe logs a message. It would be a quick way to test if it is the "pure virtual" function which is causing the null pointer dereference.
|
|
You must be logged in to post in this thread.
|
Website copyright © 2002-2025 John Fitzgibbons. All posts are copyright their respective authors.
|
|