Note: This is an old page that has been updated/formatted, you can view the original here.

Rotating Doors: 2nd Edition

Author: Heinz Högel

This is the second edition of the Rotating Doors Tutorial, Part I. It was revised to support MotS levels too and contains some improved control scripts.

Without doors every level would be extremely boring. They add a good amount of uncertainty  to a Jedi Knight's playground (who knows what is waiting behind them). But their appearance lacks a specific element of realism: they're always sliding up or aside and don't swing open like real-life doors. Well, in a high-tech environment like a spaceship or a station that makes sense, it's just the appropriate design for the Star Wars universe. But what about normal houses and low-tech areas? Ask an engineer or a carpenter how easy a swinging door is to be made compared with those sliding closings. No need for hollow spaces behind the walls, just a plain opening with a door in it.

If you look around in the original JK levels you won't find any exemplars of these real-life type doors. At first glance the reason for it seems to be: the JK engine can only move things along a linear path without changing it's orientation. But wait, there are definitely a lot of rotating things (ships, actors, fans etc.) — so how is that done? The answer depends on the type of thing. I.e. a fan has a rotational velocity (and according flags) given to it in it's template and it rotates constantly (stopping a fan by damaging it is done by replacing the rotating fan with a static one). An actor or ship is animated by so-called keyframes (.KEY files), that are much to difficult to make for most of us amateur level designers (at least until a 3DO editor appears with animating features for KEY files).

However, there are rare examples of rotating doors in the original game: level 9 (the ramps opening to Fuel Station's big pipes area, the giant semicircle doors in the pipe area itself and some concave doors in the inner pipe walls) and 10 (the large cargo doors in the ship's exterior area). Investigating more deeply into these levels you will find the cog scripts used to control these special doors: 09_tilting_floor.cog*, **09_pipedoorxx.cog (xx* stands for variations) and 10_cargodoors.cog. Besides the level specific extra code these scripts are very similar to normal door scripts. Their "secret" is the use of the function RotatePivot() instead of the usual MoveToFrame(). Because this function isn't documented in the Jedi Knight Unofficial Specs (version 0.3) yet, I did this little tutorial to give you an idea on how to use it for swinging doors.

Swinging doors in JK Basic levels

What I have in mind is a double wing door swinging open to both sides of the door frame. I assume that you're pretty experienced in making standard doors, so I will only discuss steps in detail that are different or new for rotating doors. The following descriptions will work for JK Basic levels only - rotating doors in MotS levels need special care, so there is an extra section for them at the and of this tutorial (but please read on, 'cause I won't repeat all the basics).

First, we need a room with an adjoined corridor. The corridor should be twice as wide as a single door wing and exactly as high. Also, you can cleave a sector at the entrance to the corridor that is as deep as the door wings are (this is not a must with the cog script I will present later). And of course the corridor can be a very small sector just to connect two separate rooms.

Next, we insert the door wings themselves into the beginning of the corridor. I used 3x3door_2a and 3x3door_2b for this example (each wing: height 3 m, width 1.5 m,  depth 0.25 m).

As usual with normal doors each door thing needs four additional entries in the thing editor: THINGFLAGS (0x400448), NUMFRAMES (2) and two FRAME entries. The value of the first frame should be the position of the door itself (simply click on the "Copy as Frame" button and paste it (Ctrl+V) as the value of the first FRAME entry.

The second frame makes the difference: It's not a positional frame like those used with normal doors (that is: it's not the end position of the door). Instead it's something the JK programmers call a "rotational axis frame" (according to some rare comments in JK cog scripts). The X, Y and Z values of such a frame define a center point (pivot) around which the door (or other thing) shall rotate. The values of PCH, YAW and ROL define the amount (angle) and direction of rotation.

You can define the second frame this way: Insert temporarily a (ghost) thing and move it to the position of a (nonexistent) hinge of the door. The Z coordinate doesn't matter here, so you can do this all in top view. Then copy the coordinates of that auxiliary thing to the second frame of the door (again with the the "Copy as Frame" button and Ctrl+V). Set the PCH and ROL parts of that frame (first and third value behind the colon) to 0.00000, if they aren't already. Insert as YAW value (second part behind the colon) the opening angle (that is: how far the door will swing open) - angles between 90.00000 and 180.00000 degrees would be appropriate for a realistic door (depends on the architectural details around that door). In top view a positive angle goes counter-clockwise. Of course, for a two wing door you have to go through that procedure twice.

Finally you have to link a control cog script to the doors. Unfortunately you can't use one of the predefined scripts from the game, because there is no standard script for swinging doors like 00_door.cog for sliding doors. Therefore I present kind of a standard script for our purpose here. It's not as sophisticated as the JK scripts tend to be (no special treatment of the door sector for example) but basically it's doing the job it was crafted for: h0_swingdoors5.cog.

Place it in your level directory and add it to the placed cogs list. Six parameters have to be filled in:

Door1, Door2 The thing numbers of the two door wings.
Switch0 The surface ID of a switch I added to a wall near the doors. By activating this switch you can open or close the doors. The switch is displayed in activated state as long as the doors are moving.
Switch1 Same function as Switch0. Not used in this example, therefore set to —1.
Time This time (in seconds) determines the duration of open/close actions.
AutoCloseDelay Normally you have to activate one of the switches once more to close the doors again after they were opened. Setting AutoCloseDelay to a positive value causes the doors to be closed automatically after the specified time (in seconds).
TouchOpening You can also open/close the doors by "touching" them (hitting the spacebar), if this parameter is set to 1. To disable this feature just set it to 0.

