// "Build Engine & Tools" Copyright (c) 1993-1997 Ken Silverman // Ken Silverman's official web site: "http://www.advsys.net/ken" // See the included license file "BUILDLIC.TXT" for license info. Build information to get you started: The first half of this file explains the .ART, .MAP, and PALETTE.DAT formats. The second half has documentation about every BUILD engine function, what it does, and what the parameters are. -Ken S. ------------------------------------------------------------------------------ Documentation on Ken's .ART file format by Ken Silverman I am documenting my ART format to allow you to program your own custom art utilites if you so desire. I am still planning on writing the script system. All art files must have xxxxx###.ART. When loading an art file you should keep trying to open new xxxxx###'s, incrementing the number, until an art file is not found. 1. long artversion; The first 4 bytes in the art format are the version number. The current current art version is now 1. If artversion is not 1 then either it's the wrong art version or something is wrong. 2. long numtiles; Numtiles is not really used anymore. I wouldn't trust it. Actually when I originally planning art version 1 many months ago, I thought I would need this variable, but it turned it is was unnecessary. To get the number of tiles, you should search all art files, and check the localtilestart and localtileend values for each file. 3. long localtilestart; Localtilestart is the tile number of the first tile in this art file. 4. long localtileend; Localtileend is the tile number of the last tile in this art file. Note: Localtileend CAN be higher than the last used slot in an art file. Example: If you chose 256 tiles per art file: TILES000.ART -> localtilestart = 0, localtileend = 255 TILES001.ART -> localtilestart = 256, localtileend = 511 TILES002.ART -> localtilestart = 512, localtileend = 767 TILES003.ART -> localtilestart = 768, localtileend = 1023 5. short tilesizx[localtileend-localtilestart+1]; This is an array of shorts of all the x dimensions of the tiles in this art file. If you chose 256 tiles per art file then [localtileend-localtilestart+1] should equal 256. 6. short tilesizy[localtileend-localtilestart+1]; This is an array of shorts of all the y dimensions. 7. long picanm[localtileend-localtilestart+1]; This array of longs stores a few attributes for each tile that you can set inside EDITART. You probably won't be touching this array, but I'll document it anyway. Bit: |31 24|23 16|15 8|7 0| ----------------------------------------------------------------- | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | ----------------------------------------------------------------- | Anim. | Signed char | Signed char | | Animate | | Speed | Y-center | X-center | | number | --------| offset | offset | |------------ --------------------------------| ------------ | Animate type:| | 00 - NoAnm | | 01 - Oscil | | 10 - AnmFd | | 11 - AnmBk | ---------------- You probably recognize these: Animate speed - EDITART key: 'A', + and - to adjust Signed char x&y offset - EDITART key: '`', Arrows to adjust Animate number&type - EDITART key: +/- on keypad 8. After the picanm's, the rest of the file is straight-forward rectangular art data. You must go through the tilesizx and tilesizy arrays to find where the artwork is actually stored in this file. Note: The tiles are stored in the opposite coordinate system than the screen memory is stored. Example on a 4*4 file: Offsets: ----------------- | 0 | 4 | 8 |12 | ----------------- | 1 | 5 | 9 |13 | ----------------- | 2 | 6 |10 |14 | ----------------- | 3 | 7 |11 |15 | ----------------- ---------------------------------------------------------------------------- If you wish to display the artwork, you will also need to load your palette. To load the palette, simply read the first 768 bytes of your palette.dat and write it directly to the video card - like this: Example: long i, fil; fil = open("palette.dat",O_BINARY|O_RDWR,S_IREAD); read(fil,&palette[0],768); close(fil); outp(0x3c8,0); for(i=0;i<768;i++) outp(0x3c9,palette[i]); ------------------------------------------------------------------------------ Packet format for DUKE3D (specifically for network mode 1, n(n-1) mode): Example bunch of packets: A B C D E F G H I J K L M N... O --------------------------------------------------------------------- d9 00 d9 11 01 00 - - - - - - - - - - 4f 16 31 da 00 da 11 01 00 - - - - - - - - - - b2 b7 9d db 00 db 11 01 00 - - - - - - - - - - b1 24 62 dc 00 dc 11 01 00 - - - - - - - - - - ca 1d 58 dd 00 dd 11 01 00 - - - - - - - - - - a9 94 14 de 00 de 11 01 05 00 00 - - 03 00 - - - - c5 50 b9 df 00 df 11 01 0f a1 ff fe 09 00 00 26 - - - e2 88 6f e0 00 e0 11 01 04 - - - - fd ff - - - - 77 51 d7 e1 00 e1 11 01 03 1f 00 ff 09 - - - - - - ac 14 b7 e2 00 e2 11 01 0b 9c 00 fb 09 - - 24 - - - f8 6c 22 GAME sends fields D-N MMULTI adds fields A-C and O for error correction. A: Packet count sending modulo 256 B: Error state. Usually 0. To request a resend, bit 0 is set. In order to catch up on networks, sending many packets is bad, so 2 packets are sent in 1 IPX packet. To send 2 packets in 1 packet, bit 1 is set. In special cases, this value may be different. C: Packet count receiving modulo 256 D: Message header byte. These are all the possible values currently. You are probably only interested in case 17. Note that fields E-N apply to case 17 only. 0: send movement info from master to slave (network mode 0 only) 1: send movement info from slave to master (network mode 0 only) 4: user-typed messages 5: Re-start level with given parameters 6: Send player name 7: Play Remote Ridicule sound 8: Re-start level with given parameters for a user map 17: send movement info to everybody else (network mode 1 only) 250: Wait for Everybody (Don't start until everybody's done loading) 255: Player quit to DOS E: Timing byte used to calculate lag time. This prevents the 2 computer's timers from drifting apart. F: Bits field byte. Fields G-M are sent only when certain bits in this byte are set. G: X momentum update (2 bytes). Sent only if ((F&1) != 0) H: Y momentum update (2 bytes). Sent only if ((F&2) != 0) I: Angle momentum update (2 bytes). Sent only if ((F&4) != 0) J: The states of 8 different keys (1 byte). Sent only if ((F&8) != 0) K: The states of 8 different keys (1 byte). Sent only if ((F&16) != 0) L: The states of 8 different keys (1 byte). Sent only if ((F&32) != 0) M: The states of 8 different keys (1 byte). Sent only if ((F&64) != 0) N: Sync checking byte. Useful for debugging programming errors. Can be a variable number of bytes. Actual number of sync checking bytes is calculated by length of the whole packet minus the rest of the bytes sent. O: CRC-16 ------------------------------------------------------------------------------ | @@@@@@@@@@@ @@@ @@@ @@@@@@@@@ @@@ @@@@@@@@@ | | @@@@@@@@@@@@@ @@@ @@@ @@@@@@@@@ @@@ @@@@@@@@@@@ | | @@@ @@@@ @@@ @@@ @@@ @@@ @@@ @@@@@ | | @@@ @@@ @@@ @@@ @@@ @@@ @@@ @@@@ | | @@@ @@@@ @@@ @@@ @@@ @@@ @@@ @@@ | | @@@@@@@@@@@@@ @@@ @@@ @@@ @@@ @@@ @@@ | | @@@@@@@@@@@@@ @@@ @@@ @@@ @@@ @@@ @@@ | | @@@ @@@@ @@@ @@@ @@@ @@@ @@@ @@@ | | @@@ @@@ @@@ @@@ @@@ @@@ @@@ @@@@ | | @@@ @@@@ @@@@ @@@@ @@@ @@@ @@@ @@@@@ | | @@@@@@@@@@@@@ @@@@@@@@@@@ @@@@@@@@@ @@@@@@@@@@@@ @@@@@@@@@@@ | | @@@@@@@@@@@ @@@@@@@ @@@@@@@@@ @@@@@@@@@@@@ @@@@@@@@@ | | | | M A P F O R M A T ! | ------------------------------------------------------------------------------ Here is Ken's documentation on the COMPLETE BUILD map format: BUILD engine and editor programmed completely by Ken Silverman Here's how you should read a BUILD map file: { fil = open(???); //Load map version number (current version is 7L) read(fil,&mapversion,4); //Load starting position read(fil,posx,4); read(fil,posy,4); read(fil,posz,4); //Note: Z coordinates are all shifted up 4 read(fil,ang,2); //All angles are from 0-2047, clockwise read(fil,cursectnum,2); //Sector of starting point //Load all sectors (see sector structure described below) read(fil,&numsectors,2); read(fil,§or[0],sizeof(sectortype)*numsectors); //Load all walls (see wall structure described below) read(fil,&numwalls,2); read(fil,&wall[0],sizeof(walltype)*numwalls); //Load all sprites (see sprite structure described below) read(fil,&numsprites,2); read(fil,&sprite[0],sizeof(spritetype)*numsprites); close(fil); } ------------------------------------------------------------- | @@@@@@@ @@@@@@@ @@@@@@@ @@@@@@@@ @@@@@@@ @@@@@@@ @@@@@@@ | | @@ @@ @@ @@ @@ @@ @@ @@@ @@ | | @@@@@@@ @@@@@ @@ @@ @@ @@ @@@@@@@ @@@@@@@ | | @@ @@ @@ @@ @@ @@ @@ @@@ @@ | | @@@@@@@ @@@@@@@ @@@@@@@ @@ @@@@@@@ @@ @@ @@@@@@@ | ------------------------------------------------------------- //sizeof(sectortype) = 40 typedef struct { short wallptr, wallnum; long ceilingz, floorz; short ceilingstat, floorstat; short ceilingpicnum, ceilingheinum; signed char ceilingshade; char ceilingpal, ceilingxpanning, ceilingypanning; short floorpicnum, floorheinum; signed char floorshade; char floorpal, floorxpanning, floorypanning; char visibility, filler; short lotag, hitag, extra; } sectortype; sectortype sector[1024]; wallptr - index to first wall of sector wallnum - number of walls in sector z's - z coordinate (height) of ceiling / floor at first point of sector stat's bit 0: 1 = parallaxing, 0 = not "P" bit 1: 1 = sloped, 0 = not bit 2: 1 = swap x&y, 0 = not "F" bit 3: 1 = double smooshiness "E" bit 4: 1 = x-flip "F" bit 5: 1 = y-flip "F" bit 6: 1 = Align texture to first wall of sector "R" bits 7-15: reserved picnum's - texture index into art file heinum's - slope value (rise/run) (0-parallel to floor, 4096-45 degrees) shade's - shade offset of ceiling/floor pal's - palette lookup table number (0 - use standard colors) panning's - used to align textures or to do texture panning visibility - determines how fast an area changes shade relative to distance filler - useless byte to make structure aligned lotag, hitag, extra - These variables used by the game programmer only ----------------------------------------------- | @@ @@ @@@@@@@@ @@ @@ @@@@@@@ | | @@ @@ @@ @@ @@ @@ @@ | | @@ @@ @@ @@@@@@@@ @@ @@ @@@@@@@ | | @@ @@@@ @@ @@ @@ @@ @@ @@ | | @@@ @@@@ @@ @@ @@@@@@@ @@@@@@@ @@@@@@@ | ----------------------------------------------| //sizeof(walltype) = 32 typedef struct { long x, y; short point2, nextwall, nextsector, cstat; short picnum, overpicnum; signed char shade; char pal, xrepeat, yrepeat, xpanning, ypanning; short lotag, hitag, extra; } walltype; walltype wall[8192]; x, y: Coordinate of left side of wall, get right side from next wall's left side point2: Index to next wall on the right (always in the same sector) nextwall: Index to wall on other side of wall (-1 if there is no sector) nextsector: Index to sector on other side of wall (-1 if there is no sector) cstat: bit 0: 1 = Blocking wall (use with clipmove, getzrange) "B" bit 1: 1 = bottoms of invisible walls swapped, 0 = not "2" bit 2: 1 = align picture on bottom (for doors), 0 = top "O" bit 3: 1 = x-flipped, 0 = normal "F" bit 4: 1 = masking wall, 0 = not "M" bit 5: 1 = 1-way wall, 0 = not "1" bit 6: 1 = Blocking wall (use with hitscan / cliptype 1) "H" bit 7: 1 = Transluscence, 0 = not "T" bit 8: 1 = y-flipped, 0 = normal "F" bit 9: 1 = Transluscence reversing, 0 = normal "T" bits 10-15: reserved picnum - texture index into art file overpicnum - texture index into art file for masked walls / 1-way walls shade - shade offset of wall pal - palette lookup table number (0 - use standard colors) repeat's - used to change the size of pixels (stretch textures) pannings - used to align textures or to do texture panning lotag, hitag, extra - These variables used by the game programmer only ------------------------------------------------------------- | @@@@@@@ @@@@@@@ @@@@@@@ @@@@@@ @@@@@@@@ @@@@@@@ @@@@@@@ | | @@ @@ @@ @@ @@@ @@ @@ @@ @@ | | @@@@@@@ @@@@@@@ @@@@@@@ @@ @@ @@@@@ @@@@@@@ | | @@ @@ @@ @@ @@ @@ @@ @@ | | @@@@@@@ @@ @@ @@ @@@@@@ @@ @@@@@@@ @@@@@@@ | ------------------------------------------------------------- //sizeof(spritetype) = 44 typedef struct { long x, y, z; short cstat, picnum; signed char shade; char pal, clipdist, filler; unsigned char xrepeat, yrepeat; signed char xoffset, yoffset; short sectnum, statnum; short ang, owner, xvel, yvel, zvel; short lotag, hitag, extra; } spritetype; spritetype sprite[4096]; x, y, z - position of sprite - can be defined at center bottom or center cstat: bit 0: 1 = Blocking sprite (use with clipmove, getzrange) "B" bit 1: 1 = transluscence, 0 = normal "T" bit 2: 1 = x-flipped, 0 = normal "F" bit 3: 1 = y-flipped, 0 = normal "F" bits 5-4: 00 = FACE sprite (default) "R" 01 = WALL sprite (like masked walls) 10 = FLOOR sprite (parallel to ceilings&floors) bit 6: 1 = 1-sided sprite, 0 = normal "1" bit 7: 1 = Real centered centering, 0 = foot center "C" bit 8: 1 = Blocking sprite (use with hitscan / cliptype 1) "H" bit 9: 1 = Transluscence reversing, 0 = normal "T" bits 10-14: reserved bit 15: 1 = Invisible sprite, 0 = not invisible picnum - texture index into art file shade - shade offset of sprite pal - palette lookup table number (0 - use standard colors) clipdist - the size of the movement clipping square (face sprites only) filler - useless byte to make structure aligned repeat's - used to change the size of pixels (stretch textures) offset's - used to center the animation of sprites sectnum - current sector of sprite statnum - current status of sprite (inactive/monster/bullet, etc.) ang - angle the sprite is facing owner, xvel, yvel, zvel, lotag, hitag, extra - These variables used by the game programmer only ------------------------------------------------------------------------------ ----------------------------------------------------------------------------- | IMPORTANT ENGINE FUNCTIONS: | ----------------------------------------------------------------------------- initengine() Initializes many variables for the BUILD engine. You should call this once before any other functions of the BUILD engine are used. uninitengine(); Frees buffers. You should call this once at the end of the program before quitting to dos. loadboard(char *filename, long *posx, long *posy, long *posz, short *ang, short *cursectnum) saveboard(char *filename, long *posx, long *posy, long *posz, short *ang, short *cursectnum) Loads/saves the given board file from memory. Returns -1 if file not found. If no extension is given, .MAP will be appended to the filename. loadpics(char *filename); Loads the given artwork file into memory for the BUILD engine. Returns -1 if file not found. If no extension is given, .ART will be appended to the filename. loadtile(short tilenum) Loads a given tile number from disk into memory if it is not already in memory. This function calls allocache internally. A tile is not in the cache if (waloff[tilenum] == 0) ----------------------------------------------------------------------------- | SCREEN STATUS FUNCTIONS: | ----------------------------------------------------------------------------- setgamemode(char vidoption, long xdim, long ydim); This function sets the video mode to 320*200*256color graphics. Since BUILD supports several different modes including mode x, mode 13h, and other special modes, I don't expect you to write any graphics output functions. (Soon I have all the necessary functions) If for some reason, you use your own graphics mode, you must call this function again before using the BUILD drawing functions. vidoption can be anywhere from 0-6 xdim,ydim can be any vesa resolution if vidoption = 1 xdim,ydim must be 320*200 for any other mode. (see graphics mode selection in my setup program) setview(long x1, long y1, long x2, long y2) Sets the viewing window to a given rectangle of the screen. Example: For full screen 320*200, call like this: setview(0L,0L,319L,199L); nextpage(); This function flips to the next video page. After a screen is prepared, use this function to view the screen. ----------------------------------------------------------------------------- | DRAWING FUNCTIONS: | ----------------------------------------------------------------------------- drawrooms(long posx, long posy, long posz, short ang, long horiz, short cursectnum) This function draws the 3D screen to the current drawing page, which is not yet shown. This way, you can overwrite some things over the 3D screen such as a gun. Be sure to call the drawmasks() function soon after you call the drawrooms() function. To view the screen, use the nextpage() function. The nextpage() function should always be called sometime after each draw3dscreen() function. drawmasks(); This function draws all the sprites and masked walls to the current drawing page which is not yet shown. The reason I have the drawing split up into these 2 routines is so you can animate just the sprites that are about to be drawn instead of having to animate all the sprites on the whole board. Drawrooms() prepares these variables: spritex[], spritey[], spritepicnum[], thesprite[], and spritesortcnt. Spritesortcnt is the number of sprites about to be drawn to the page. To change the sprite's picnum, simply modify the spritepicnum array If you want to change other parts of the sprite structure, then you can use the thesprite array to get an index to the actual sprite number. clearview(long col) Clears the current video page to the given color clearallviews(long col) Clears all video pages to the given color drawmapview (long x, long y, long zoom, short ang) Draws the 2-D texturized map at the given position into the viewing window. rotatesprite (long sx, long sy, long z, short a, short picnum, signed char dashade, char dapalnum, char dastat, long cx1, long cy1, long cx2, long cy2) (sx, sy) is the center of the sprite to draw defined as screen coordinates shifted up by 16. (z) is the zoom. Normal zoom is 65536. Ex: 131072 is zoomed in 2X and 32768 is zoomed out 2X. (a) is the angle (0 is straight up) (picnum) is the tile number (dashade) is 0 normally but can be any standard shade up to 31 or 63. (dapalnum) can be from 0-255. if ((dastat&1) == 0) - no transluscence if ((dastat&1) != 0) - transluscence if ((dastat&2) == 0) - don't scale to setview's viewing window if ((dastat&2) != 0) - scale to setview's viewing window (windowx1,etc.) if ((dastat&4) == 0) - nuttin' special if ((dastat&4) != 0) - y-flip image if ((dastat&8) == 0) - clip to startumost/startdmost if ((dastat&8) != 0) - don't clip to startumost/startdmost if ((dastat&16) == 0) - use Editart center as point passed if ((dastat&16) != 0) - force point passed to be top-left corner if ((dastat&32) == 0) - nuttin' special if ((dastat&32) != 0) - use reverse transluscence if ((dastat&64) == 0) - masked drawing (check 255's) (slower) if ((dastat&64) != 0) - draw everything (don't check 255's) (faster) if ((dastat&128) == 0) - nuttin' special if ((dastat&128) != 0) - automatically draw to all video pages Note: As a special case, if both ((dastat&2) != 0) and ((dastat&8) != 0) then rotatesprite will scale to the full screen (0,0,xdim-1,ydim-1) rather than setview's viewing window. (windowx1,windowy1,etc.) This case is useful for status bars, etc. Ex: rotatesprite(160L<<16,100L<<16,65536,totalclock<<4, DEMOSIGN,2,50L,50L,270L,150L); This example will draw the DEMOSIGN tile in the center of the screen and rotate about once per second. The sprite will only get drawn inside the rectangle from (50,50) to (270,150) drawline256(long x1, long y1, long x2, long y2, char col) Draws a solid line from (x1,y1) to (x2,y2) with color (col) For this function, screen coordinates are all shifted up 16 for precision. printext256(long xpos, long ypos, short col, short backcol, char *message, char fontsize) Draws a text message to the screen. (xpos,ypos) - position of top left corner col - color of text backcol - background color, if -1, then background is transparent message - text message fontsize - 0 - 8*8 font 1 - 4*6 font ----------------------------------------------------------------------------- | MOVEMENT COLLISION FUNCTIONS: | ----------------------------------------------------------------------------- clipmove(long *x, long *y, long *z, short *sectnum, long xvect, long yvect, long walldist, long ceildist, long flordist, unsigned long cliptype) Moves any object (x, y, z) in any direction at any velocity and will make sure the object will stay a certain distance from walls (walldist) Pass the pointers of the starting position (x, y, z). Then pass the starting position's sector number as a pointer also. Also these values will be modified accordingly. Pass the direction and velocity by using a vector (xvect, yvect). If you don't fully understand these equations, please call me. xvect = velocity * cos(angle) yvect = velocity * sin(angle) Walldist tells how close the object can get to a wall. I use 128L as my default. If you increase walldist all of a sudden for a certain object, the object might leak through a wall, so don't do that! Cliptype is a mask that tells whether the object should be clipped to or not. The lower 16 bits are anded with wall[].cstat and the higher 16 bits are anded with sprite[].cstat. Clipmove can either return 0 (touched nothing) 32768+wallnum (wall first touched) 49152+spritenum (sprite first touched) pushmove (long *x, long *y, long *z, short *sectnum, long walldist, long ceildist, long flordist, unsigned long cliptype) This function makes sure a player or monster (defined by x, y, z, sectnum) is not too close to a wall. If it is, then it attempts to push it away. If after 256 tries, it is unable to push it away, it returns -1, in which case the thing should gib. getzrange(long x, long y, long z, short sectnum, long *ceilz, long *ceilhit, long *florz, long *florhit, long walldist, unsigned long cliptype) Use this in conjunction with clipmove. This function will keep the player from falling off cliffs when you're too close to the edge. This function finds the highest and lowest z coordinates that your clipping BOX can get to. It must search for all sectors (and sprites) that go into your clipping box. This method is better than using sector[cursectnum].ceilingz and sector[cursectnum].floorz because this searches the whole clipping box for objects, not just 1 point. Pass x, y, z, sector normally. Walldist can be 128. Cliptype is defined the same way as it is for clipmove. This function returns the z extents in ceilz and florz. It will return the object hit in ceilhit and florhit. Ceilhit and florhit will also be either: 16384+sector (sector first touched) or 49152+spritenum (sprite first touched) hitscan(long xstart, long ystart, long zstart, short startsectnum, long vectorx, long vectory, long vectorz, short *hitsect, short *hitwall, short *hitsprite, long *hitx, long *hity, long *hitz); Pass the starting 3D position: (xstart, ystart, zstart, startsectnum) Then pass the 3D angle to shoot (defined as a 3D vector): (vectorx, vectory, vectorz) Then set up the return values for the object hit: (hitsect, hitwall, hitsprite) and the exact 3D point where the ray hits: (hitx, hity, hitz) How to determine what was hit: * Hitsect is always equal to the sector that was hit (always >= 0). * If the ray hits a sprite then: hitsect = thesectornumber hitsprite = thespritenumber hitwall = -1 * If the ray hits a wall then: hitsect = thesectornumber hitsprite = -1 hitwall = thewallnumber * If the ray hits the ceiling of a sector then: hitsect = thesectornumber hitsprite = -1 hitwall = -1 vectorz < 0 (If vectorz < 0 then you're shooting upward which means that you couldn't have hit a floor) * If the ray hits the floor of a sector then: hitsect = thesectornumber hitsprite = -1 hitwall = -1 vectorz > 0 (If vectorz > 0 then you're shooting downard which means that you couldn't have hit a ceiling) neartag(long x, long y, long z, short sectnum, short ang, //Starting position & angle short *neartagsector, //Returns near sector if sector[].tag != 0 short *neartagwall, //Returns near wall if wall[].tag != 0 short *neartagsprite, //Returns near sprite if sprite[].tag != 0 long *neartaghitdist, //Returns actual distance to object (scale: 1024=largest grid size) long neartagrange, //Choose maximum distance to scan (scale: 1024=largest grid size) char tagsearch) //1-lotag only, 2-hitag only, 3-lotag&hitag Neartag works sort of like hitscan, but is optimized to scan only close objects and scan only objects with tags != 0. Neartag is perfect for the first line of your space bar code. It will tell you what door you want to open or what switch you want to flip. cansee(long x1, long y1, long z1, short sectnum1, long x2, long y2, long z2, short sectnum2); returns 0 or 1 This function determines whether or not two 3D points can "see" each other or not. All you do is pass it the coordinates of a 3D line defined by two 3D points (with their respective sectors) The function will return a 1 if the points can see each other or a 0 if there is something blocking the two points from seeing each other. This is how I determine whether a monster can see you or not. Try playing DOOM1.DAT to fully enjoy this great function! updatesector(long x, long y, §num); This function updates the sector number according to the x and y values passed to it. Be careful when you use this function with sprites because remember that the sprite's sector number should not be modified directly. If you want to update a sprite's sector, I recomment using the setsprite function described below. inside(long x, long y, short sectnum); Tests to see whether the overhead point (x, y) is inside sector (sectnum) Returns either 0 or 1, where 1 means it is inside, and 0 means it is not. clipinsidebox(long x, long y, short wallnum, long walldist) Returns TRUE only if the given line (wallnum) intersects the square with center (x,y) and radius, walldist. dragpoint(short wallnum, long newx, long newy); This function will drag a point in the exact same way a point is dragged in 2D EDIT MODE using the left mouse button. Simply pass it which wall to drag and then pass the new x and y coordinates for that point. Please use this function because if you don't and try to drag points yourself, I can guarantee that it won't work as well as mine and you will get confused. Note: Every wall of course has 2 points. When you pass a wall number to this function, you are actually passing 1 point, the left side of the wall (given that you are in the sector of that wall) Got it? ----------------------------------------------------------------------------- | MATH HELPER FUNCTIONS: | ----------------------------------------------------------------------------- krand() Random number function - returns numbers from 0-65535 ksqrt(long num) Returns the integer square root of the number. getangle(long xvect, long yvect) Gets the angle of a vector (xvect,yvect) These are 2048 possible angles starting from the right, going clockwise rotatepoint(long xpivot, long ypivot, long x, long y, short daang, long *x2, long *y2); This function is a very convenient and fast math helper function. Rotate points easily with this function without having to juggle your cosines and sines. Simply pass it: Input: 1. Pivot point (xpivot,ypivot) 2. Original point (x,y) 3. Angle to rotate (0 = nothing, 512 = 90� CW, etc.) Output: 4. Rotated point (*x2,*y2) lastwall(short point); Use this function as a reverse function of wall[].point2. In order to save memory, my walls are only on a single linked list. nextsectorneighborz(short sectnum, long thez, short topbottom, short direction) This function is used to tell where elevators should stop. It searches nearby sectors for the next closest ceilingz or floorz it should stop at. sectnum - elevator sector thez - current z to start search from topbottom - search ceilingz's/floorz's only direction - search upwards/downwards getceilzofslope(short sectnum, long x, long y) getflorzofslope(short sectnum, long x, long y) getzsofslope(short sectnum, long x, long y, long *ceilz, long *florz) These 3 functions get the height of a ceiling and/or floor in a sector at any (x,y) location. Use getzsofslope only if you need both the ceiling and floor. alignceilslope(short sectnum, long x, long y, long z) alignflorslope(short sectnum, long x, long y, long z) Given a sector and assuming it's first wall is the pivot wall of the slope, this function makes the slope pass through the x,y,z point. One use of this function is used for sin-wave floors. ----------------------------------------------------------------------------- | SPRITE FUNCTIONS: | ----------------------------------------------------------------------------- insertsprite(short sectnum, short statnum); //returns (short)spritenum; Whenever you insert a sprite, you must pass it the sector number, and a status number (statnum). The status number can be any number from 0 to MAXSTATUS-1. Insertsprite works like a memory allocation function and returns the sprite number. deletesprite(short spritenum); Deletes the sprite. changespritesect(short spritenum, short newsectnum); Changes the sector of sprite (spritenum) to the newsector (newsectnum). This function may become internal to the engine in the movesprite function. But this function is necessary since all the sectors have their own doubly-linked lists of sprites. changespritestat(short spritenum, short newstatnum); Changes the status of sprite (spritenum) to status (newstatus). Newstatus can be any number from 0 to MAXSTATUS-1. You can use this function to put a monster on a list of active sprites when it first sees you. setsprite(short spritenum, long newx, long newy, long newz); This function simply sets the sprite's position to a specified coordinate (newx, newy, newz) without any checking to see whether the position is valid or not. You could directly modify the sprite[].x, sprite[].y, and sprite[].z values, but if you use my function, the sprite is guaranteed to be in the right sector. ----------------------------------------------------------------------------- | CACHE FUNCTIONS: | ----------------------------------------------------------------------------- initcache(long dacachestart, long dacachesize) First allocate a really large buffer (as large as possible), then pass off the memory bufer the initcache dacachestart: 32-bit offset in memory of start of cache dacachesize: number of bytes that were allocated for the cache to use allocache (long *bufptr, long bufsiz, char *lockptr) *bufptr = pointer to 4-byte pointer to buffer. This allows allocache to remove previously allocated things from the cache safely by setting the 4-byte pointer to 0. bufsiz = number of bytes to allocate *lockptr = pointer to locking char which tells whether the region can be removed or not. If *lockptr = 0 then the region is not locked else its locked. ----------------------------------------------------------------------------- | GROUP FILE FUNCTIONS: | ----------------------------------------------------------------------------- initgroupfile(char *filename) Tells the engine what the group file name is. You should call this before any of the following group file functions. uninitgroupfile() Frees buffers. You should call this once at the end of the program before quitting to dos. kopen4load(char *filename, char searchfirst) Open a file. First tries to open a stand alone file. Then searches for it in the group file. If searchfirst is nonzero, it will check the group file only. kread(long handle, void *buffer, long leng) klseek(long handle, long offset, long whence) kfilelength(long handle) kclose(long handle) These 4 functions simply shadow the dos file functions - they can do file I/O on the group file in addition to stand-along files. ----------------------------------------------------------------------------- | COMMUNICATIONS FUNCTIONS: | ----------------------------------------------------------------------------- Much of the following code is to keep compatibity with older network code: initmultiplayers(char damultioption, char dacomrateoption, char dapriority) The parameters are ignored - just pass 3 0's uninitmultiplayers() Does nothing sendpacket(long other, char *bufptr, long messleng) other - who to send the packet to bufptr - pointer to message to send messleng - length of message short getpacket (short *other, char *bufptr) returns the number of bytes of the packet received, 0 if no packet other - who the packet was received from bufptr - pointer to message that was received sendlogon() Does nothing sendlogoff() Sends a packet to everyone else where the first byte is 255, and the second byte is myconnectindex getoutputcirclesize() Does nothing - just a stub function, returns 0 setsocket(short newsocket) Does nothing flushpackets() Clears all packet buffers genericmultifunction(long other, char *bufptr, long messleng, long command) Passes a buffer to the commit driver. This command provides a gateway for game programmer to access COMMIT directly. ----------------------------------------------------------------------------- | PALETTE FUNCTIONS: | ----------------------------------------------------------------------------- VBE_setPalette(long start, long num, char *palettebuffer) VBE_getPalette(long start, long num, char *palettebuffer) Set (num) palette palette entries starting at (start) palette entries are in a 4-byte format in this order: 0: Blue (0-63) 1: Green (0-63) 2: Red (0-63) 3: Reserved makepalookup(long palnum, char *remapbuf, signed char r, signed char g, signed char b, char dastat) This function allows different shirt colors for sprites. First prepare remapbuf, which is a 256 byte buffer of chars which the colors to remap. Palnum can be anywhere from 1-15. Since 0 is where the normal palette is stored, it is a bad idea to call this function with palnum=0. In BUILD.H notice I added a new variable, spritepal[MAXSPRITES]. Usually the value of this is 0 for the default palette. But if you change it to the palnum in the code between drawrooms() and drawmasks then the sprite will be drawn with that remapped palette. The last 3 parameters are the color that the palette fades to as you get further away. This color is normally black (0,0,0). White would be (63,63,63). if ((dastat&1) == 0) then makepalookup will allocate & deallocate the memory block for use but will not waste the time creating a palookup table (assuming you will create one yourself) setbrightness(char gammalevel, char *dapal) Use this function to adjust for gamma correction. Gammalevel - ranges from 0-15, 0 is darkest, 15 brightest. Default: 0 dapal: standard VGA palette (768 bytes) ------------------------------------------------------------------------------ | This document brought to you by: | | | | @@@@@@ @@@@@@ @@@@@@@@@@@@@@@@@@ @@@@@@@@ @@@@@@ | | @@@@@@ @@@@@@ @@@@@@@@@@@@@@@@@@ @@@@@@@@@ @@@@@@ | | @@@@@@ @@@@@@@ @@@@@@@@@@@@@@@@@@ @@@@@@@@@@ @@@@@@ | | @@@@@@ @@@@@@@ @@@@@@ @@@@@@@@@@@ @@@@@@ | | @@@@@@ @@@@@@@ @@@@@@ @@@@@@@@@@@@ @@@@@@ | | @@@@@@ @@@@@@@ @@@@@@ @@@@@@@@@@@@@ @@@@@@ | | @@@@@@@@@@@@ @@@@@@@@@@@@@@@ @@@@@@ @@@@@@@ @@@@@@ | | @@@@@@@ @@@@@@@@@@@ @@@@@@@@@@@@@@@ @@@@@@ @@@@@@@ @@@@@@ | | @@@@@@@@@@@@ @@@@@@@@@@@@@@@ @@@@@@ @@@@@@@ @@@@@@ | | @@@@@@ @@@@@@@ @@@@@@ @@@@@@ @@@@@@@@@@@@@ | | @@@@@@ @@@@@@@ @@@@@@ @@@@@@ @@@@@@@@@@@@ | | @@@@@@ @@@@@@@ @@@@@@ @@@@@@ @@@@@@@@@@@ | | @@@@@@ @@@@@@@ @@@@@@@@@@@@@@@@@@ @@@@@@ @@@@@@@@@@ | | @@@@@@ @@@@@@ @@@@@@@@@@@@@@@@@@ @@@@@@ @@@@@@@@@ | | @@@@@@ @@@@@@ @@@@@@@@@@@@@@@@@@ @@@@@@ @@@@@@@@ | | | | Ken Silverman of East Greenwich, RI USA | ------------------------------------------------------------------------------