That's all, folks! Save the level and start it in JK. Activate the switch or just "touch" the doors and voilá — they swing open.

Heard the noise? It comes from the same source as with normal doors, and that is the SND file specified in the template of the doors. In this particular case it's med_door.snd linked to the 3x3door_2a thing:

# Foley description for medium doors
startmove df_door2-1.wav  0x0     0.1  1.0
stopmove  df_door2-3.wav  0x0     0.1  1.0
moving    df_door2-2.wav  0x0001  0.1  1.0   0.5

The flag value 0x0001 in the "moving" action class causes the sound to be repeated until the move stops. If you would like to hear different sounds or even none of them at all, you can add a template to the JED master list, based on the door you've chosen and overloading the soundclass adjective with the name of a self supplied SND file that contains the sounds you want. Done in this way you can use a SND file without the action classes startmove, stopmove and moving to make your door silent. Also notice that, in our little example, only the 3x3door_2a thing has a SND file associated. The second door wing is "silent by nature" — otherwise every sound would be played twice!

One more annotation: Rotating doors (or similar things) ignore the sector geometry around them. Therefore you don't have to worry about little edges or ledges hindering the door from moving along the planned path. But look out for other things with move class "physics" (i.e. actors of all kind, displaceable crates, debris etc.) - they will block the rotation! Doors with vertical rotation axes will push away such obstacles in many cases, however there is no guarantee for that!

You find the described door example as a (very small) demonstration level here (containing both JK and MotS versions):  rotdoor1.zip

The technical aspect

For those of you who want to take advantage of the RotatePivot() function in their own cog scripts, here is the formal syntax of that verb (in JK, not in MotS) in the Jedi Knight Unofficial Specs format:

RotatePivot() Control 

Rotates the thing specified by RotThing in a time (in seconds) given in RotTime. The center of rotation is specified by the X/Y/Z part of the frame with number RotFrame of the thing. The PCH/YAW/ROL part of that frame defines direction and angle of rotation. A negative value in RotTime inverses the direction of rotation. 

Use: RotatePivot(thing RotThing, int RotFrame, flex RotTime); 

The MotS Disaster (and a solution for it)

Soon after the first edition of this tutorial was published, I received some alarming feedback from JK designers who tried to implement swinging doors in their new MotS levels: It simply didn't work! My naive thinking that all cog verbs in MotS would function the same way than in JK prevented me from testing it - directly leading to that disaster. So here are the facts on the irritating behavior of RotatePivot() in MotS:

First it sets the thing (that is: it's center) to the position given in RotFrame (similar to the function JumpToFrame()), ignoring the PCH/YAW/ROL part of RotFrame and setting the orientation simply to 0/0/0. Then it rotates the relocated thing around it's center by the angle(s) given in the PCH/YAW/ROL part of RotFrame. This behavior doesn't seem to fullfill any practical purpose (the verb isn't used in any MotS level!). It's most probably an ordinary program bug (blame LEC for that).

But in spite of that faulty RotatePivot() function there is a way to build rotating doors in MotS. However, you have to take an approach a little different to the one described above. First of all, the hinge thing is no longer only an auxiliary object for calculating the door frames. Every door needs a separate hinge thing (but still a ghost) permanently located at the actual hinge position. Like the doors in JK Basic the hinge needs certain attributes: THINGFLAGS (0x400448), NUMFRAMES (2) and two FRAMEs. The first frame must be set to the initial position of the hinge (PCH/YAW/ROL all zero!!!). The second one alike, but with YAW set to the opening angle of the door. The door itself needs only one FRAME set to the start position of it.

Prepared this way you can use the cog script h0_swingdoors4.cog similar to the one for JK Basic levels. Of course, to make this possible, there is a rather complicated workaround for the RotatePivot() fault in the code: It attaches the door to the hinge (this is done first in the startup section), rotates the hinge (swinging the door with it in the correct way), detaches the two things, sets the hinge thing back to its initial orientation and reattaches door and hinge for the next rotation.

So the hinge rotates and not the door itself. Remember what I wrote about the sounds of the doors? Well, to make the swinging doors in MotS sound like their sisters in JK another trick is needed: You have to provide new ghost type things with associated SND files (taken from typical door things) and use one of them for a hinge. Simply add the following lines to the MOTS.TPL file in the JEDDATA folder:

# DESC:
# BBOX: 0 0 0 0 0 0
ghost_hinge_sm   none   orient=(0/0/0) type=ghost move=path soundclass=sm_door.snd
# DESC:
# BBOX: 0 0 0 0 0 0
ghost_hinge_med  none   orient=(0/0/0) type=ghost move=path soundclass=med_door.snd
# DESC:
# BBOX: 0 0 0 0 0 0
ghost_hinge_lg   none   orient=(0/0/0) type=ghost move=path soundclass=lg_door.snd

The very indirect way to rotate doors in MotS produces another little disadvantage: If you (the walkplayer) are pushing a door while it's in the process of opening, you can dislocate it a little (try it)! As a small solution to that problem the script relocates each door to its initial location after it's closed again (that's why the frame for the door itself is needed). Maybe one of the readers of this tutorial will find a better solution - let the Massassi Temple know!

Part II of the Rotating Doors Tutorial (if ever published) will feature some "iris doors", build of wedge style segments (6 or 8), which are opened by rotating them back into the wall.