mirror of
https://github.com/ZDoom/raze-gles.git
synced 2025-01-28 09:50:42 +00:00
b3639ae8e7
I have commented out the versions of these functions that perform bitmasks and shifts and replaced them with versions that cast to and from integral types, pending performance and compatibility research across platforms. git-svn-id: https://svn.eduke32.com/eduke32@5174 1a8010ca-5511-0410-912e-c29ae57300e0
10927 lines
351 KiB
C
10927 lines
351 KiB
C
// "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.
|
|
//
|
|
// This file has been modified from Ken Silverman's original release
|
|
// by Jonathon Fowler (jf@jonof.id.au)
|
|
|
|
#include "build.h"
|
|
#include "compat.h"
|
|
#include "pragmas.h"
|
|
#include "osd.h"
|
|
#include "cache1d.h"
|
|
#include "editor.h"
|
|
#include "common.h"
|
|
|
|
#include "baselayer.h"
|
|
#include "renderlayer.h"
|
|
|
|
#ifdef _WIN32
|
|
# include "winbits.h"
|
|
#endif
|
|
|
|
|
|
#include "m32script.h"
|
|
|
|
char levelname[BMAX_PATH] = {0};
|
|
|
|
#define TIMERINTSPERSECOND 120
|
|
|
|
#define updatecrc16(crc,dat) (crc = (((crc<<8)&65535)^crctable[((((uint16_t)crc)>>8)&65535)^dat]))
|
|
static int32_t crctable[256];
|
|
static char kensig[64];
|
|
|
|
static const char *CallExtGetVer(void);
|
|
static int32_t CallExtInit(void);
|
|
static int32_t CallExtPreInit(int32_t argc,const char **argv);
|
|
static int32_t CallExtPostStartupWindow(void);
|
|
static void CallExtUnInit(void);
|
|
static void CallExtPreCheckKeys(void);
|
|
static void CallExtAnalyzeSprites(int32_t, int32_t, int32_t, int32_t);
|
|
static void CallExtCheckKeys(void);
|
|
static void CallExtPreLoadMap(void);
|
|
static void CallExtSetupMapFilename(const char *mapname);
|
|
static void CallExtLoadMap(const char *mapname);
|
|
static int32_t CallExtPreSaveMap(void);
|
|
static void CallExtSaveMap(const char *mapname);
|
|
static const char *CallExtGetSectorCaption(int16_t sectnum);
|
|
static const char *CallExtGetWallCaption(int16_t wallnum);
|
|
static const char *CallExtGetSpriteCaption(int16_t spritenum);
|
|
static void CallExtShowSectorData(int16_t sectnum);
|
|
static void CallExtShowWallData(int16_t wallnum);
|
|
static void CallExtShowSpriteData(int16_t spritenum);
|
|
static void CallExtEditSectorData(int16_t sectnum);
|
|
static void CallExtEditWallData(int16_t wallnum);
|
|
static void CallExtEditSpriteData(int16_t spritenum);
|
|
// static const char *CallExtGetSectorType(int32_t lotag);
|
|
|
|
int8_t m32_clipping=2;
|
|
static int32_t m32_rotateang = 0;
|
|
|
|
// 0 1 2 3 4 5 6 7
|
|
// up, down, left, right, lshift, rctrl, lctrl, space
|
|
// 8 9 10 11 12 13
|
|
// a, z, pgdn, pgup, [,], [.]
|
|
// 14 15 16 17 18 19
|
|
// kpenter, enter, =, -, tab, `
|
|
uint8_t buildkeys[NUMBUILDKEYS] =
|
|
{
|
|
0xc8,0xd0,0xcb,0xcd,0x2a,0x9d,0x1d,0x39,
|
|
0x1e,0x2c,0xd1,0xc9,0x33,0x34,
|
|
0x9c,0x1c,0xd,0xc,0xf,0x29
|
|
};
|
|
|
|
// Start position
|
|
vec3_t startpos;
|
|
int16_t startang, startsectnum;
|
|
|
|
// Current position
|
|
vec3_t pos;
|
|
int32_t horiz = 100;
|
|
int16_t ang, cursectnum;
|
|
static int32_t hvel, vel, svel, angvel;
|
|
int32_t g_doHardcodedMovement = 1;
|
|
|
|
static int32_t mousexsurp = 0, mouseysurp = 0;
|
|
double msens = 1.0;
|
|
|
|
int32_t grponlymode = 0;
|
|
int32_t graphicsmode = 0;
|
|
|
|
int32_t synctics = 0, lockclock = 0;
|
|
|
|
// those ones save the respective 3d video vars while in 2d mode
|
|
// so that exiting from mapster32 in 2d mode saves the correct ones
|
|
float vid_gamma_3d=-1, vid_contrast_3d=-1, vid_brightness_3d=-1;
|
|
|
|
int32_t xdim2d = 640, ydim2d = 480, xdimgame = 640, ydimgame = 480, bppgame = 8;
|
|
int32_t forcesetup = 1;
|
|
|
|
#ifndef GEKKO
|
|
int32_t g_maxCacheSize = 24<<20;
|
|
#else
|
|
int32_t g_maxCacheSize = 8<<20;
|
|
#endif
|
|
|
|
static int16_t oldmousebstatus = 0;
|
|
|
|
char game_executable[BMAX_PATH] = {0};
|
|
|
|
int32_t zlock = 0x7fffffff, zmode = 0, kensplayerheight = 32;
|
|
//int16_t defaultspritecstat = 0;
|
|
|
|
int16_t localartfreq[MAXTILES];
|
|
int16_t localartlookup[MAXTILES], localartlookupnum;
|
|
|
|
char tempbuf[4096];
|
|
|
|
char names[MAXTILES][25];
|
|
const char *g_namesFileName = "NAMES.H";
|
|
|
|
int16_t asksave = 0;
|
|
int32_t osearchx, osearchy; //old search input
|
|
|
|
int32_t grid = 3, autogrid = 0, gridlock = 1, showtags = 2;
|
|
int32_t zoom = 768, gettilezoom = 1;
|
|
int32_t lastpm16time = 0;
|
|
|
|
extern int32_t mapversion;
|
|
|
|
int16_t highlight[MAXWALLS+MAXSPRITES];
|
|
int16_t highlightsector[MAXSECTORS], highlightsectorcnt = -1;
|
|
extern char textfont[128][8];
|
|
|
|
int32_t tempsectornum = -1; // for auto ceiling/floor alignment
|
|
int32_t temppicnum, tempcstat, templotag, temphitag, tempextra;
|
|
uint32_t temppal, tempvis, tempxrepeat, tempyrepeat, tempxpanning=0, tempypanning=0;
|
|
int32_t tempshade, tempxvel, tempyvel, tempzvel;
|
|
char somethingintab = 255;
|
|
|
|
// Only valid when highlightsectorcnt>0 and no structural
|
|
// modifications (deleting/inserting sectors or points, setting new firstwall)
|
|
// have been made:
|
|
static int16_t onextwall[MAXWALLS]; // onextwall[i]>=0 implies wall[i].nextwall < 0
|
|
static void mkonwvalid(void) { chsecptr_onextwall = onextwall; }
|
|
static void mkonwinvalid(void) { chsecptr_onextwall = NULL; tempsectornum=-1; }
|
|
static void mkonwinvalid_keeptempsect(void) { chsecptr_onextwall = NULL; }
|
|
static int32_t onwisvalid(void) { return chsecptr_onextwall != NULL; }
|
|
|
|
int32_t mlook = 0, mskip=0;
|
|
int32_t revertCTRL=0,scrollamount=3;
|
|
int32_t unrealedlook=1, quickmapcycling=1; //PK
|
|
|
|
char program_origcwd[BMAX_PATH];
|
|
const char *mapster32_fullpath;
|
|
char *testplay_addparam = 0;
|
|
|
|
static char boardfilename[BMAX_PATH], selectedboardfilename[BMAX_PATH];
|
|
//extern char levelname[BMAX_PATH]; // in astub.c XXX: clean up this mess!!!
|
|
|
|
void B_SetBoardFileName(const char *fn)
|
|
{
|
|
Bstrncpyz(boardfilename, fn, BMAX_PATH);
|
|
}
|
|
|
|
static fnlist_t fnlist;
|
|
static CACHE1D_FIND_REC *finddirshigh=NULL, *findfileshigh=NULL;
|
|
static int32_t currentlist=0;
|
|
|
|
//static int32_t repeatcountx, repeatcounty;
|
|
|
|
static int32_t fillist[640];
|
|
// used for fillsector, batch point insertion, backup_highlighted_map
|
|
static int32_t tempxyar[MAXWALLS][2];
|
|
|
|
static int32_t mousx, mousy;
|
|
int16_t prefixtiles[10] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
|
|
uint8_t hlsectorbitmap[MAXSECTORS>>3]; // show2dsector is already taken...
|
|
static int32_t minhlsectorfloorz, numhlsecwalls;
|
|
|
|
// used for:
|
|
// - hl_all_bunch_sectors_p
|
|
// - AlignWalls
|
|
// - trace_loop
|
|
static uint8_t visited[MAXWALLS>>3];
|
|
|
|
typedef struct
|
|
{
|
|
int16_t numsectors, numwalls, numsprites;
|
|
#ifdef YAX_ENABLE
|
|
int16_t numyaxbunches;
|
|
int16_t *bunchnum; // [numsectors][2]
|
|
int16_t *ynextwall; // [numwalls][2]
|
|
#endif
|
|
tsectortype *sector;
|
|
twalltype *wall;
|
|
tspritetype *sprite;
|
|
} mapinfofull_t;
|
|
|
|
int32_t g_doScreenShot;
|
|
|
|
#define eitherALT (keystatus[0x38]|keystatus[0xb8])
|
|
#define eitherCTRL (keystatus[0x1d]|keystatus[0x9d])
|
|
#define eitherSHIFT (keystatus[0x2a]|keystatus[0x36])
|
|
|
|
#define DOWN_BK(BuildKey) (keystatus[buildkeys[BK_##BuildKey]])
|
|
|
|
int32_t pk_turnaccel=16;
|
|
int32_t pk_turndecel=12;
|
|
int32_t pk_uedaccel=3;
|
|
|
|
int8_t keeptexturestretch = 1;
|
|
int8_t sideview_reversehrot = 0;
|
|
|
|
char lastpm16buf[156];
|
|
|
|
//static int32_t checksectorpointer_warn = 0;
|
|
static int32_t saveboard_savedtags, saveboard_fixedsprites;
|
|
static int32_t saveboard_canceled;
|
|
|
|
static int32_t backup_highlighted_map(mapinfofull_t *mapinfo);
|
|
static int32_t restore_highlighted_map(mapinfofull_t *mapinfo, int32_t forreal);
|
|
static void SaveBoardAndPrintMessage(const char *fn);
|
|
|
|
static int32_t adjustmark(int32_t *xplc, int32_t *yplc, int16_t danumwalls);
|
|
static void locktogrid(int32_t *dax, int32_t *day);
|
|
static int32_t checkautoinsert(int32_t dax, int32_t day, int16_t danumwalls);
|
|
static void keytimerstuff(void);
|
|
static void flipwalls(int16_t numwalls, int16_t newnumwalls);
|
|
static int32_t insertpoint(int16_t linehighlight, int32_t dax, int32_t day, int32_t *mapwallnum);
|
|
static void deletepoint(int16_t point, int32_t runi);
|
|
static int32_t deletesector(int16_t sucksect);
|
|
static int16_t whitelinescan(int16_t sucksect, int16_t dalinehighlight);
|
|
static void printcoords16(int32_t posxe, int32_t posye, int16_t ange);
|
|
static void overheadeditor(void);
|
|
static int32_t getlinehighlight(int32_t xplc, int32_t yplc, int32_t line, int8_t ignore_pointhighlight);
|
|
static int32_t movewalls(int32_t start, int32_t offs);
|
|
static int32_t loadnames(const char *namesfile, int8_t root);
|
|
static void getclosestpointonwall(int32_t x, int32_t y, int32_t dawall, int32_t *nx, int32_t *ny,
|
|
int32_t maybe_screen_coord_p);
|
|
static void initcrc(void);
|
|
|
|
static int32_t menuselect(void);
|
|
static int32_t menuselect_auto(int32_t); //PK
|
|
|
|
static int32_t insert_sprite_common(int32_t sucksect, int32_t dax, int32_t day);
|
|
static void correct_ornamented_sprite(int32_t i, int32_t hitw);
|
|
|
|
static int32_t getfilenames(const char *path, const char *kind);
|
|
|
|
// Get basename of BUILD file name (forward slashes as directory separators).
|
|
static const char *getbasefn(const char *fn)
|
|
{
|
|
const char *slash = Bstrrchr(fn, '/');
|
|
return slash ? slash+1 : fn;
|
|
}
|
|
|
|
void clearkeys(void)
|
|
{
|
|
Bmemset(keystatus,0,sizeof(keystatus));
|
|
}
|
|
|
|
#ifdef USE_OPENGL
|
|
int32_t osdcmd_restartvid(const osdfuncparm_t *parm)
|
|
{
|
|
UNREFERENCED_PARAMETER(parm);
|
|
|
|
if (!in3dmode()) return OSDCMD_OK;
|
|
|
|
resetvideomode();
|
|
if (setgamemode(fullscreen,xdim,ydim,bpp))
|
|
OSD_Printf("restartvid: Reset failed...\n");
|
|
|
|
return OSDCMD_OK;
|
|
}
|
|
#endif
|
|
|
|
static int32_t osdcmd_vidmode(const osdfuncparm_t *parm)
|
|
{
|
|
int32_t newx = xdim, newy = ydim, newbpp = bpp, newfullscreen = fullscreen;
|
|
#ifdef USE_OPENGL
|
|
int32_t tmp;
|
|
#endif
|
|
|
|
switch (parm->numparms)
|
|
{
|
|
#ifdef USE_OPENGL
|
|
case 1: // bpp switch
|
|
tmp = Batol(parm->parms[0]);
|
|
if (!(tmp==8 || tmp==16 || tmp==32))
|
|
return OSDCMD_SHOWHELP;
|
|
newbpp = tmp;
|
|
break;
|
|
case 4: // fs, res, bpp switch
|
|
newfullscreen = (Batol(parm->parms[3]) != 0);
|
|
case 3: // res & bpp switch
|
|
tmp = Batol(parm->parms[2]);
|
|
if (!(tmp==8 || tmp==16 || tmp==32))
|
|
return OSDCMD_SHOWHELP;
|
|
newbpp = tmp;
|
|
#endif
|
|
case 2: // res switch
|
|
newx = Batol(parm->parms[0]);
|
|
newy = Batol(parm->parms[1]);
|
|
break;
|
|
default:
|
|
return OSDCMD_SHOWHELP;
|
|
}
|
|
|
|
if (!in3dmode())
|
|
{
|
|
qsetmodeany(newx,newy);
|
|
xdim2d = xdim;
|
|
ydim2d = ydim;
|
|
|
|
begindrawing(); //{{{
|
|
CLEARLINES2D(0, ydim16, 0);
|
|
enddrawing(); //}}}
|
|
|
|
ydim16 = ydim-STATUS2DSIZ2;
|
|
|
|
return OSDCMD_OK;
|
|
}
|
|
|
|
if (setgamemode(newfullscreen,newx,newy,newbpp))
|
|
OSD_Printf("vidmode: Mode change failed!\n");
|
|
|
|
xdimgame = newx;
|
|
ydimgame = newy;
|
|
bppgame = newbpp;
|
|
fullscreen = newfullscreen;
|
|
|
|
return OSDCMD_OK;
|
|
}
|
|
|
|
|
|
#ifdef M32_SHOWDEBUG
|
|
char m32_debugstr[64][128];
|
|
int32_t m32_numdebuglines=0;
|
|
|
|
static void M32_drawdebug(void)
|
|
{
|
|
int32_t i;
|
|
int32_t x=4, y=8;
|
|
|
|
#if 0
|
|
{
|
|
static char tstr[128];
|
|
Bsprintf(tstr, "search... stat=%d, sector=%d, wall=%d (%d), isbottom=%d, asksave=%d",
|
|
searchstat, searchsector, searchwall, searchbottomwall, searchisbottom, asksave);
|
|
printext256(x,y,whitecol,0,tstr,xdimgame>640?0:1);
|
|
}
|
|
#endif
|
|
if (m32_numdebuglines>0)
|
|
{
|
|
begindrawing();
|
|
for (i=0; i<m32_numdebuglines && y<ydim-8; i++, y+=8)
|
|
printext256(x,y,whitecol,0,m32_debugstr[i],xdimgame>640?0:1);
|
|
enddrawing();
|
|
}
|
|
m32_numdebuglines=0;
|
|
}
|
|
#endif
|
|
|
|
#ifdef YAX_ENABLE
|
|
// Check whether bunchnum has exactly one corresponding floor and ceiling
|
|
// and return it in this case. If not 1-to-1, return -1.
|
|
int32_t yax_is121(int16_t bunchnum, int16_t getfloor)
|
|
{
|
|
int32_t i;
|
|
i = headsectbunch[0][bunchnum];
|
|
if (i<0 || nextsectbunch[0][i]>=0)
|
|
return -1;
|
|
i = headsectbunch[1][bunchnum];
|
|
if (i<0 || nextsectbunch[1][i]>=0)
|
|
return -1;
|
|
|
|
return headsectbunch[getfloor][bunchnum];
|
|
}
|
|
|
|
static int32_t yax_numsectsinbunch(int16_t bunchnum, int16_t cf)
|
|
{
|
|
int32_t i, n=0;
|
|
|
|
if (bunchnum<0 || bunchnum>=numyaxbunches)
|
|
return -1;
|
|
|
|
for (SECTORS_OF_BUNCH(bunchnum, cf, i))
|
|
n++;
|
|
|
|
return n;
|
|
}
|
|
|
|
static void yax_fixreverselinks(int16_t oldwall, int16_t newwall)
|
|
{
|
|
int32_t cf, ynw;
|
|
for (cf=0; cf<2; cf++)
|
|
{
|
|
ynw = yax_getnextwall(oldwall, cf);
|
|
if (ynw >= 0)
|
|
yax_setnextwall(ynw, !cf, newwall);
|
|
}
|
|
}
|
|
|
|
static void yax_tweakwalls(int16_t start, int16_t offs)
|
|
{
|
|
int32_t i, nw, cf;
|
|
for (i=0; i<numwalls; i++)
|
|
for (cf=0; cf<2; cf++)
|
|
{
|
|
nw = yax_getnextwall(i, cf);
|
|
if (nw >= start)
|
|
yax_setnextwall(i, cf, nw+offs);
|
|
}
|
|
}
|
|
|
|
static void yax_resetbunchnums(void)
|
|
{
|
|
int32_t i;
|
|
|
|
for (i=0; i<MAXSECTORS; i++)
|
|
yax_setbunches(i, -1, -1);
|
|
yax_update(1);
|
|
yax_updategrays(pos.z);
|
|
}
|
|
|
|
// Whether a wall is constrained by sector extensions.
|
|
// If false, it's a wall that you can freely move around,
|
|
// attach points to, etc...
|
|
static int32_t yax_islockedwall(int16_t line)
|
|
{
|
|
#ifdef NEW_MAP_FORMAT
|
|
return (wall[line].upwall>=0 || wall[line].dnwall>=0);
|
|
#else
|
|
return !!(wall[line].cstat&YAX_NEXTWALLBITS);
|
|
#endif
|
|
}
|
|
|
|
# define DEFAULT_YAX_HEIGHT (2048<<4)
|
|
#endif
|
|
|
|
static void reset_default_mapstate(void)
|
|
{
|
|
pos.x = 32768; //new board!
|
|
pos.y = 32768;
|
|
pos.z = 0;
|
|
ang = 1536;
|
|
cursectnum = -1;
|
|
|
|
numsectors = 0;
|
|
numwalls = 0;
|
|
|
|
editorzrange[0] = INT32_MIN;
|
|
editorzrange[1] = INT32_MAX;
|
|
|
|
initspritelists();
|
|
taglab_init();
|
|
E_MapArt_Clear();
|
|
#ifdef YAX_ENABLE
|
|
yax_resetbunchnums();
|
|
#endif
|
|
g_loadedMapVersion = -1;
|
|
}
|
|
|
|
static void m32_keypresscallback(int32_t code, int32_t downp)
|
|
{
|
|
UNREFERENCED_PARAMETER(downp);
|
|
|
|
g_iReturnVar = code;
|
|
VM_OnEvent(EVENT_KEYPRESS, -1);
|
|
}
|
|
|
|
void M32_ResetFakeRORTiles(void)
|
|
{
|
|
#ifdef POLYMER
|
|
# ifdef YAX_ENABLE
|
|
// END_TWEAK ceiling/floor fake 'TROR' pics, see BEGIN_TWEAK in engine.c
|
|
if (getrendermode() == REND_POLYMER && showinvisibility)
|
|
{
|
|
int32_t i;
|
|
|
|
for (i=0; i<numyaxbunches; i++)
|
|
{
|
|
yax_tweakpicnums(i, YAX_CEILING, 1);
|
|
yax_tweakpicnums(i, YAX_FLOOR, 1);
|
|
}
|
|
}
|
|
# endif
|
|
#endif
|
|
}
|
|
|
|
void M32_DrawRoomsAndMasks(void)
|
|
{
|
|
static int srchwall = -1;
|
|
|
|
VM_OnEvent(EVENT_PREDRAW3DSCREEN, -1);
|
|
|
|
yax_preparedrawrooms();
|
|
drawrooms(pos.x,pos.y,pos.z,ang,horiz,cursectnum);
|
|
yax_drawrooms(CallExtAnalyzeSprites, cursectnum, 0, 0);
|
|
|
|
const int osearchwall=searchwall, osearchstat=searchstat;
|
|
if (srchwall >= 0)
|
|
{
|
|
// a.m32 states 'tduprot' and 'tduplin' need searchstat to check for
|
|
// whether we've hit a sprite, but these would be only set after the
|
|
// drawmasks(). Hence this hackish workaround.
|
|
searchstat = 3;
|
|
searchwall = srchwall;
|
|
}
|
|
CallExtAnalyzeSprites(0,0,0,0);
|
|
searchwall = osearchwall, searchstat=osearchstat;
|
|
|
|
drawmasks();
|
|
srchwall = (searchstat == 3) ? searchwall : -1;
|
|
M32_ResetFakeRORTiles();
|
|
|
|
#ifdef POLYMER
|
|
if (getrendermode() == REND_POLYMER && searchit == 2)
|
|
{
|
|
polymer_editorpick();
|
|
drawrooms(pos.x,pos.y,pos.z,ang,horiz,cursectnum);
|
|
CallExtAnalyzeSprites(0,0,0,0);
|
|
drawmasks();
|
|
M32_ResetFakeRORTiles();
|
|
}
|
|
#endif
|
|
|
|
VM_OnEvent(EVENT_DRAW3DSCREEN, -1);
|
|
|
|
if (g_doScreenShot)
|
|
{
|
|
screencapture("mcapxxxx.tga", 0, "Mapster32, from script");
|
|
g_doScreenShot = 0;
|
|
}
|
|
}
|
|
|
|
int32_t app_main(int32_t argc, const char **argv)
|
|
{
|
|
#ifdef STARTUP_SETUP_WINDOW
|
|
char cmdsetup = 0;
|
|
#endif
|
|
char quitflag;
|
|
int32_t i;
|
|
|
|
pathsearchmode = 1; // unrestrict findfrompath so that full access to the filesystem can be had
|
|
|
|
#ifdef USE_OPENGL
|
|
OSD_RegisterFunction("restartvid","restartvid: reinitialize the video mode",osdcmd_restartvid);
|
|
OSD_RegisterFunction("vidmode","vidmode <xdim> <ydim> <bpp> <fullscreen>: immediately change the video mode",osdcmd_vidmode);
|
|
baselayer_osdcmd_vidmode_func = osdcmd_vidmode;
|
|
#else
|
|
OSD_RegisterFunction("vidmode","vidmode <xdim> <ydim>: immediately change the video mode",osdcmd_vidmode);
|
|
#endif
|
|
|
|
wm_setapptitle(AppProperName);
|
|
|
|
editstatus = 1;
|
|
newaspect_enable = 1;
|
|
|
|
if ((i = CallExtPreInit(argc,argv)) < 0) return -1;
|
|
|
|
#ifdef _WIN32
|
|
backgroundidle = 1;
|
|
#endif
|
|
|
|
for (i=1; i<argc; i++)
|
|
{
|
|
if (argv[i][0] == '-')
|
|
{
|
|
#ifdef STARTUP_SETUP_WINDOW
|
|
if (!Bstrcmp(argv[i], "-setup")) cmdsetup = 1;
|
|
else
|
|
#endif
|
|
if (!Bstrcmp(argv[i], "-help") || !Bstrcmp(argv[i], "--help") || !Bstrcmp(argv[i], "-?"))
|
|
{
|
|
#ifdef WM_MSGBOX_WINDOW
|
|
wm_msgbox(AppProperName,
|
|
#else
|
|
Bprintf(
|
|
#endif
|
|
"%s\n"
|
|
"Syntax: %s [options] mapname\n"
|
|
"Options:\n"
|
|
"\t-grp\tUse an extra GRP or ZIP file.\n"
|
|
"\t-g\tSame as above.\n"
|
|
#ifdef STARTUP_SETUP_WINDOW
|
|
"\t-setup\tDisplays the configuration dialogue box before entering the editor.\n"
|
|
#endif
|
|
, AppProperName, AppTechnicalName);
|
|
return 0;
|
|
}
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (boardfilename[0] == 0)
|
|
Bstrcpy(boardfilename,"newboard.map");
|
|
else if (Bstrchr(boardfilename,'.') == 0)
|
|
Bstrcat(boardfilename, ".map");
|
|
//Bcanonicalisefilename(boardfilename,0);
|
|
|
|
OSD_SetFunctions(
|
|
NULL, NULL, NULL, NULL, NULL,
|
|
COMMON_clearbackground,
|
|
BGetTime,
|
|
NULL
|
|
);
|
|
|
|
OSD_SetParameters(0,2, 0,0, 4,0);
|
|
|
|
if (!getcwd(program_origcwd,BMAX_PATH))
|
|
program_origcwd[0] = '\0';
|
|
|
|
Bstrncpy(game_executable, DefaultGameLocalExec, sizeof(game_executable));
|
|
|
|
if (preinitengine())
|
|
{
|
|
wm_msgbox("Build Engine Initialization Error",
|
|
"There was a problem initializing the Build engine: %s", engineerrstr);
|
|
ERRprintf("app_main: There was a problem initializing the Build engine: %s\n", engineerrstr);
|
|
Bexit(2);
|
|
}
|
|
|
|
if ((i = CallExtInit()) < 0) return -1;
|
|
|
|
#ifdef STARTUP_SETUP_WINDOW
|
|
if (i || forcesetup || cmdsetup)
|
|
{
|
|
if (quitevent || !startwin_run())
|
|
{
|
|
uninitengine();
|
|
Bexit(0);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
if (CallExtPostStartupWindow() < 0) return -1;
|
|
|
|
loadnames(g_namesFileName, 1);
|
|
|
|
if (initinput()) return -1;
|
|
|
|
initmouse();
|
|
|
|
inittimer(TIMERINTSPERSECOND);
|
|
installusertimercallback(keytimerstuff);
|
|
|
|
loadpics("tiles000.art", g_maxCacheSize);
|
|
|
|
#ifdef YAX_ENABLE
|
|
// init dummy texture for YAX
|
|
// must be after loadpics(), which inits BUILD's cache
|
|
|
|
i = MAXTILES-1;
|
|
if (tilesiz[i].x==0 && tilesiz[i].x==0)
|
|
{
|
|
static char R[8*16] = { //
|
|
0, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 0, 0,
|
|
0, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 0,
|
|
0, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 0, 0, 1, 0,
|
|
0, 1, 1, 1, 0, 0, 0, 1, 1, 0, 0, 1, 1, 1, 0, 0,
|
|
0, 1, 0, 1, 0, 0, 1, 0, 0, 1, 0, 1, 0, 0, 1, 0,
|
|
0, 1, 0, 0, 1, 0, 1, 0, 0, 1, 0, 1, 0, 0, 1, 0,
|
|
0, 1, 0, 0, 1, 0, 0, 1, 1, 0, 0, 1, 0, 0, 1, 0,
|
|
};
|
|
|
|
char *newtile;
|
|
int32_t sx=32, sy=32, col, j;
|
|
|
|
walock[i] = 255; // permanent tile
|
|
picsiz[i] = 5 + (5<<4);
|
|
tilesiz[i].x = sx; tilesiz[i].y = sy;
|
|
allocache(&waloff[i], sx*sy, &walock[i]);
|
|
newtile = (char *)waloff[i];
|
|
|
|
col = getclosestcol(128>>2, 128>>2, 0);
|
|
for (j=0; j<(signed)sizeof(R); j++)
|
|
R[j] *= col;
|
|
|
|
Bmemset(newtile, 0, sx*sy);
|
|
for (j=0; j<8; j++)
|
|
Bmemcpy(&newtile[32*j], &R[16*j], 16);
|
|
}
|
|
#endif
|
|
|
|
Bstrcpy(kensig,"Uses BUILD technology by Ken Silverman");
|
|
initcrc();
|
|
|
|
{
|
|
const char *defsfile = G_DefFile();
|
|
if (!loaddefinitionsfile(defsfile))
|
|
initprintf("Definitions file \"%s\" loaded.\n",defsfile);
|
|
}
|
|
|
|
for (i=0; i < g_defModulesNum; ++i)
|
|
Bfree (g_defModules[i]);
|
|
Bfree (g_defModules);
|
|
g_defModules = NULL; // be defensive...
|
|
|
|
// Here used to be the 'whitecol' calculation
|
|
|
|
#ifdef HAVE_CLIPSHAPE_FEATURE
|
|
int k = clipmapinfo_load();
|
|
if (k>0)
|
|
initprintf("There was an error loading the sprite clipping map (status %d).\n", k);
|
|
|
|
for (i=0; i < g_clipMapFilesNum; ++i)
|
|
Bfree (g_clipMapFiles[i]);
|
|
Bfree (g_clipMapFiles);
|
|
g_clipMapFiles = NULL;
|
|
#endif
|
|
|
|
taglab_init();
|
|
|
|
mkonwinvalid();
|
|
|
|
if (LoadBoard(boardfilename, 1))
|
|
reset_default_mapstate();
|
|
|
|
totalclock = 0;
|
|
|
|
updatesector(pos.x,pos.y,&cursectnum);
|
|
|
|
setkeypresscallback(&m32_keypresscallback);
|
|
|
|
if (cursectnum == -1)
|
|
{
|
|
vid_gamma_3d = vid_gamma;
|
|
vid_brightness_3d = vid_brightness;
|
|
vid_contrast_3d = vid_contrast;
|
|
|
|
vid_gamma = vid_contrast = 1.0;
|
|
vid_brightness = 0.0;
|
|
|
|
setbrightness(0,0,0);
|
|
if (setgamemode(fullscreen, xdim2d, ydim2d, 8) < 0)
|
|
{
|
|
CallExtUnInit();
|
|
uninitengine();
|
|
Bprintf("%d * %d not supported in this graphics mode\n",xdim2d,ydim2d);
|
|
Bexit(0);
|
|
}
|
|
|
|
// executed once per init, but after setgamemode so that OSD has the right width
|
|
OSD_Exec("m32_autoexec.cfg");
|
|
|
|
system_getcvars();
|
|
|
|
overheadeditor();
|
|
keystatus[buildkeys[BK_MODE2D_3D]] = 0;
|
|
|
|
vid_gamma = vid_gamma_3d;
|
|
vid_contrast = vid_contrast_3d;
|
|
vid_brightness = vid_brightness_3d;
|
|
|
|
vid_gamma_3d = vid_contrast_3d = vid_brightness_3d = -1;
|
|
|
|
setbrightness(GAMMA_CALC,0,0);
|
|
}
|
|
else
|
|
{
|
|
if (setgamemode(fullscreen, xdimgame, ydimgame, bppgame) < 0)
|
|
{
|
|
CallExtUnInit();
|
|
uninitengine();
|
|
Bprintf("%d * %d not supported in this graphics mode\n",xdim,ydim);
|
|
Bexit(0);
|
|
}
|
|
|
|
// executed once per init, but after setgamemode so that OSD has the right width
|
|
OSD_Exec("m32_autoexec.cfg");
|
|
|
|
system_getcvars();
|
|
|
|
setbrightness(GAMMA_CALC,0,0);
|
|
}
|
|
|
|
CANCEL:
|
|
quitflag = 0;
|
|
while (quitflag == 0)
|
|
{
|
|
if (handleevents())
|
|
{
|
|
if (quitevent)
|
|
{
|
|
keystatus[1] = 1;
|
|
quitevent = 0;
|
|
}
|
|
}
|
|
|
|
OSD_DispatchQueued();
|
|
|
|
nextpage();
|
|
synctics = totalclock-lockclock;
|
|
lockclock += synctics;
|
|
|
|
CallExtPreCheckKeys();
|
|
|
|
M32_DrawRoomsAndMasks();
|
|
|
|
inputchecked = 1;
|
|
|
|
#ifdef M32_SHOWDEBUG
|
|
if (searchstat>=0 && (searchwall<0 || searchsector<0))
|
|
{
|
|
if (m32_numdebuglines<64)
|
|
Bsprintf(m32_debugstr[m32_numdebuglines++], "inconsistent search variables!");
|
|
searchstat = -1;
|
|
}
|
|
|
|
M32_drawdebug();
|
|
#endif
|
|
CallExtCheckKeys();
|
|
|
|
|
|
if (keystatus[1])
|
|
{
|
|
keystatus[1] = 0;
|
|
|
|
printext256(0,0,whitecol,0,"Are you sure you want to quit?",0);
|
|
|
|
showframe(1);
|
|
synctics = totalclock-lockclock;
|
|
lockclock += synctics;
|
|
|
|
while ((keystatus[1]|keystatus[0x1c]|keystatus[0x39]|keystatus[0x31]) == 0)
|
|
{
|
|
idle_waitevent();
|
|
if (handleevents())
|
|
{
|
|
if (quitevent)
|
|
{
|
|
quitflag = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (keystatus[0x15]||keystatus[0x1c]) // Y or ENTER
|
|
{
|
|
keystatus[0x15] = 0;
|
|
keystatus[0x1c] = 0;
|
|
quitflag = 1; break;
|
|
}
|
|
}
|
|
while (keystatus[1])
|
|
{
|
|
keystatus[1] = 0;
|
|
quitevent = 0;
|
|
goto CANCEL;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (asksave)
|
|
{
|
|
i = CheckMapCorruption(4, 0);
|
|
|
|
printext256(0,8,whitecol,0,i<4?"Save changes?":"Map is heavily corrupt. Save changes?",0);
|
|
showframe(1);
|
|
|
|
while ((keystatus[1]|keystatus[0x1c]|keystatus[0x39]|keystatus[0x31]|keystatus[0x2e]) == 0)
|
|
{
|
|
idle_waitevent();
|
|
if (handleevents()) { if (quitevent) break; } // like saying no
|
|
|
|
if (keystatus[0x15] || keystatus[0x1c]) // Y or ENTER
|
|
{
|
|
keystatus[0x15] = keystatus[0x1c] = 0;
|
|
|
|
SaveBoard(NULL, M32_SB_ASKOV);
|
|
|
|
break;
|
|
}
|
|
}
|
|
while (keystatus[1]||keystatus[0x2e])
|
|
{
|
|
keystatus[1] = keystatus[0x2e] = 0;
|
|
quitevent = 0;
|
|
goto CANCEL;
|
|
}
|
|
}
|
|
|
|
|
|
CallExtUnInit();
|
|
// clearfilenames();
|
|
uninitengine();
|
|
|
|
return(0);
|
|
}
|
|
|
|
static int32_t mhk=0;
|
|
static void loadmhk(int32_t domessage)
|
|
{
|
|
char levname[BMAX_PATH];
|
|
|
|
if (!mhk)
|
|
return;
|
|
|
|
Bstrcpy(levname, boardfilename);
|
|
append_ext_UNSAFE(levname, ".mhk");
|
|
|
|
if (!loadmaphack(levname))
|
|
{
|
|
if (domessage)
|
|
message("Loaded map hack file \"%s\"",levname);
|
|
else
|
|
initprintf("Loaded map hack file \"%s\"\n",levname);
|
|
}
|
|
else
|
|
{
|
|
mhk=2;
|
|
if (domessage)
|
|
message("No maphack found for map \"%s\"",boardfilename);
|
|
}
|
|
}
|
|
|
|
// this is spriteon{ceiling,ground}z from astub.c packed into
|
|
// one convenient function
|
|
void spriteoncfz(int32_t i, int32_t *czptr, int32_t *fzptr)
|
|
{
|
|
int32_t height, zofs;
|
|
|
|
getzsofslope(sprite[i].sectnum, sprite[i].x,sprite[i].y, czptr, fzptr);
|
|
if ((sprite[i].cstat&48)==32)
|
|
return;
|
|
|
|
zofs = spriteheightofs(i, &height, 0);
|
|
|
|
*czptr += height - zofs;
|
|
*fzptr -= zofs;
|
|
}
|
|
|
|
static void move_and_update(int32_t xvect, int32_t yvect, int32_t addshr)
|
|
{
|
|
if (m32_clipping==0)
|
|
{
|
|
pos.x += xvect>>(14+addshr);
|
|
pos.y += yvect>>(14+addshr);
|
|
updatesector(pos.x,pos.y, &cursectnum);
|
|
}
|
|
else
|
|
{
|
|
clipmove(&pos,&cursectnum, xvect>>addshr,yvect>>addshr,
|
|
128,4<<8,4<<8, (m32_clipping==1) ? 0 : CLIPMASK0);
|
|
}
|
|
}
|
|
|
|
static void mainloop_move(void)
|
|
{
|
|
int32_t xvect, yvect, doubvel;
|
|
|
|
if (angvel != 0) //ang += angvel * constant
|
|
{
|
|
//ENGINE calculates angvel for you
|
|
|
|
//Lt. shift makes turn velocity 50% faster
|
|
doubvel = (synctics + DOWN_BK(RUN)*(synctics>>1));
|
|
|
|
ang += ((angvel*doubvel)>>4);
|
|
ang &= 2047;
|
|
}
|
|
if ((vel|svel) != 0)
|
|
{
|
|
//Lt. shift doubles forward velocity
|
|
doubvel = (1+(DOWN_BK(RUN)))*synctics;
|
|
|
|
xvect = 0;
|
|
yvect = 0;
|
|
|
|
if (vel != 0)
|
|
{
|
|
xvect += (vel*doubvel*(int32_t)sintable[(ang+2560)&2047])>>3;
|
|
yvect += (vel*doubvel*(int32_t)sintable[(ang+2048)&2047])>>3;
|
|
}
|
|
if (svel != 0)
|
|
{
|
|
xvect += (svel*doubvel*(int32_t)sintable[(ang+2048)&2047])>>3;
|
|
yvect += (svel*doubvel*(int32_t)sintable[(ang+1536)&2047])>>3;
|
|
}
|
|
|
|
move_and_update(xvect, yvect, 0);
|
|
}
|
|
}
|
|
|
|
static void handle_sprite_in_clipboard(int32_t i)
|
|
{
|
|
if (somethingintab == 3)
|
|
{
|
|
int32_t j, k;
|
|
|
|
sprite[i].picnum = temppicnum;
|
|
if (tilesiz[temppicnum].x <= 0 || tilesiz[temppicnum].y <= 0)
|
|
{
|
|
j = 0;
|
|
for (k=0; k<MAXTILES; k++)
|
|
if (tilesiz[k].x > 0 && tilesiz[k].y > 0)
|
|
{
|
|
j = k;
|
|
break;
|
|
}
|
|
sprite[i].picnum = j;
|
|
}
|
|
sprite[i].shade = tempshade;
|
|
sprite[i].pal = temppal;
|
|
sprite[i].xrepeat = max(tempxrepeat, 1);
|
|
sprite[i].yrepeat = max(tempyrepeat, 1);
|
|
sprite[i].cstat = tempcstat;
|
|
}
|
|
}
|
|
|
|
|
|
void editinput(void)
|
|
{
|
|
int32_t mousz, bstatus;
|
|
int32_t i, tempint=0;
|
|
int32_t goalz, xvect, yvect, hiz, loz, oposz;
|
|
int32_t dax, day, hihit, lohit, omlook=mlook;
|
|
|
|
// 3B 3C 3D 3E 3F 40 41 42 43 44 57 58 46
|
|
// F1 F2 F3 F4 F5 F6 F7 F8 F9 F10 F11 F12 SCROLL
|
|
|
|
mousz = 0;
|
|
getmousevalues(&mousx,&mousy,&bstatus);
|
|
mousx = (mousx<<16) + mousexsurp;
|
|
mousy = (mousy<<16) + mouseysurp;
|
|
|
|
if (unrealedlook && !mskip)
|
|
{
|
|
if (mlook==0 && (bstatus&(1|2|4))==2)
|
|
mlook = 3;
|
|
else if ((bstatus&(1|2|4))==1)
|
|
mlook = 3;
|
|
}
|
|
|
|
{
|
|
ldiv_t ld;
|
|
if (mlook)
|
|
{
|
|
ld = ldiv(mousx, (int32_t)((1<<16)/(msens*0.5f))); mousx = ld.quot; mousexsurp = ld.rem;
|
|
ld = ldiv(mousy, (int32_t)((1<<16)/(msens*0.25f))); mousy = ld.quot; mouseysurp = ld.rem;
|
|
}
|
|
else
|
|
{
|
|
ld = ldiv(mousx, (int32_t)((1<<16)/msens)); mousx = ld.quot; mousexsurp = ld.rem;
|
|
ld = ldiv(mousy, (int32_t)((1<<16)/msens)); mousy = ld.quot; mouseysurp = ld.rem;
|
|
}
|
|
}
|
|
|
|
if (mlook == 3)
|
|
mlook = omlook;
|
|
|
|
// UnrealEd:
|
|
// rmb: mouselook
|
|
// lbm: x:turn y:fwd/back local x
|
|
// lmb&rmb: x:strafe y:up/dn (move in local yz plane)
|
|
// mmb: fwd/back in viewing vector
|
|
|
|
if (unrealedlook && !mskip) //PK
|
|
{
|
|
if ((bstatus&(1|2|4))==1)
|
|
{
|
|
ang += mousx;
|
|
xvect = -((mousy*(int32_t)sintable[(ang+2560)&2047])<<(3+pk_uedaccel));
|
|
yvect = -((mousy*(int32_t)sintable[(ang+2048)&2047])<<(3+pk_uedaccel));
|
|
|
|
move_and_update(xvect, yvect, 0);
|
|
}
|
|
else if (!mlook && (bstatus&(1|2|4))==2)
|
|
{
|
|
mlook=2;
|
|
}
|
|
else if ((bstatus&(1|2|4))==(1|2))
|
|
{
|
|
zmode = 2;
|
|
xvect = -((mousx*(int32_t)sintable[(ang+2048)&2047])<<pk_uedaccel);
|
|
yvect = -((mousx*(int32_t)sintable[(ang+1536)&2047])<<pk_uedaccel);
|
|
pos.z += mousy<<(4+pk_uedaccel);
|
|
|
|
move_and_update(xvect, yvect, 0);
|
|
}
|
|
else if ((bstatus&(1|2|4))==4)
|
|
{
|
|
zmode = 2;
|
|
|
|
// horiz-100 of 200 is viewing at 326.4 build angle units (=atan(200/128)) upward
|
|
tempint = getangle(128, horiz-100);
|
|
|
|
xvect = -((mousy*
|
|
((int32_t)sintable[(ang+2560)&2047]>>6)*
|
|
((int32_t)sintable[(tempint+512)&2047])>>6)
|
|
<<pk_uedaccel);
|
|
yvect = -((mousy*
|
|
((int32_t)sintable[(ang+2048)&2047]>>6)*
|
|
((int32_t)sintable[(tempint+512)&2047])>>6)
|
|
<<pk_uedaccel);
|
|
|
|
pos.z += mousy*(((int32_t)sintable[(tempint+2048)&2047])>>(10-pk_uedaccel));
|
|
|
|
move_and_update(xvect, yvect, 2);
|
|
}
|
|
}
|
|
|
|
if (mskip)
|
|
{
|
|
// mskip was set in astub.c to not trigger UEd mouse movements.
|
|
// Reset now.
|
|
mskip = 0;
|
|
}
|
|
else
|
|
{
|
|
if (mlook && (unrealedlook==0 || (bstatus&(1|4))==0))
|
|
{
|
|
ang += mousx;
|
|
horiz -= mousy;
|
|
|
|
/*
|
|
if (mousy && !(mousy/4))
|
|
horiz--;
|
|
if (mousx && !(mousx/2))
|
|
ang++;
|
|
*/
|
|
|
|
inpclamp(&horiz, -99, 299);
|
|
|
|
if (mlook == 1)
|
|
{
|
|
searchx = xdim>>1;
|
|
searchy = ydim>>1;
|
|
}
|
|
osearchx = searchx-mousx;
|
|
osearchy = searchy-mousy;
|
|
}
|
|
else if (unrealedlook==0 || (bstatus&(1|2|4))==0)
|
|
{
|
|
osearchx = searchx;
|
|
osearchy = searchy;
|
|
searchx += mousx;
|
|
searchy += mousy;
|
|
|
|
inpclamp(&searchx, 12, xdim-13);
|
|
inpclamp(&searchy, 12, ydim-13);
|
|
}
|
|
}
|
|
|
|
// showmouse();
|
|
|
|
if (keystatus[0x43]) // F9
|
|
{
|
|
if (mhk)
|
|
{
|
|
Bmemset(spriteext, 0, sizeof(spriteext_t) * MAXSPRITES);
|
|
Bmemset(spritesmooth, 0, sizeof(spritesmooth_t) * (MAXSPRITES+MAXUNIQHUDID));
|
|
delete_maphack_lights();
|
|
mhk = 0;
|
|
message("Maphacks disabled");
|
|
}
|
|
else
|
|
{
|
|
mhk = 1;
|
|
loadmhk(1);
|
|
}
|
|
|
|
keystatus[0x43] = 0;
|
|
}
|
|
|
|
mainloop_move();
|
|
|
|
getzrange(&pos,cursectnum, &hiz,&hihit, &loz,&lohit, 128, (m32_clipping==1)?0:CLIPMASK0);
|
|
/*
|
|
{
|
|
int32_t his = !(hihit&32768), los = !(lohit&32768);
|
|
if (m32_numdebuglines<64)
|
|
Bsprintf(m32_debugstr[m32_numdebuglines++], "s%d: cf[%s%d, %s%d] z(%d, %d)", cursectnum,
|
|
his?"s":"w",hihit&16383, los?"s":"w",lohit&16383, hiz,loz);
|
|
}
|
|
*/
|
|
oposz = pos.z;
|
|
if (zmode == 0)
|
|
{
|
|
goalz = loz-(kensplayerheight<<8); //playerheight pixels above floor
|
|
if (goalz < hiz+(16<<8)) //ceiling&floor too close
|
|
goalz = (loz+hiz)>>1;
|
|
goalz += mousz;
|
|
|
|
if (DOWN_BK(MOVEUP)) //A (stand high)
|
|
{
|
|
goalz -= (16<<8);
|
|
if (DOWN_BK(RUN))
|
|
goalz -= (24<<8);
|
|
}
|
|
if (DOWN_BK(MOVEDOWN)) //Z (stand low)
|
|
{
|
|
goalz += (12<<8);
|
|
if (DOWN_BK(RUN))
|
|
goalz += (12<<8);
|
|
}
|
|
|
|
if (goalz != pos.z)
|
|
{
|
|
if (pos.z < goalz) hvel += 64;
|
|
if (pos.z > goalz) hvel = ((goalz-pos.z)>>3);
|
|
|
|
pos.z += hvel;
|
|
if (pos.z > loz-(4<<8)) pos.z = loz-(4<<8), hvel = 0;
|
|
if (pos.z < hiz+(4<<8)) pos.z = hiz+(4<<8), hvel = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
goalz = pos.z;
|
|
if (DOWN_BK(MOVEUP)) //A
|
|
{
|
|
if (eitherCTRL)
|
|
{
|
|
horiz = max(-100,horiz-((DOWN_BK(RUN)+1)*synctics*2));
|
|
}
|
|
else
|
|
{
|
|
if (zmode != 1)
|
|
goalz -= (8<<8);
|
|
else
|
|
{
|
|
zlock += (4<<8);
|
|
DOWN_BK(MOVEUP) = 0;
|
|
}
|
|
}
|
|
}
|
|
if (DOWN_BK(MOVEDOWN)) //Z (stand low)
|
|
{
|
|
if (eitherCTRL)
|
|
{
|
|
horiz = min(300,horiz+((DOWN_BK(RUN)+1)*synctics*2));
|
|
}
|
|
else
|
|
{
|
|
if (zmode != 1)
|
|
goalz += (8<<8);
|
|
else if (zlock > 0)
|
|
{
|
|
zlock -= (4<<8);
|
|
DOWN_BK(MOVEDOWN) = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (m32_clipping)
|
|
inpclamp(&goalz, hiz+(4<<8), loz-(4<<8));
|
|
|
|
if (zmode == 1) goalz = loz-zlock;
|
|
if (m32_clipping && (goalz < hiz+(4<<8)))
|
|
goalz = ((loz+hiz)>>1); //ceiling&floor too close
|
|
if (zmode == 1) pos.z = goalz;
|
|
|
|
if (goalz != pos.z)
|
|
{
|
|
//if (pos.z < goalz) hvel += (32<<DOWN_BK(RUN));
|
|
//if (pos.z > goalz) hvel -= (32<<DOWN_BK(RUN));
|
|
if (pos.z < goalz)
|
|
hvel = ((192*synctics)<<DOWN_BK(RUN));
|
|
else
|
|
hvel = -((192*synctics)<<DOWN_BK(RUN));
|
|
|
|
pos.z += hvel;
|
|
|
|
if (m32_clipping)
|
|
{
|
|
if (pos.z > loz-(4<<8)) pos.z = loz-(4<<8), hvel = 0;
|
|
if (pos.z < hiz+(4<<8)) pos.z = hiz+(4<<8), hvel = 0;
|
|
}
|
|
}
|
|
else
|
|
hvel = 0;
|
|
}
|
|
|
|
{
|
|
int16_t ocursectnum = cursectnum;
|
|
updatesectorz(pos.x,pos.y,pos.z, &cursectnum);
|
|
if (cursectnum<0)
|
|
{
|
|
if (zmode != 2)
|
|
pos.z = oposz; // don't allow to fall into infinity when in void space
|
|
cursectnum = ocursectnum;
|
|
}
|
|
}
|
|
|
|
searchit = 2;
|
|
if (searchstat >= 0)
|
|
{
|
|
if ((bstatus&(1|2|4)) || keystatus[0x39]) // SPACE
|
|
searchit = 0;
|
|
|
|
if (keystatus[0x1f]) //S (insert sprite) (3D)
|
|
{
|
|
hitdata_t hit;
|
|
|
|
dax = 16384;
|
|
day = divscale14(searchx-(xdim>>1), xdim>>1);
|
|
rotatepoint(0,0, dax,day, ang, &dax,&day);
|
|
|
|
hitscan((const vec3_t *)&pos,cursectnum, //Start position
|
|
dax,day,(scale(searchy,200,ydim)-horiz)*2000, //vector of 3D ang
|
|
&hit,CLIPMASK1);
|
|
|
|
if (hit.sect >= 0)
|
|
{
|
|
dax = hit.pos.x;
|
|
day = hit.pos.y;
|
|
if (gridlock && grid > 0)
|
|
{
|
|
if (AIMING_AT_WALL || AIMING_AT_MASKWALL)
|
|
hit.pos.z &= 0xfffffc00;
|
|
else
|
|
locktogrid(&dax, &day);
|
|
}
|
|
|
|
i = insert_sprite_common(hit.sect, dax, day);
|
|
|
|
if (i < 0)
|
|
message("Couldn't insert sprite.");
|
|
else
|
|
{
|
|
int32_t cz, fz;
|
|
|
|
handle_sprite_in_clipboard(i);
|
|
|
|
spriteoncfz(i, &cz, &fz);
|
|
sprite[i].z = clamp2(hit.pos.z, cz, fz);
|
|
|
|
if (AIMING_AT_WALL || AIMING_AT_MASKWALL)
|
|
{
|
|
sprite[i].cstat &= ~48;
|
|
sprite[i].cstat |= (16+64);
|
|
|
|
correct_ornamented_sprite(i, hit.wall);
|
|
}
|
|
else
|
|
sprite[i].cstat |= (tilesiz[sprite[i].picnum].y>=32);
|
|
|
|
correct_sprite_yoffset(i);
|
|
|
|
asksave = 1;
|
|
|
|
VM_OnEvent(EVENT_INSERTSPRITE3D, i);
|
|
}
|
|
}
|
|
|
|
keystatus[0x1f] = 0;
|
|
}
|
|
|
|
if (keystatus[0x3f]||keystatus[0x40]) //F5,F6
|
|
{
|
|
switch (searchstat)
|
|
{
|
|
case SEARCH_CEILING:
|
|
case SEARCH_FLOOR:
|
|
CallExtShowSectorData(searchsector); break;
|
|
case SEARCH_WALL:
|
|
case SEARCH_MASKWALL:
|
|
CallExtShowWallData(searchwall); break;
|
|
case SEARCH_SPRITE:
|
|
CallExtShowSpriteData(searchwall); break;
|
|
}
|
|
|
|
keystatus[0x3f] = keystatus[0x40] = 0;
|
|
}
|
|
if (keystatus[0x41]||keystatus[0x42]) //F7,F8
|
|
{
|
|
switch (searchstat)
|
|
{
|
|
case SEARCH_CEILING:
|
|
case SEARCH_FLOOR:
|
|
CallExtEditSectorData(searchsector); break;
|
|
case SEARCH_WALL:
|
|
case SEARCH_MASKWALL:
|
|
CallExtEditWallData(searchwall); break;
|
|
case SEARCH_SPRITE:
|
|
CallExtEditSpriteData(searchwall); break;
|
|
}
|
|
|
|
keystatus[0x41] = keystatus[0x42] = 0;
|
|
}
|
|
|
|
}
|
|
|
|
if (keystatus[buildkeys[BK_MODE2D_3D]]) // Enter
|
|
{
|
|
|
|
vid_gamma_3d = vid_gamma;
|
|
vid_contrast_3d = vid_contrast;
|
|
vid_brightness_3d = vid_brightness;
|
|
|
|
vid_gamma = vid_contrast = 1.0;
|
|
vid_brightness = 0.0;
|
|
|
|
setbrightness(0,0,0);
|
|
|
|
keystatus[buildkeys[BK_MODE2D_3D]] = 0;
|
|
overheadeditor();
|
|
keystatus[buildkeys[BK_MODE2D_3D]] = 0;
|
|
|
|
vid_gamma = vid_gamma_3d;
|
|
vid_contrast = vid_contrast_3d;
|
|
vid_brightness = vid_brightness_3d;
|
|
|
|
vid_gamma_3d = vid_contrast_3d = vid_brightness_3d = -1;
|
|
|
|
setbrightness(GAMMA_CALC,0,0);
|
|
}
|
|
}
|
|
|
|
char changechar(char dachar, int32_t dadir, char smooshyalign, char boundcheck)
|
|
{
|
|
if (dadir < 0)
|
|
{
|
|
if ((dachar > 0) || (boundcheck == 0))
|
|
{
|
|
dachar--;
|
|
if (smooshyalign > 0)
|
|
dachar = (dachar&0xf8);
|
|
}
|
|
}
|
|
else if (dadir > 0)
|
|
{
|
|
if ((dachar < 255) || (boundcheck == 0))
|
|
{
|
|
dachar++;
|
|
if (smooshyalign > 0)
|
|
{
|
|
if (dachar >= 256-8) dachar = 255;
|
|
else dachar = ((dachar+7)&0xf8);
|
|
}
|
|
}
|
|
}
|
|
return(dachar);
|
|
}
|
|
|
|
|
|
////////////////////// OVERHEADEDITOR //////////////////////
|
|
|
|
// some 2d mode state
|
|
static struct overheadstate
|
|
{
|
|
// number of backed up drawn walls
|
|
int32_t bak_wallsdrawn;
|
|
|
|
// state related to line drawing
|
|
int16_t suckwall, split;
|
|
int16_t splitsect;
|
|
int16_t splitstartwall;
|
|
} ovh;
|
|
|
|
|
|
static int32_t inside_editor(const vec3_t *pos, int32_t searchx, int32_t searchy, int32_t zoom,
|
|
int32_t x, int32_t y, int16_t sectnum)
|
|
{
|
|
if (!m32_sideview)
|
|
return inside(x, y, sectnum);
|
|
|
|
// if in side-view mode, use the screen coords instead
|
|
{
|
|
int32_t dst = MAXSECTORS+M32_FIXME_SECTORS-1, i, oi;
|
|
int32_t srcw=sector[sectnum].wallptr, dstw=MAXWALLS;
|
|
int32_t ret;
|
|
|
|
if (sector[sectnum].wallnum > M32_FIXME_WALLS)
|
|
return -1;
|
|
|
|
Bmemcpy(§or[dst], §or[sectnum], sizeof(sectortype));
|
|
sector[dst].wallptr = dstw;
|
|
|
|
Bmemcpy(&wall[dstw], &wall[srcw], sector[dst].wallnum*sizeof(walltype));
|
|
for (i=dstw, oi=srcw; i<dstw+sector[dst].wallnum; i++, oi++)
|
|
{
|
|
wall[i].point2 += dstw-srcw;
|
|
|
|
screencoords(&wall[i].x, &wall[i].y, wall[i].x-pos->x, wall[i].y-pos->y, zoom);
|
|
wall[i].y += getscreenvdisp(getflorzofslope(sectnum,wall[oi].x,wall[oi].y)-pos->z, zoom);
|
|
wall[i].x += halfxdim16;
|
|
wall[i].y += midydim16;
|
|
}
|
|
|
|
i = numsectors;
|
|
numsectors = dst+1;
|
|
ret = inside(searchx, searchy, dst);
|
|
numsectors = i;
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
int32_t inside_editor_curpos(int16_t sectnum)
|
|
{
|
|
// TODO: take care: mous[xy]plc global vs overheadeditor auto
|
|
return inside_editor(&pos, searchx,searchy, zoom, mousxplc,mousyplc, sectnum);
|
|
}
|
|
|
|
|
|
static inline void drawline16base(int32_t bx, int32_t by, int32_t x1, int32_t y1, int32_t x2, int32_t y2, char col)
|
|
{
|
|
drawline16(bx+x1, by+y1, bx+x2, by+y2, col);
|
|
}
|
|
|
|
void drawsmallabel(const char *text, char col, char backcol, int32_t dax, int32_t day, int32_t daz)
|
|
{
|
|
int32_t x1, y1, x2, y2;
|
|
|
|
screencoords(&dax,&day, dax-pos.x,day-pos.y, zoom);
|
|
if (m32_sideview)
|
|
day += getscreenvdisp(daz-pos.z, zoom);
|
|
|
|
x1 = halfxdim16+dax-(Bstrlen(text)<<1);
|
|
y1 = midydim16+day-4;
|
|
x2 = x1 + (Bstrlen(text)<<2)+2;
|
|
y2 = y1 + 7;
|
|
|
|
if ((x1 > 3) && (x2 < xdim) && (y1 > 1) && (y2 < ydim16))
|
|
{
|
|
printext16(x1,y1, col,backcol, text,1);
|
|
drawline16(x1-1,y1-1, x2-3,y1-1, backcol);
|
|
drawline16(x1-1,y2+1, x2-3,y2+1, backcol);
|
|
|
|
drawline16(x1-2,y1, x1-2,y2, backcol);
|
|
drawline16(x2-2,y1, x2-2,y2, backcol);
|
|
drawline16(x2-3,y1, x2-3,y2, backcol);
|
|
}
|
|
}
|
|
|
|
// backup highlighted sectors with sprites as mapinfo for later restoration
|
|
// return values:
|
|
// -1: highlightsectorcnt<=0
|
|
// 0: ok
|
|
static int32_t backup_highlighted_map(mapinfofull_t *mapinfo)
|
|
{
|
|
int32_t i, j, k, m, tmpnumwalls=0, tmpnumsprites=0;
|
|
int16_t *const otonsect = (int16_t *)tempxyar; // STRICTALIASING
|
|
int16_t *const otonwall = ((int16_t *)tempxyar) + MAXWALLS;
|
|
#ifdef YAX_ENABLE
|
|
int16_t otonbunch[YAX_MAXBUNCHES];
|
|
int16_t numsectsofbunch[YAX_MAXBUNCHES]; // ceilings + floors
|
|
#endif
|
|
|
|
if (highlightsectorcnt <= 0)
|
|
return -1;
|
|
|
|
#ifdef YAX_ENABLE
|
|
for (i=0; i<numyaxbunches; i++)
|
|
numsectsofbunch[i] = 0;
|
|
#endif
|
|
|
|
// set up old-->new mappings
|
|
j = 0;
|
|
k = 0;
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
int32_t startwall, endwall;
|
|
|
|
if (hlsectorbitmap[i>>3]&(1<<(i&7)))
|
|
{
|
|
#ifdef YAX_ENABLE
|
|
int16_t bn[2], cf;
|
|
|
|
yax_getbunches(i, &bn[0], &bn[1]);
|
|
for (cf=0; cf<2; cf++)
|
|
if (bn[cf] >= 0)
|
|
numsectsofbunch[bn[cf]]++;
|
|
#endif
|
|
otonsect[i] = j++;
|
|
|
|
for (WALLS_OF_SECTOR(i, m))
|
|
otonwall[m] = k++;
|
|
}
|
|
else
|
|
{
|
|
otonsect[i] = -1;
|
|
|
|
for (WALLS_OF_SECTOR(i, m))
|
|
otonwall[m] = -1;
|
|
}
|
|
}
|
|
|
|
#ifdef YAX_ENABLE
|
|
j = 0;
|
|
for (i=0; i<numyaxbunches; i++)
|
|
{
|
|
// only back up complete bunches
|
|
if (numsectsofbunch[i] == yax_numsectsinbunch(i, 0)+yax_numsectsinbunch(i, 1))
|
|
otonbunch[i] = j++; // kept bunch
|
|
else
|
|
otonbunch[i] = -1; // discarded bunch
|
|
}
|
|
mapinfo->numyaxbunches = j;
|
|
#endif
|
|
|
|
// count walls & sprites
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
tmpnumwalls += sector[highlightsector[i]].wallnum;
|
|
|
|
m = headspritesect[highlightsector[i]];
|
|
while (m != -1)
|
|
{
|
|
tmpnumsprites++;
|
|
m = nextspritesect[m];
|
|
}
|
|
}
|
|
|
|
// allocate temp storage
|
|
mapinfo->sector = (tsectortype *)Xmalloc(highlightsectorcnt * sizeof(sectortype));
|
|
mapinfo->wall = (twalltype *)Xmalloc(tmpnumwalls * sizeof(walltype));
|
|
|
|
#ifdef YAX_ENABLE
|
|
if (mapinfo->numyaxbunches > 0)
|
|
{
|
|
mapinfo->bunchnum = (int16_t *)Xmalloc(highlightsectorcnt*2*sizeof(int16_t));
|
|
mapinfo->ynextwall = (int16_t *)Xmalloc(tmpnumwalls*2*sizeof(int16_t));
|
|
}
|
|
else
|
|
{
|
|
mapinfo->bunchnum = mapinfo->ynextwall = NULL;
|
|
}
|
|
#endif
|
|
|
|
if (tmpnumsprites>0)
|
|
{
|
|
mapinfo->sprite = (tspritetype *)Xmalloc(tmpnumsprites * sizeof(spritetype));
|
|
}
|
|
else
|
|
{
|
|
// would never be accessed because mapinfo->numsprites is 0, but cleaner
|
|
mapinfo->sprite = NULL;
|
|
}
|
|
|
|
|
|
// copy everything over
|
|
tmpnumwalls = 0;
|
|
tmpnumsprites = 0;
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
k = highlightsector[i];
|
|
Bmemcpy(&mapinfo->sector[i], §or[k], sizeof(sectortype));
|
|
mapinfo->sector[i].wallptr = tmpnumwalls;
|
|
|
|
#ifdef YAX_ENABLE
|
|
if (mapinfo->numyaxbunches > 0 || numyaxbunches > 0)
|
|
{
|
|
int16_t bn[2];
|
|
|
|
yax_getbunches(k, &bn[0], &bn[1]);
|
|
for (j=0; j<2; j++)
|
|
{
|
|
// old bunchnum, new bunchnum
|
|
int32_t obn=bn[j], nbn=(obn>=0) ? otonbunch[obn] : -1;
|
|
|
|
if (mapinfo->numyaxbunches > 0)
|
|
mapinfo->bunchnum[2*i + j] = nbn;
|
|
|
|
if (obn >= 0 && nbn < 0)
|
|
{
|
|
// A bunch was discarded.
|
|
tsectortype *const sec = &mapinfo->sector[i];
|
|
# if !defined NEW_MAP_FORMAT
|
|
uint16_t *const cs = j==YAX_CEILING ? &sec->ceilingstat : &sec->floorstat;
|
|
uint8_t *const xp = j==YAX_CEILING ? &sec->ceilingxpanning : &sec->floorxpanning;
|
|
|
|
*cs &= ~YAX_BIT;
|
|
*xp = 0;
|
|
# else
|
|
if (j == YAX_CEILING)
|
|
sec->ceilingbunch = -1;
|
|
else
|
|
sec->floorbunch = -1;
|
|
# endif
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
for (j=0; j<sector[k].wallnum; j++)
|
|
{
|
|
m = sector[k].wallptr;
|
|
Bmemcpy(&mapinfo->wall[tmpnumwalls+j], &wall[m+j], sizeof(walltype));
|
|
mapinfo->wall[tmpnumwalls+j].point2 += (tmpnumwalls-m);
|
|
|
|
#ifdef YAX_ENABLE
|
|
if (mapinfo->numyaxbunches > 0 || numyaxbunches > 0)
|
|
{
|
|
int32_t cf;
|
|
|
|
for (cf=0; cf<2; cf++)
|
|
{
|
|
const int32_t ynw = yax_getnextwall(m+j, cf);
|
|
const int32_t nynw = (ynw >= 0) ? otonwall[ynw] : -1;
|
|
|
|
if (mapinfo->numyaxbunches > 0)
|
|
mapinfo->ynextwall[2*(tmpnumwalls+j) + cf] = nynw;
|
|
|
|
if (ynw >= 0 && nynw < 0) // CLEAR_YNEXTWALLS
|
|
YAX_PTRNEXTWALL(mapinfo->wall, tmpnumwalls+j, cf) = YAX_NEXTWALLDEFAULT(cf);
|
|
}
|
|
}
|
|
#endif
|
|
m = mapinfo->wall[tmpnumwalls+j].nextsector;
|
|
if (m < 0 || otonsect[m] < 0)
|
|
{
|
|
mapinfo->wall[tmpnumwalls+j].nextsector = -1;
|
|
mapinfo->wall[tmpnumwalls+j].nextwall = -1;
|
|
}
|
|
else
|
|
{
|
|
mapinfo->wall[tmpnumwalls+j].nextsector = otonsect[m];
|
|
m = mapinfo->wall[tmpnumwalls+j].nextwall;
|
|
mapinfo->wall[tmpnumwalls+j].nextwall = otonwall[m];
|
|
}
|
|
}
|
|
tmpnumwalls += j;
|
|
|
|
m = headspritesect[highlightsector[i]];
|
|
while (m != -1)
|
|
{
|
|
Bmemcpy(&mapinfo->sprite[tmpnumsprites], &sprite[m], sizeof(spritetype));
|
|
mapinfo->sprite[tmpnumsprites].sectnum = otonsect[highlightsector[i]];
|
|
m = nextspritesect[m];
|
|
tmpnumsprites++;
|
|
}
|
|
}
|
|
|
|
|
|
mapinfo->numsectors = highlightsectorcnt;
|
|
mapinfo->numwalls = tmpnumwalls;
|
|
mapinfo->numsprites = tmpnumsprites;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void mapinfofull_free(mapinfofull_t *mapinfo)
|
|
{
|
|
Bfree(mapinfo->sector);
|
|
#ifdef YAX_ENABLE
|
|
if (mapinfo->numyaxbunches > 0)
|
|
{
|
|
Bfree(mapinfo->bunchnum);
|
|
Bfree(mapinfo->ynextwall);
|
|
}
|
|
#endif
|
|
Bfree(mapinfo->wall);
|
|
if (mapinfo->numsprites>0)
|
|
Bfree(mapinfo->sprite);
|
|
}
|
|
|
|
// restore map saved with backup_highlighted_map, also
|
|
// frees mapinfo's sector, wall, (sprite) in any case.
|
|
// return values:
|
|
// -1: limits exceeded
|
|
// 0: ok
|
|
// forreal: if 0, only test if we have enough space (same return values)
|
|
static int32_t restore_highlighted_map(mapinfofull_t *mapinfo, int32_t forreal)
|
|
{
|
|
int32_t i, j, onumsectors=numsectors, newnumsectors, newnumwalls;
|
|
|
|
if (numsectors+mapinfo->numsectors>MAXSECTORS || numwalls+mapinfo->numwalls>MAXWALLS
|
|
#ifdef YAX_ENABLE
|
|
|| numyaxbunches+mapinfo->numyaxbunches > YAX_MAXBUNCHES
|
|
#endif
|
|
|| Numsprites+mapinfo->numsprites>MAXSPRITES)
|
|
{
|
|
mapinfofull_free(mapinfo);
|
|
return -1;
|
|
}
|
|
|
|
if (!forreal)
|
|
return 0;
|
|
|
|
newnumsectors = numsectors + mapinfo->numsectors;
|
|
newnumwalls = numwalls + mapinfo->numwalls;
|
|
|
|
// copy sectors & walls
|
|
Bmemcpy(§or[numsectors], mapinfo->sector, mapinfo->numsectors*sizeof(sectortype));
|
|
Bmemcpy(&wall[numwalls], mapinfo->wall, mapinfo->numwalls*sizeof(walltype));
|
|
|
|
// tweak index members
|
|
for (i=numwalls; i<newnumwalls; i++)
|
|
{
|
|
wall[i].point2 += numwalls;
|
|
|
|
if (wall[i].nextsector >= 0)
|
|
{
|
|
wall[i].nextsector += numsectors;
|
|
wall[i].nextwall += numwalls;
|
|
}
|
|
#ifdef YAX_ENABLE
|
|
for (j=0; j<2; j++)
|
|
{
|
|
if (mapinfo->numyaxbunches > 0)
|
|
{
|
|
yax_setnextwall(i, j, mapinfo->ynextwall[2*(i-numwalls) + j]>=0 ?
|
|
numwalls+mapinfo->ynextwall[2*(i-numwalls) + j] : -1);
|
|
}
|
|
else
|
|
{
|
|
# if !defined NEW_MAP_FORMAT
|
|
// XXX: When copying a TROR portion into a non-TROR map (e.g. a
|
|
// new one), tags denoting ynextwalls are left in place.
|
|
wall[i].cstat &= ~YAX_NEXTWALLBIT(j); // CLEAR_YNEXTWALLS
|
|
# else
|
|
yax_setnextwall(i, j, -1);
|
|
# endif
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
for (i=numsectors; i<newnumsectors; i++)
|
|
sector[i].wallptr += numwalls;
|
|
|
|
// highlight copied sectors
|
|
|
|
numsectors = newnumsectors;
|
|
|
|
Bmemset(hlsectorbitmap, 0, sizeof(hlsectorbitmap));
|
|
for (i=onumsectors; i<newnumsectors; i++)
|
|
{
|
|
hlsectorbitmap[i>>3] |= (1<<(i&7));
|
|
|
|
#ifdef YAX_ENABLE
|
|
for (j=0; j<2; j++)
|
|
{
|
|
if (mapinfo->numyaxbunches > 0)
|
|
{
|
|
int32_t bn = mapinfo->bunchnum[2*(i-onumsectors)+j];
|
|
yax_setbunch(i, j, bn>=0 ? numyaxbunches+bn : -2);
|
|
// -2 clears forward yax-nextwall links.
|
|
// XXX: still may wrongly reset xpanning.
|
|
}
|
|
else
|
|
Bassert(yax_getbunch(i, j) < 0);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// insert sprites
|
|
for (i=0; i<mapinfo->numsprites; i++)
|
|
{
|
|
const tspritetype *srcspr = &mapinfo->sprite[i];
|
|
int32_t sect = onumsectors + srcspr->sectnum;
|
|
|
|
j = insertsprite(sect, srcspr->statnum);
|
|
Bassert(j >= 0);
|
|
Bmemcpy(&sprite[j], srcspr, sizeof(spritetype));
|
|
sprite[j].sectnum = sect;
|
|
}
|
|
|
|
mapinfofull_free(mapinfo);
|
|
|
|
numwalls = newnumwalls;
|
|
|
|
update_highlightsector();
|
|
|
|
#ifdef YAX_ENABLE
|
|
if (mapinfo->numyaxbunches > 0)
|
|
yax_update(0);
|
|
#endif
|
|
yax_updategrays(pos.z);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int32_t newnumwalls=-1;
|
|
|
|
void ovh_whiteoutgrab(int32_t restoreredwalls)
|
|
{
|
|
int32_t i, j, k, startwall, endwall;
|
|
#if 0
|
|
//def YAX_ENABLE
|
|
int16_t cb, fb;
|
|
#endif
|
|
|
|
if (restoreredwalls)
|
|
{
|
|
// restore onextwalls first
|
|
for (i=0; i<numsectors; i++)
|
|
for (WALLS_OF_SECTOR(i, j))
|
|
checksectorpointer(j, i);
|
|
}
|
|
|
|
for (i=0; i<MAXWALLS; i++)
|
|
onextwall[i] = -1;
|
|
|
|
//White out all bordering lines of grab that are
|
|
//not highlighted on both sides
|
|
for (i=highlightsectorcnt-1; i>=0; i--)
|
|
for (WALLS_OF_SECTOR(highlightsector[i], j))
|
|
{
|
|
if (wall[j].nextwall < 0)
|
|
continue;
|
|
|
|
k = wall[j].nextsector;
|
|
|
|
if (hlsectorbitmap[k>>3]&(1<<(k&7)))
|
|
continue;
|
|
#if 0
|
|
//def YAX_ENABLE
|
|
// internal red walls are kept red
|
|
yax_getbunches(highlightsector[i], &cb, &fb);
|
|
if (cb>=0 && yax_getbunch(k, YAX_CEILING)>=0)
|
|
continue;
|
|
if (fb>=0 && yax_getbunch(k, YAX_FLOOR)>=0)
|
|
continue;
|
|
#endif
|
|
onextwall[j] = wall[j].nextwall;
|
|
|
|
NEXTWALL(j).nextwall = -1;
|
|
NEXTWALL(j).nextsector = -1;
|
|
wall[j].nextwall = -1;
|
|
wall[j].nextsector = -1;
|
|
}
|
|
|
|
if (highlightsectorcnt > 0)
|
|
mkonwvalid();
|
|
else
|
|
mkonwinvalid_keeptempsect();
|
|
}
|
|
|
|
static void duplicate_selected_sectors(void)
|
|
{
|
|
mapinfofull_t mapinfo;
|
|
int32_t i, j, onumsectors;
|
|
#ifdef YAX_ENABLE
|
|
int32_t onumyaxbunches;
|
|
#endif
|
|
int32_t minx=INT32_MAX, maxx=INT32_MIN, miny=INT32_MAX, maxy=INT32_MIN, dx, dy;
|
|
|
|
i = backup_highlighted_map(&mapinfo);
|
|
|
|
if (i < 0)
|
|
{
|
|
message("Out of memory!");
|
|
return;
|
|
}
|
|
|
|
i = restore_highlighted_map(&mapinfo, 0);
|
|
if (i < 0)
|
|
{
|
|
// XXX: no, might be another limit too. Better message needed.
|
|
printmessage16("Copying sectors would exceed sector or wall limit.");
|
|
return;
|
|
}
|
|
|
|
// restoring would succeed, tweak things...
|
|
Bmemset(hlsectorbitmap, 0, sizeof(hlsectorbitmap));
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
int32_t startwall, endwall;
|
|
|
|
// first, make red lines of old selected sectors, effectively
|
|
// restoring the original state
|
|
for (WALLS_OF_SECTOR(highlightsector[i], j))
|
|
{
|
|
if (wall[j].nextwall >= 0)
|
|
checksectorpointer(wall[j].nextwall,wall[j].nextsector);
|
|
checksectorpointer(j, highlightsector[i]);
|
|
|
|
minx = min(minx, wall[j].x);
|
|
maxx = max(maxx, wall[j].x);
|
|
miny = min(miny, wall[j].y);
|
|
maxy = max(maxy, wall[j].y);
|
|
}
|
|
}
|
|
|
|
// displace walls & sprites of new sectors by a small amount:
|
|
// calculate displacement
|
|
if (grid>0 && grid<9)
|
|
dx = max(2048>>grid, 128);
|
|
else
|
|
dx = 512;
|
|
dy = -dx;
|
|
if (maxx+dx >= editorgridextent) dx*=-1;
|
|
if (minx+dx <= -editorgridextent) dx*=-1;
|
|
if (maxy+dy >= editorgridextent) dy*=-1;
|
|
if (miny+dy <= -editorgridextent) dy*=-1;
|
|
|
|
onumsectors = numsectors;
|
|
#ifdef YAX_ENABLE
|
|
onumyaxbunches = numyaxbunches;
|
|
#endif
|
|
// restore! this will not fail.
|
|
restore_highlighted_map(&mapinfo, 1);
|
|
|
|
// displace
|
|
for (i=onumsectors; i<numsectors; i++)
|
|
{
|
|
for (j=sector[i].wallptr; j<sector[i].wallptr+sector[i].wallnum; j++)
|
|
{
|
|
wall[j].x += dx;
|
|
wall[j].y += dy;
|
|
}
|
|
|
|
for (j=headspritesect[i]; j>=0; j=nextspritesect[j])
|
|
{
|
|
sprite[j].x += dx;
|
|
sprite[j].y += dy;
|
|
}
|
|
}
|
|
|
|
#ifdef YAX_ENABLE
|
|
if (numyaxbunches > onumyaxbunches)
|
|
printmessage16("Sectors duplicated, creating %d new bunches.", numyaxbunches-onumyaxbunches);
|
|
else
|
|
#endif
|
|
printmessage16("Sectors duplicated and stamped.");
|
|
asksave = 1;
|
|
|
|
#ifdef YAX_ENABLE
|
|
if (numyaxbunches > onumyaxbunches)
|
|
yax_update(0);
|
|
#endif
|
|
yax_updategrays(pos.z);
|
|
}
|
|
|
|
|
|
static void duplicate_selected_sprites(void)
|
|
{
|
|
int32_t i, j, k=0;
|
|
|
|
for (i=0; i<highlightcnt; i++)
|
|
if ((highlight[i]&0xc000) == 16384)
|
|
k++;
|
|
|
|
if (Numsprites + k <= MAXSPRITES)
|
|
{
|
|
for (i=0; i<highlightcnt; i++)
|
|
if ((highlight[i]&0xc000) == 16384)
|
|
{
|
|
//duplicate sprite
|
|
k = (highlight[i]&16383);
|
|
j = insertsprite(sprite[k].sectnum,sprite[k].statnum);
|
|
Bmemcpy(&sprite[j],&sprite[k],sizeof(spritetype));
|
|
// sprite[j].sectnum = sprite[k].sectnum; //Don't let memcpy overwrite sector!
|
|
// setsprite(j,(vec3_t *)&sprite[j]);
|
|
}
|
|
|
|
printmessage16("Sprites duplicated and stamped.");
|
|
asksave = 1;
|
|
}
|
|
else
|
|
{
|
|
printmessage16("Copying sprites would exceed sprite limit.");
|
|
}
|
|
}
|
|
|
|
static void correct_ornamented_sprite(int32_t i, int32_t hitw)
|
|
{
|
|
int32_t j;
|
|
|
|
if (hitw >= 0)
|
|
{
|
|
sprite[i].ang = (getangle(POINT2(hitw).x-wall[hitw].x,
|
|
POINT2(hitw).y-wall[hitw].y)+512)&2047;
|
|
|
|
//Make sure sprite's in right sector
|
|
if (inside(sprite[i].x, sprite[i].y, sprite[i].sectnum) != 1)
|
|
{
|
|
j = wall[hitw].point2;
|
|
sprite[i].x -= ksgn(wall[j].y-wall[hitw].y);
|
|
sprite[i].y += ksgn(wall[j].x-wall[hitw].x);
|
|
}
|
|
}
|
|
}
|
|
|
|
void DoSpriteOrnament(int32_t i)
|
|
{
|
|
hitdata_t hit;
|
|
|
|
hitscan((const vec3_t *)&sprite[i],sprite[i].sectnum,
|
|
sintable[(sprite[i].ang+1536)&2047],
|
|
sintable[(sprite[i].ang+1024)&2047],
|
|
0,
|
|
&hit,CLIPMASK1);
|
|
|
|
if (hit.sect == -1)
|
|
return;
|
|
|
|
sprite[i].x = hit.pos.x;
|
|
sprite[i].y = hit.pos.y;
|
|
sprite[i].z = hit.pos.z;
|
|
changespritesect(i, hit.sect);
|
|
|
|
correct_ornamented_sprite(i, hit.wall);
|
|
}
|
|
|
|
void update_highlight(void)
|
|
{
|
|
int32_t i;
|
|
|
|
highlightcnt = 0;
|
|
for (i=0; i<numwalls; i++)
|
|
if (show2dwall[i>>3]&(1<<(i&7)))
|
|
highlight[highlightcnt++] = i;
|
|
for (i=0; i<MAXSPRITES; i++)
|
|
if (sprite[i].statnum < MAXSTATUS)
|
|
{
|
|
if (show2dsprite[i>>3]&(1<<(i&7)))
|
|
highlight[highlightcnt++] = i+16384;
|
|
}
|
|
else
|
|
show2dsprite[i>>3] &= ~(1<<(i&7));
|
|
|
|
if (highlightcnt == 0)
|
|
highlightcnt = -1;
|
|
}
|
|
|
|
void update_highlightsector(void)
|
|
{
|
|
int32_t i;
|
|
|
|
minhlsectorfloorz = INT32_MAX;
|
|
numhlsecwalls = 0;
|
|
|
|
highlightsectorcnt = 0;
|
|
for (i=0; i<numsectors; i++)
|
|
if (hlsectorbitmap[i>>3]&(1<<(i&7)))
|
|
{
|
|
highlightsector[highlightsectorcnt++] = i;
|
|
minhlsectorfloorz = min(minhlsectorfloorz, sector[i].floorz);
|
|
numhlsecwalls += sector[i].wallnum;
|
|
}
|
|
|
|
if (highlightsectorcnt==0)
|
|
{
|
|
minhlsectorfloorz = 0;
|
|
highlightsectorcnt = -1;
|
|
}
|
|
}
|
|
|
|
// Get average point of sectors
|
|
static void get_sectors_center(const int16_t *sectors, int32_t numsecs, int32_t *cx, int32_t *cy)
|
|
{
|
|
int32_t i, j, k=0, dax = 0, day = 0;
|
|
int32_t startwall, endwall;
|
|
|
|
for (i=0; i<numsecs; i++)
|
|
{
|
|
for (WALLS_OF_SECTOR(sectors[i], j))
|
|
{
|
|
dax += wall[j].x;
|
|
day += wall[j].y;
|
|
k++;
|
|
}
|
|
}
|
|
|
|
if (k > 0)
|
|
{
|
|
dax /= k;
|
|
day /= k;
|
|
}
|
|
|
|
*cx = dax;
|
|
*cy = day;
|
|
}
|
|
|
|
static int32_t insert_sprite_common(int32_t sectnum, int32_t dax, int32_t day)
|
|
{
|
|
int32_t i, j, k;
|
|
|
|
i = insertsprite(sectnum,0);
|
|
if (i < 0)
|
|
return -1;
|
|
|
|
sprite[i].x = dax, sprite[i].y = day;
|
|
sprite[i].cstat = DEFAULT_SPRITE_CSTAT;
|
|
sprite[i].shade = 0;
|
|
sprite[i].pal = 0;
|
|
sprite[i].xrepeat = 64, sprite[i].yrepeat = 64;
|
|
sprite[i].xoffset = 0, sprite[i].yoffset = 0;
|
|
sprite[i].ang = 1536;
|
|
sprite[i].xvel = 0; sprite[i].yvel = 0; sprite[i].zvel = 0;
|
|
sprite[i].owner = -1;
|
|
sprite[i].clipdist = 32;
|
|
sprite[i].lotag = 0;
|
|
sprite[i].hitag = 0;
|
|
sprite[i].extra = -1;
|
|
|
|
Bmemset(localartfreq, 0, sizeof(localartfreq));
|
|
for (k=0; k<MAXSPRITES; k++)
|
|
if (sprite[k].statnum < MAXSTATUS && k!=i)
|
|
localartfreq[sprite[k].picnum]++;
|
|
|
|
j = 0;
|
|
for (k=0; k<MAXTILES; k++)
|
|
if (localartfreq[k] > localartfreq[j])
|
|
j = k;
|
|
|
|
if (localartfreq[j] > 0)
|
|
sprite[i].picnum = j;
|
|
else
|
|
sprite[i].picnum = 0;
|
|
|
|
return i;
|
|
}
|
|
|
|
void correct_sprite_yoffset(int32_t i)
|
|
{
|
|
int32_t tileyofs = picanm[sprite[i].picnum].yofs;
|
|
int32_t tileysiz = tilesiz[sprite[i].picnum].y;
|
|
|
|
if (klabs(tileyofs) >= tileysiz)
|
|
{
|
|
tileyofs *= -1;
|
|
if (tileyofs == 128)
|
|
tileyofs = 127;
|
|
|
|
sprite[i].yoffset = tileyofs;
|
|
}
|
|
else
|
|
sprite[i].yoffset = 0;
|
|
}
|
|
|
|
// keepcol >= 0 && <256: keep that idx-color
|
|
// keepcol < 0: keep none
|
|
// keepcol >= 256: 0x00ffffff is mask for 3 colors
|
|
void fade_editor_screen(int32_t keepcol)
|
|
{
|
|
char blackcol=0, greycol=whitecol-25, *cp;
|
|
int32_t pix, i, threecols = (keepcol >= 256);
|
|
char cols[3] = {(char)(keepcol&0xff), (char)((keepcol>>8)&0xff), (char)((keepcol>>16)&0xff)};
|
|
|
|
begindrawing();
|
|
cp = (char *)frameplace;
|
|
for (i=0; i<bytesperline*(ydim-STATUS2DSIZ2); i++, cp++)
|
|
{
|
|
pix = (uint8_t)(*cp);
|
|
|
|
if (!threecols && pix == keepcol)
|
|
continue;
|
|
if (threecols)
|
|
if (pix==cols[0] || pix==cols[1] || pix==cols[2])
|
|
continue;
|
|
|
|
if (*cp==greycol)
|
|
*cp = blackcol;
|
|
else if (*cp != blackcol)
|
|
*cp = greycol;
|
|
}
|
|
enddrawing();
|
|
showframe(1);
|
|
}
|
|
|
|
static void copy_some_wall_members(int16_t dst, int16_t src, int32_t reset_some)
|
|
{
|
|
static twalltype nullwall;
|
|
walltype * const dstwal = &wall[dst];
|
|
const twalltype *srcwal = src >= 0 ? (twalltype *)&wall[src] : &nullwall;
|
|
|
|
memset(&nullwall, 0, sizeof(nullwall));
|
|
nullwall.yrepeat = 8;
|
|
nullwall.extra = -1;
|
|
|
|
if (reset_some)
|
|
{
|
|
dstwal->cstat = srcwal->cstat;
|
|
}
|
|
else
|
|
{
|
|
dstwal->cstat &= ~(4+8+256);
|
|
dstwal->cstat |= (srcwal->cstat&(4+8+256));
|
|
}
|
|
dstwal->shade = srcwal->shade;
|
|
dstwal->yrepeat = srcwal->yrepeat;
|
|
fixrepeats(dst); // xrepeat
|
|
dstwal->picnum = srcwal->picnum;
|
|
dstwal->overpicnum = srcwal->overpicnum;
|
|
|
|
dstwal->pal = srcwal->pal;
|
|
dstwal->xpanning = srcwal->xpanning;
|
|
dstwal->ypanning = srcwal->ypanning;
|
|
|
|
if (reset_some)
|
|
{
|
|
dstwal->nextwall = -1;
|
|
dstwal->nextsector = -1;
|
|
|
|
dstwal->lotag = 0; //srcwal->lotag;
|
|
dstwal->hitag = 0; //srcwal->hitag;
|
|
dstwal->extra = -1; //srcwal->extra;
|
|
#ifdef YAX_ENABLE
|
|
yax_setnextwall(dst, YAX_CEILING, -1);
|
|
yax_setnextwall(dst, YAX_FLOOR, -1);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
static void init_new_wall1(int16_t *suckwall_ret, int32_t mousxplc, int32_t mousyplc)
|
|
{
|
|
int32_t i;
|
|
|
|
Bmemset(&wall[newnumwalls], 0, sizeof(walltype));
|
|
wall[newnumwalls].extra = -1;
|
|
|
|
wall[newnumwalls].x = mousxplc;
|
|
wall[newnumwalls].y = mousyplc;
|
|
wall[newnumwalls].nextsector = -1;
|
|
wall[newnumwalls].nextwall = -1;
|
|
|
|
for (i=0; i<numwalls; i++)
|
|
{
|
|
YAX_SKIPWALL(i);
|
|
if (wall[i].nextwall >= 0)
|
|
YAX_SKIPWALL(wall[i].nextwall);
|
|
|
|
if (wall[i].x == mousxplc && wall[i].y == mousyplc)
|
|
*suckwall_ret = i;
|
|
}
|
|
|
|
wall[newnumwalls].point2 = newnumwalls+1;
|
|
newnumwalls++;
|
|
}
|
|
|
|
// helpers for often needed ops:
|
|
static int32_t do_while_copyloop1(int16_t startwall, int16_t endwall,
|
|
int16_t *danumwalls, int16_t lastpoint2)
|
|
{
|
|
int32_t m = startwall;
|
|
|
|
do
|
|
{
|
|
if (*danumwalls >= MAXWALLS + M32_FIXME_WALLS)
|
|
return 1;
|
|
|
|
Bmemcpy(&wall[*danumwalls], &wall[m], sizeof(walltype));
|
|
wall[*danumwalls].point2 = *danumwalls+1;
|
|
(*danumwalls)++;
|
|
m = wall[m].point2;
|
|
}
|
|
while (m != endwall);
|
|
|
|
if (lastpoint2 >= 0)
|
|
wall[(*danumwalls)-1].point2 = lastpoint2;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void updatesprite1(int16_t i)
|
|
{
|
|
setsprite(i, (vec3_t *)&sprite[i]);
|
|
|
|
if (sprite[i].sectnum>=0)
|
|
{
|
|
int32_t cz, fz;
|
|
spriteoncfz(i, &cz, &fz);
|
|
inpclamp(&sprite[i].z, cz, fz);
|
|
}
|
|
}
|
|
|
|
#ifdef YAX_ENABLE
|
|
// highlighted OR grayed-out sectors:
|
|
static uint8_t hlorgraysectbitmap[MAXSECTORS>>3];
|
|
static int32_t ask_above_or_below(void);
|
|
#else
|
|
# define hlorgraysectbitmap hlsectorbitmap
|
|
#endif
|
|
|
|
// returns:
|
|
// 0: continue
|
|
// >0: newnumwalls
|
|
// <0: error
|
|
// ignore_ret and refsect_ret are for the 'auto-red-wall' feature
|
|
static int32_t trace_loop(int32_t j, uint8_t *visitedwall, int16_t *ignore_ret, int16_t *refsect_ret,
|
|
int16_t trace_loop_yaxcf)
|
|
{
|
|
int16_t refsect, ignore;
|
|
int32_t k, n, refwall;
|
|
#if 0
|
|
//def YAX_ENABLE
|
|
int32_t yaxp = (ignore_ret==NULL); // bleh
|
|
#else
|
|
UNREFERENCED_PARAMETER(trace_loop_yaxcf);
|
|
#endif
|
|
|
|
if (wall[j].nextwall>=0 || (visitedwall[j>>3]&(1<<(j&7))))
|
|
return 0;
|
|
|
|
n=2*MAXWALLS; // simple inf loop check
|
|
refwall = j;
|
|
k = numwalls;
|
|
|
|
ignore = 0;
|
|
|
|
if (ignore_ret)
|
|
{
|
|
refsect = -1;
|
|
updatesectorexclude(wall[j].x, wall[j].y, &refsect, hlorgraysectbitmap);
|
|
if (refsect<0)
|
|
return -1;
|
|
}
|
|
|
|
do
|
|
{
|
|
if (j!=refwall && visitedwall[j>>3]&(1<<(j&7)))
|
|
ignore = 1;
|
|
visitedwall[j>>3] |= (1<<(j&7));
|
|
|
|
if (ignore_ret)
|
|
{
|
|
if (inside(wall[j].x, wall[j].y, refsect) != 1)
|
|
ignore = 1;
|
|
}
|
|
|
|
if (!ignore)
|
|
{
|
|
if (k>=MAXWALLS)
|
|
{
|
|
message("Wall limits exceeded while tracing outer loop.");
|
|
return -2;
|
|
}
|
|
|
|
if (ignore_ret) // auto-red wall feature
|
|
onextwall[k] = onextwall[j];
|
|
|
|
Bmemcpy(&wall[k], &wall[j], sizeof(walltype));
|
|
wall[k].point2 = k+1;
|
|
// TODO: protect lotag/extra; see also hl-sector copying stuff
|
|
wall[k].nextsector = wall[k].nextwall = wall[k].extra = -1;
|
|
#ifdef YAX_ENABLE
|
|
if (trace_loop_yaxcf >= 0)
|
|
yax_setnextwall(k, trace_loop_yaxcf, j);
|
|
#endif
|
|
k++;
|
|
}
|
|
|
|
j = wall[j].point2;
|
|
n--;
|
|
|
|
while (wall[j].nextwall>=0 && n>0)
|
|
{
|
|
#if 0
|
|
//def YAX_ENABLE
|
|
if (yaxp)
|
|
{
|
|
int32_t ns = wall[j].nextsector;
|
|
if ((hlsectorbitmap[ns>>3]&(1<<(ns&7)))==0)
|
|
break;
|
|
}
|
|
#endif
|
|
j = wall[wall[j].nextwall].point2;
|
|
// if (j!=refwall && (visitedwall[j>>3]&(1<<(j&7))))
|
|
// ignore = 1;
|
|
// visitedwall[j>>3] |= (1<<(j&7));
|
|
n--;
|
|
}
|
|
}
|
|
while (j!=refwall && n>0);
|
|
|
|
if (j!=refwall)
|
|
{
|
|
message("internal error while tracing outer loop: didn't reach refwall");
|
|
return -3;
|
|
}
|
|
|
|
if (ignore_ret)
|
|
{
|
|
*ignore_ret = ignore;
|
|
if (refsect_ret)
|
|
*refsect_ret = refsect;
|
|
}
|
|
|
|
return k;
|
|
}
|
|
|
|
// Backup drawn walls for carrying out other operations in the middle.
|
|
// 0: back up, set newnumwalls to -1
|
|
// 1: restore drawn walls and free mem
|
|
// 2: only free memory needed for backing up walls but don't restore walls
|
|
// (use this if the map has been mangled too much for a safe restoration)
|
|
// Context that needs special treatment: suckwall, splitsect, splitstartwall
|
|
static int32_t backup_drawn_walls(int32_t restore)
|
|
{
|
|
static twalltype *tmpwall;
|
|
|
|
// back up
|
|
if (restore==0)
|
|
{
|
|
// ovh.bak_wallsdrawn should be 0 here
|
|
|
|
if (newnumwalls != -1)
|
|
{
|
|
if (newnumwalls <= numwalls) // shouldn't happen
|
|
return 2;
|
|
|
|
Bfree(tmpwall);
|
|
tmpwall = (twalltype *)Xmalloc((newnumwalls-numwalls) * sizeof(walltype));
|
|
|
|
ovh.bak_wallsdrawn = newnumwalls-numwalls;
|
|
|
|
Bmemcpy(tmpwall, &wall[numwalls], ovh.bak_wallsdrawn*sizeof(walltype));
|
|
newnumwalls = -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// restore/clear
|
|
if (tmpwall)
|
|
{
|
|
if (restore==1) // really restore
|
|
{
|
|
const int32_t nnumwalls = numwalls + ovh.bak_wallsdrawn;
|
|
|
|
if (nnumwalls < MAXWALLS) // else, silently discard drawn walls
|
|
{
|
|
int32_t i;
|
|
|
|
Bmemcpy(&wall[numwalls], tmpwall, ovh.bak_wallsdrawn*sizeof(walltype));
|
|
|
|
newnumwalls = nnumwalls;
|
|
for (i=numwalls; i<newnumwalls; i++)
|
|
wall[i].point2 = i+1;
|
|
}
|
|
}
|
|
|
|
Bfree(tmpwall);
|
|
tmpwall = NULL;
|
|
|
|
ovh.bak_wallsdrawn = 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// VARIOUS RESETTING FUNCTIONS
|
|
#define RESET_EDITOR_VARS() do { \
|
|
sectorhighlightstat = -1; \
|
|
newnumwalls = -1; \
|
|
joinsector[0] = -1; \
|
|
circlewall = -1; \
|
|
circlepoints = 7; \
|
|
} while (0)
|
|
|
|
void reset_highlightsector(void)
|
|
{
|
|
Bmemset(hlsectorbitmap, 0, sizeof(hlsectorbitmap));
|
|
update_highlightsector();
|
|
}
|
|
|
|
void reset_highlight(void) // walls and sprites
|
|
{
|
|
Bmemset(show2dwall, 0, sizeof(show2dwall));
|
|
Bmemset(show2dsprite, 0, sizeof(show2dsprite));
|
|
update_highlight();
|
|
}
|
|
|
|
#ifdef YAX_ENABLE
|
|
static int32_t collnumsects[2];
|
|
static int16_t collsectlist[2][MAXSECTORS];
|
|
static uint8_t collsectbitmap[2][MAXSECTORS>>3];
|
|
|
|
static void collect_sectors1(int16_t *sectlist, uint8_t *sectbitmap, int32_t *numsectptr,
|
|
int16_t startsec, int32_t alsoyaxnext, int32_t alsoonw)
|
|
{
|
|
int32_t j, startwall, endwall, sectcnt;
|
|
|
|
bfirst_search_init(sectlist, sectbitmap, numsectptr, MAXSECTORS, startsec);
|
|
|
|
for (sectcnt=0; sectcnt<*numsectptr; sectcnt++)
|
|
{
|
|
for (WALLS_OF_SECTOR(sectlist[sectcnt], j))
|
|
{
|
|
if (wall[j].nextsector >= 0)
|
|
bfirst_search_try(sectlist, sectbitmap, numsectptr, wall[j].nextsector);
|
|
else if (alsoonw && onextwall[j]>=0)
|
|
bfirst_search_try(sectlist, sectbitmap, numsectptr, sectorofwall(onextwall[j]));
|
|
}
|
|
|
|
if (alsoyaxnext)
|
|
{
|
|
int16_t bn[2], cf;
|
|
yax_getbunches(sectlist[sectcnt], &bn[0], &bn[1]);
|
|
for (cf=0; cf<2; cf++)
|
|
if (bn[cf]>=0)
|
|
{
|
|
for (SECTORS_OF_BUNCH(bn[cf], !cf, j))
|
|
bfirst_search_try(sectlist, sectbitmap, numsectptr, j);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static int32_t sectors_components(int16_t hlsectcnt, const int16_t *hlsectors, int32_t alsoyaxnext, int32_t alsoonw);
|
|
static int32_t highlighted_sectors_components(int32_t alsoyaxnext, int32_t alsoonw)
|
|
{
|
|
return sectors_components(highlightsectorcnt, highlightsector, alsoyaxnext, alsoonw);
|
|
}
|
|
|
|
// whether all highlighted sectors are in one (returns 1), two (2)
|
|
// or more (>2) connected components wrt the nextsector relation
|
|
// -1 means error
|
|
// alsoyaxnext: also consider "yax-nextsector" relation
|
|
// alsoonw: also consider "old-nextwall" relation (must be valid)
|
|
static int32_t sectors_components(int16_t hlsectcnt, const int16_t *hlsector, int32_t alsoyaxnext, int32_t alsoonw)
|
|
{
|
|
int32_t j, k, tmp;
|
|
|
|
if (hlsectcnt<1)
|
|
return 0;
|
|
|
|
collect_sectors1(collsectlist[0], collsectbitmap[0], &collnumsects[0],
|
|
hlsector[0], alsoyaxnext, alsoonw);
|
|
|
|
for (k=1; k<hlsectcnt; k++)
|
|
{
|
|
j = hlsector[k];
|
|
if ((collsectbitmap[0][j>>3]&(1<<(j&7)))==0)
|
|
{
|
|
// sector j not collected --> more than 1 conn. comp.
|
|
collect_sectors1(collsectlist[1], collsectbitmap[1], &collnumsects[1],
|
|
j, alsoyaxnext, alsoonw);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (k == hlsectcnt)
|
|
return 1;
|
|
|
|
for (k=0; k<hlsectcnt; k++)
|
|
{
|
|
j = hlsector[k];
|
|
tmp = (((collsectbitmap[0][j>>3]&(1<<(j&7)))!=0) + (((collsectbitmap[1][j>>3]&(1<<(j&7)))!=0)<<1));
|
|
|
|
if (tmp==3)
|
|
return -1; // components only weakly connected
|
|
|
|
if (tmp==0)
|
|
return 3; // sector j not reached
|
|
}
|
|
|
|
return 2;
|
|
}
|
|
|
|
static int cmpgeomwal1(const void *w1, const void *w2)
|
|
{
|
|
twalltype const * const wal1 = (twalltype *)&wall[B_UNBUF16(w1)];
|
|
twalltype const * const wal2 = (twalltype *)&wall[B_UNBUF16(w2)];
|
|
|
|
if (wal1->x == wal2->x)
|
|
return wal1->y - wal2->y;
|
|
|
|
return wal1->x - wal2->x;
|
|
}
|
|
|
|
static void sort_walls_geometrically(int16_t *wallist, int32_t nwalls)
|
|
{
|
|
qsort(wallist, nwalls, sizeof(int16_t), &cmpgeomwal1);
|
|
}
|
|
#endif
|
|
|
|
void SetFirstWall(int32_t sectnum, int32_t wallnum, int32_t alsoynw)
|
|
{
|
|
#ifdef YAX_ENABLE
|
|
int32_t i, j, k=0;
|
|
#endif
|
|
const sectortype *sec = §or[sectnum];
|
|
|
|
if (sec->wallptr == wallnum)
|
|
{
|
|
message("Wall %d already first wall of sector %d", wallnum, sectnum);
|
|
return;
|
|
}
|
|
|
|
#ifdef YAX_ENABLE
|
|
if (alsoynw)
|
|
{
|
|
// Also consider upper/lower TROR neighbor walls.
|
|
int32_t startwall, endwall;
|
|
int16_t cf;
|
|
|
|
for (i=0; i<numwalls; i++)
|
|
wall[i].cstat &= ~(1<<14);
|
|
|
|
for (cf=0; cf<2; cf++)
|
|
{
|
|
int16_t bunchnum;
|
|
int32_t tempsect=sectnum, tempwall=wallnum;
|
|
|
|
while ((bunchnum = yax_getbunch(tempsect, cf)) >= 0 &&
|
|
(tempsect=yax_is121(bunchnum, cf)) >= 0)
|
|
{
|
|
tempwall = yax_getnextwall(tempwall, cf);
|
|
if (tempwall < 0)
|
|
break; // corrupt!
|
|
wall[tempwall].cstat |= (1<<14);
|
|
}
|
|
}
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
for (WALLS_OF_SECTOR(i, j))
|
|
{
|
|
if (wall[j].cstat & (1<<14))
|
|
{
|
|
setfirstwall(i, j);
|
|
k++;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Only consider aimed at wall <wallnum>.
|
|
int16_t cb = yax_getbunch(sectnum, YAX_CEILING);
|
|
int16_t fb = yax_getbunch(sectnum, YAX_FLOOR);
|
|
|
|
if ((cb>=0 && (sec->ceilingstat&2)) || (fb >= 0 && (sec->floorstat&2)))
|
|
{
|
|
message("Extended ceilings/floors must not be sloped to set first wall");
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (k > 0)
|
|
message("Set first walls (sector[].wallptr) for %d sectors", k+1);
|
|
|
|
if (k == 0)
|
|
#endif
|
|
message("This wall now sector %d's first wall (sector[].wallptr)", sectnum);
|
|
|
|
setfirstwall(sectnum, wallnum);
|
|
|
|
mkonwinvalid_keeptempsect();
|
|
|
|
asksave = 1;
|
|
}
|
|
|
|
static void handlesecthighlight1(int32_t i, int32_t sub, int32_t nograycheck)
|
|
{
|
|
int32_t j;
|
|
|
|
if (sub)
|
|
{
|
|
hlsectorbitmap[i>>3] &= ~(1<<(i&7));
|
|
for (j=sector[i].wallptr; j<sector[i].wallptr+sector[i].wallnum; j++)
|
|
{
|
|
if (wall[j].nextwall >= 0)
|
|
checksectorpointer(wall[j].nextwall,wall[j].nextsector);
|
|
checksectorpointer(j, i);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (nograycheck || (graysectbitmap[i>>3]&(1<<(i&7)))==0)
|
|
hlsectorbitmap[i>>3] |= (1<<(i&7));
|
|
}
|
|
}
|
|
|
|
#ifdef YAX_ENABLE
|
|
// 1: good, 0: bad
|
|
static int32_t hl_all_bunch_sectors_p()
|
|
{
|
|
uint8_t *const havebunch = visited;
|
|
int16_t cf, cb, fb;
|
|
int32_t i, j;
|
|
|
|
if (numyaxbunches > 0)
|
|
{
|
|
Bmemset(havebunch, 0, (numyaxbunches+7)>>3);
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
yax_getbunches(highlightsector[i], &cb, &fb);
|
|
if (cb>=0)
|
|
havebunch[cb>>3] |= (1<<(cb&7));
|
|
if (fb>=0)
|
|
havebunch[fb>>3] |= (1<<(fb&7));
|
|
}
|
|
|
|
for (i=0; i<numyaxbunches; i++)
|
|
{
|
|
if ((havebunch[i>>3] & (1<<(i&7)))==0)
|
|
continue;
|
|
|
|
for (cf=0; cf<2; cf++)
|
|
for (SECTORS_OF_BUNCH(i,cf, j))
|
|
if ((hlsectorbitmap[j>>3]&(1<<(j&7)))==0)
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
#endif
|
|
|
|
static int32_t find_nextwall(int32_t sectnum, int32_t sectnum2)
|
|
{
|
|
int32_t j, startwall, endwall;
|
|
|
|
if (sectnum<0 || sectnum2<0)
|
|
return -1;
|
|
|
|
for (WALLS_OF_SECTOR(sectnum, j))
|
|
if (wall[j].nextsector == sectnum2)
|
|
return j;
|
|
|
|
return -1;
|
|
}
|
|
|
|
static int32_t bakframe_fillandfade(char **origframeptr, int32_t sectnum, const char *querystr)
|
|
{
|
|
if (!*origframeptr)
|
|
{
|
|
*origframeptr = (char *)Xmalloc(xdim*ydim);
|
|
|
|
begindrawing();
|
|
Bmemcpy(*origframeptr, (char *)frameplace, xdim*ydim);
|
|
enddrawing();
|
|
}
|
|
else
|
|
{
|
|
begindrawing();
|
|
Bmemcpy((char *)frameplace, *origframeptr, xdim*ydim);
|
|
enddrawing();
|
|
}
|
|
|
|
fillsector(sectnum, editorcolors[9]);
|
|
fade_editor_screen(editorcolors[9]);
|
|
|
|
return ask_if_sure(querystr, 0);
|
|
}
|
|
|
|
// High-level insert point, handles TROR constrained walls too
|
|
// onewnumwalls: old numwalls + drawn walls.
|
|
// <mapwallnum>: see insertpoint()
|
|
// Returns:
|
|
// 0 if wall limit would be reached.
|
|
// 1 if inserted point on a plain white or 2 points on a plain red wall.
|
|
// N >= 2 if inserted N points on TROR-constrained wall.
|
|
// N|(EXPECTED<<16) if inserted N points but EXPECTED walls were expected.
|
|
static int32_t M32_InsertPoint(int32_t thewall, int32_t dax, int32_t day, int32_t onewnumwalls, int32_t *mapwallnum)
|
|
{
|
|
#ifdef YAX_ENABLE
|
|
int32_t nextw = wall[thewall].nextwall;
|
|
int32_t i, j, k, m, tmpcf;
|
|
|
|
if (yax_islockedwall(thewall) || (nextw>=0 && yax_islockedwall(nextw)))
|
|
{
|
|
// yax'ed wall -- first find out which walls are affected
|
|
for (i=0; i<numwalls; i++)
|
|
wall[i].cstat &= ~(1<<14);
|
|
|
|
// round 1
|
|
for (YAX_ITER_WALLS(thewall, i, tmpcf))
|
|
wall[i].cstat |= (1<<14);
|
|
if (nextw >= 0)
|
|
for (YAX_ITER_WALLS(nextw, i, tmpcf))
|
|
wall[i].cstat |= (1<<14);
|
|
// round 2 (enough?)
|
|
for (YAX_ITER_WALLS(thewall, i, tmpcf))
|
|
if (wall[i].nextwall >= 0 && (wall[wall[i].nextwall].cstat&(1<<14))==0)
|
|
wall[wall[i].nextwall].cstat |= (1<<14);
|
|
if (nextw >= 0)
|
|
for (YAX_ITER_WALLS(nextw, i, tmpcf))
|
|
if (wall[i].nextwall >= 0 && (wall[wall[i].nextwall].cstat&(1<<14))==0)
|
|
wall[wall[i].nextwall].cstat |= (1<<14);
|
|
|
|
j = 0;
|
|
for (i=0; i<numwalls; i++)
|
|
j += !!(wall[i].cstat&(1<<14));
|
|
if (max(numwalls,onewnumwalls)+j > MAXWALLS)
|
|
{
|
|
return 0; // no points inserted, would exceed limits
|
|
}
|
|
|
|
// the actual insertion!
|
|
m = 0;
|
|
for (i=0; i<numwalls /* rises with ins. */; i++)
|
|
{
|
|
if (wall[i].cstat&(1<<14))
|
|
if (wall[i].nextwall<0 || i<wall[i].nextwall) // || !(NEXTWALL(i).cstat&(1<<14)) ??
|
|
{
|
|
m += insertpoint(i, dax,day, mapwallnum);
|
|
}
|
|
}
|
|
|
|
for (i=0; i<numwalls; i++)
|
|
{
|
|
if (wall[i].cstat&(1<<14))
|
|
{
|
|
wall[i].cstat &= ~(1<<14);
|
|
k = yax_getnextwall(i+1, YAX_CEILING);
|
|
if (k >= 0)
|
|
yax_setnextwall(i+1, YAX_CEILING, k+1);
|
|
k = yax_getnextwall(i+1, YAX_FLOOR);
|
|
if (k >= 0)
|
|
yax_setnextwall(i+1, YAX_FLOOR, k+1);
|
|
}
|
|
}
|
|
|
|
if (m==j)
|
|
return m;
|
|
else
|
|
return m|(j<<16);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
insertpoint(thewall, dax,day, mapwallnum);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
|
|
// based on lineintersect in engine.c, but lines are considered as infinitely
|
|
// extending
|
|
void inflineintersect(int32_t x1, int32_t y1, int32_t x2, int32_t y2,
|
|
int32_t x3, int32_t y3, int32_t x4, int32_t y4,
|
|
int32_t *intx, int32_t *inty, int32_t *sign12, int32_t *sign34)
|
|
{
|
|
//p1 to p2 is a line segment
|
|
int64_t x21, y21, x34, y34, x31, y31, bot, topt, topu, t;
|
|
|
|
x21 = x2-x1; x34 = x3-x4;
|
|
y21 = y2-y1; y34 = y3-y4;
|
|
bot = x21*y34 - y21*x34;
|
|
if (bot >= 0)
|
|
{
|
|
if (bot == 0) { *sign12 = *sign34 = 0; return; };
|
|
x31 = x3-x1; y31 = y3-y1;
|
|
topt = x31*y34 - y31*x34;
|
|
topu = x21*y31 - y21*x31;
|
|
}
|
|
else
|
|
{
|
|
x31 = x3-x1; y31 = y3-y1;
|
|
topt = x31*y34 - y31*x34;
|
|
topu = x21*y31 - y21*x31;
|
|
}
|
|
|
|
t = (topt*(1<<24))/bot;
|
|
*intx = x1 + ((x21*t)>>24);
|
|
*inty = y1 + ((y21*t)>>24);
|
|
|
|
*sign12 = topt < 0 ? -1 : 1;
|
|
*sign34 = topu < 0 ? -1 : 1;
|
|
|
|
return;
|
|
}
|
|
|
|
static int32_t lineintersect2v(const vec2_t *p1, const vec2_t *p2, // line segment 1
|
|
const vec2_t *q1, const vec2_t *q2, // line segment 2
|
|
vec2_t *pint)
|
|
{
|
|
int32_t intz;
|
|
return lintersect(p1->x, p1->y, 0, p2->x, p2->y, 0,
|
|
q1->x, q1->y, q2->x, q2->y,
|
|
&pint->x, &pint->y, &intz);
|
|
}
|
|
|
|
static int32_t vec2eq(const vec2_t *v1, const vec2_t *v2)
|
|
{
|
|
return (v1->x==v2->x && v1->y==v2->y);
|
|
}
|
|
|
|
#ifdef YAX_ENABLE
|
|
// After auto-creating inner sector <ns> in existing sector <os>, we need to
|
|
// see if some sprites contained in <os> need to change their sector.
|
|
static void CorrectSpriteSectnums(int32_t os, int32_t ns)
|
|
{
|
|
int32_t i, ni;
|
|
|
|
for (SPRITES_OF_SECT_SAFE(os, i, ni))
|
|
{
|
|
if (inside(sprite[i].x, sprite[i].y, ns)==1)
|
|
changespritesect(i, ns);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// precondition: [numwalls, newnumwalls-1] form a new loop (may be of wrong orientation)
|
|
// ret_ofirstwallofs: if != NULL, *ret_ofirstwallofs will contain the offset of the old
|
|
// first wall from the new first wall of the sector k, and the automatic
|
|
// restoring of the old first wll will not be carried out
|
|
// returns:
|
|
// -1, -2: errors
|
|
// 0, 1: OK, 1 means it was an extended sector and an inner loop has been added automatically
|
|
static int32_t AddLoopToSector(int32_t k, int32_t *ret_ofirstwallofs)
|
|
{
|
|
int32_t extendedSector=0, firstwall, i, j;
|
|
#ifdef YAX_ENABLE
|
|
int16_t cbunch, fbunch;
|
|
int32_t newnumwalls2;
|
|
|
|
yax_getbunches(k, &cbunch, &fbunch);
|
|
extendedSector = (cbunch>=0 || fbunch>=0);
|
|
#endif
|
|
j = newnumwalls-numwalls;
|
|
#ifdef YAX_ENABLE
|
|
newnumwalls2 = newnumwalls + j;
|
|
|
|
if (extendedSector)
|
|
{
|
|
if ((cbunch>=0 && (sector[k].ceilingstat&2))
|
|
|| (fbunch>=0 && (sector[k].floorstat&2)))
|
|
{
|
|
printmessage16("Sloped extended sectors cannot be subdivided.");
|
|
newnumwalls--;
|
|
return -1;
|
|
}
|
|
|
|
if (newnumwalls + j > MAXWALLS || numsectors+1 > MAXSECTORS)
|
|
{
|
|
message("Automatically adding inner sector to new extended sector would exceed limits!");
|
|
newnumwalls--;
|
|
return -2;
|
|
}
|
|
}
|
|
#endif
|
|
if (clockdir(numwalls) == CLOCKDIR_CW)
|
|
flipwalls(numwalls,newnumwalls);
|
|
|
|
sector[k].wallnum += j;
|
|
for (i=k+1; i<numsectors; i++)
|
|
sector[i].wallptr += j;
|
|
firstwall = sector[k].wallptr;
|
|
|
|
for (i=0; i<numwalls; i++)
|
|
{
|
|
if (wall[i].nextwall >= firstwall)
|
|
wall[i].nextwall += j;
|
|
if (wall[i].point2 >= firstwall)
|
|
wall[i].point2 += j;
|
|
}
|
|
#ifdef YAX_ENABLE
|
|
yax_tweakwalls(firstwall, j);
|
|
#endif
|
|
|
|
Bmemmove(&wall[firstwall+j], &wall[firstwall], (newnumwalls-firstwall)*sizeof(walltype));
|
|
// add new loop to beginning of sector
|
|
Bmemmove(&wall[firstwall], &wall[newnumwalls], j*sizeof(walltype));
|
|
|
|
for (i=firstwall; i<firstwall+j; i++)
|
|
{
|
|
wall[i].point2 += (firstwall-numwalls);
|
|
|
|
copy_some_wall_members(i, firstwall+j, 1);
|
|
wall[i].cstat &= ~(1+16+32+64);
|
|
}
|
|
|
|
numwalls = newnumwalls;
|
|
newnumwalls = -1;
|
|
#ifdef YAX_ENABLE
|
|
if (extendedSector)
|
|
{
|
|
newnumwalls = whitelinescan(k, firstwall);
|
|
if (newnumwalls != newnumwalls2)
|
|
message("AddLoopToSector: newnumwalls != newnumwalls2!!! WTF?");
|
|
for (i=numwalls; i<newnumwalls; i++)
|
|
{
|
|
NEXTWALL(i).nextwall = i;
|
|
NEXTWALL(i).nextsector = numsectors;
|
|
}
|
|
|
|
yax_setbunches(numsectors, cbunch, fbunch);
|
|
|
|
numwalls = newnumwalls;
|
|
newnumwalls = -1;
|
|
numsectors++;
|
|
|
|
CorrectSpriteSectnums(k, numsectors-1);
|
|
}
|
|
#endif
|
|
if (ret_ofirstwallofs)
|
|
*ret_ofirstwallofs = j;
|
|
else
|
|
setfirstwall(k, firstwall+j); // restore old first wall
|
|
|
|
return extendedSector;
|
|
}
|
|
|
|
int32_t select_sprite_tag(int32_t spritenum)
|
|
{
|
|
int32_t lt = taglab_linktags(1, spritenum);
|
|
spritetype *spr = &sprite[spritenum];
|
|
|
|
if (lt==0)
|
|
return INT32_MIN;
|
|
|
|
if (lt&1)
|
|
return spr->lotag;
|
|
if (lt&2)
|
|
return spr->hitag;
|
|
if (lt&4)
|
|
return spr->extra;
|
|
if (lt&8)
|
|
return spr->xvel;
|
|
if (lt&16)
|
|
return spr->yvel;
|
|
if (lt&32)
|
|
return spr->zvel;
|
|
if (lt&64)
|
|
return spr->extra;
|
|
|
|
return INT32_MIN;
|
|
}
|
|
|
|
static void drawlinebetween(const vec3_t *v1, const vec3_t *v2, int32_t col, uint32_t pat)
|
|
{
|
|
// based on m32exec.c/drawline*
|
|
const int32_t xofs=halfxdim16, yofs=midydim16;
|
|
const uint32_t opat=drawlinepat;
|
|
|
|
int32_t x1, x2, y1, y2;
|
|
|
|
screencoords(&x1,&y1, v1->x-pos.x,v1->y-pos.y, zoom);
|
|
screencoords(&x2,&y2, v2->x-pos.x,v2->y-pos.y, zoom);
|
|
|
|
if (m32_sideview)
|
|
{
|
|
y1 += getscreenvdisp(v1->z-pos.z,zoom);
|
|
y2 += getscreenvdisp(v2->z-pos.z,zoom);
|
|
}
|
|
|
|
drawlinepat = pat;
|
|
drawline16(xofs+x1,yofs+y1, xofs+x2,yofs+y2, col);
|
|
drawlinepat = opat;
|
|
}
|
|
|
|
// world -> screen coords for overhead mode
|
|
void ovhscrcoords(int32_t x, int32_t y, int32_t *scrx, int32_t *scry)
|
|
{
|
|
*scrx = halfxdim16 + mulscale14(x-pos.x, zoom);
|
|
*scry = midydim16 + mulscale14(y-pos.y, zoom);
|
|
}
|
|
|
|
static void draw_cross(int32_t centerx, int32_t centery, int32_t radius, int32_t col)
|
|
{
|
|
int32_t dax, day;
|
|
ovhscrcoords(centerx, centery, &dax, &day);
|
|
drawline16base(dax, day, -radius,-radius, +radius,+radius, col);
|
|
drawline16base(dax, day, -radius,+radius, +radius,-radius, col);
|
|
}
|
|
|
|
static void draw_square(int32_t dax, int32_t day, int32_t ps, int32_t col)
|
|
{
|
|
ovhscrcoords(dax, day, &dax, &day);
|
|
drawline16base(dax, day, -ps,-ps, +ps,-ps, col);
|
|
drawline16base(dax, day, +ps,-ps, +ps,+ps, col);
|
|
drawline16base(dax, day, +ps,+ps, -ps,+ps, col);
|
|
drawline16base(dax, day, -ps,+ps, -ps,-ps, col);
|
|
}
|
|
|
|
//// Interactive Scaling
|
|
static struct {
|
|
int8_t active, rotatep;
|
|
int32_t pivx, pivy; // pivot point
|
|
int32_t dragx, dragy; // dragged point
|
|
int32_t xsc, ysc, ang;
|
|
} isc;
|
|
|
|
static void isc_transform(int32_t *x, int32_t *y)
|
|
{
|
|
if (!isc.rotatep)
|
|
{
|
|
*x = isc.pivx + mulscale16(*x-isc.pivx, isc.xsc);
|
|
*y = isc.pivy + mulscale16(*y-isc.pivy, isc.ysc);
|
|
}
|
|
else
|
|
{
|
|
rotatepoint(isc.pivx, isc.pivy, *x, *y, isc.ang, x, y);
|
|
}
|
|
}
|
|
|
|
#define EDITING_MAP_P() (newnumwalls>=0 || joinsector[0]>=0 || circlewall>=0 || (bstatus&1) || isc.active)
|
|
|
|
#define HLMEMBERX(Hl, Member) (*(((Hl)&16384) ? &sprite[(Hl)&16383].Member : &wall[Hl].Member))
|
|
#define HLMEMBER(Hlidx, Member) HLMEMBERX(highlight[Hlidx], Member)
|
|
|
|
void overheadeditor(void)
|
|
{
|
|
char buffer[80];
|
|
const char *dabuffer;
|
|
int32_t i, j, k, m=0, mousxplc, mousyplc, firstx=0, firsty=0, oposz, col;
|
|
int32_t numwalls_bak;
|
|
int32_t startwall=0, endwall, dax, day, x1, y1, x2, y2, x3, y3; //, x4, y4;
|
|
int32_t highlightx1, highlighty1, highlightx2, highlighty2;
|
|
int16_t bad, joinsector[2];
|
|
int32_t bstatus, mousewaitmask=0;
|
|
int16_t circlepoints;
|
|
int32_t sectorhighlightx=0, sectorhighlighty=0;
|
|
int16_t cursectorhighlight, sectorhighlightstat;
|
|
int32_t prefixarg = 0, tsign;
|
|
int32_t resetsynctics = 0, lasttick=getticks(), waitdelay=totalclock, lastdraw=getticks();
|
|
int32_t olen[2]={0,0}, dragwall[2] = {-1, -1};
|
|
int16_t linehighlight2;
|
|
|
|
ovh.suckwall = -1;
|
|
ovh.split = 0;
|
|
ovh.splitsect = -1;
|
|
ovh.splitstartwall = -1;
|
|
|
|
qsetmodeany(xdim2d,ydim2d);
|
|
xdim2d = xdim;
|
|
ydim2d = ydim;
|
|
|
|
osearchx = searchx;
|
|
osearchy = searchy;
|
|
|
|
searchx = clamp(scale(searchx,xdim2d,xdimgame), 8, xdim2d-8-1);
|
|
searchy = clamp(scale(searchy,ydim2d-STATUS2DSIZ2,ydimgame), 8, ydim2d-STATUS2DSIZ-8-1);
|
|
oposz = pos.z;
|
|
|
|
yax_updategrays(pos.z);
|
|
|
|
begindrawing(); //{{{
|
|
CLEARLINES2D(0, ydim, 0);
|
|
enddrawing(); //}}}
|
|
|
|
ydim16 = ydim-STATUS2DSIZ2;
|
|
|
|
cursectorhighlight = -1;
|
|
lastpm16time = -1;
|
|
|
|
update_highlightsector();
|
|
ovh_whiteoutgrab(0);
|
|
|
|
highlightcnt = -1;
|
|
Bmemset(show2dwall, 0, sizeof(show2dwall)); //Clear all highlights
|
|
Bmemset(show2dsprite, 0, sizeof(show2dsprite));
|
|
|
|
RESET_EDITOR_VARS();
|
|
bstatus = 0;
|
|
|
|
while ((keystatus[buildkeys[BK_MODE2D_3D]]>>1) == 0)
|
|
{
|
|
int32_t mousx, mousy;
|
|
|
|
if (!((vel|angvel|svel) //DOWN_BK(MOVEFORWARD) || DOWN_BK(MOVEBACKWARD) || DOWN_BK(TURNLEFT) || DOWN_BK(TURNRIGHT)
|
|
|| DOWN_BK(MOVEUP) || DOWN_BK(MOVEDOWN) || keystatus[0x10] || keystatus[0x11]
|
|
|| keystatus[0x48] || keystatus[0x4b] || keystatus[0x4d] || keystatus[0x50] // keypad keys
|
|
|| bstatus || OSD_IsMoving()))
|
|
{
|
|
if (totalclock > waitdelay)
|
|
{
|
|
uint32_t ms = (highlightsectorcnt>0) ? 75 : 200;
|
|
// wait for event, timeout after 200 ms - (last loop time)
|
|
idle_waitevent_timeout(ms - min(getticks()-lasttick, ms));
|
|
// have synctics reset to 0 after we've slept to avoid zooming out to the max instantly
|
|
resetsynctics = 1;
|
|
}
|
|
}
|
|
else waitdelay = totalclock + 30; // should be 250 ms
|
|
|
|
lasttick = getticks();
|
|
|
|
if (handleevents())
|
|
{
|
|
if (quitevent)
|
|
{
|
|
keystatus[1] = 1;
|
|
quitevent = 0;
|
|
}
|
|
}
|
|
|
|
if (resetsynctics)
|
|
{
|
|
resetsynctics = 0;
|
|
lockclock = totalclock;
|
|
synctics = 0;
|
|
}
|
|
|
|
OSD_DispatchQueued();
|
|
|
|
if (totalclock < 120*3)
|
|
printmessage16("Uses BUILD technology by Ken Silverman.");
|
|
else if (totalclock < 120*6)
|
|
{
|
|
printmessage16("Press F1 for help. This is a test release; always keep backups of your maps.");
|
|
// printext16(8L,ydim-STATUS2DSIZ+32L,editorcolors[9],-1,kensig,0);
|
|
}
|
|
|
|
oldmousebstatus = bstatus;
|
|
getmousevalues(&mousx,&mousy,&bstatus);
|
|
|
|
{
|
|
int32_t bs = bstatus;
|
|
bstatus &= ~mousewaitmask;
|
|
mousewaitmask &= bs;
|
|
}
|
|
|
|
mousx = (mousx<<16)+mousexsurp;
|
|
mousy = (mousy<<16)+mouseysurp;
|
|
{
|
|
ldiv_t ld;
|
|
ld = ldiv(mousx, 1<<16); mousx = ld.quot; mousexsurp = ld.rem;
|
|
ld = ldiv(mousy, 1<<16); mousy = ld.quot; mouseysurp = ld.rem;
|
|
}
|
|
searchx += mousx;
|
|
searchy += mousy;
|
|
|
|
inpclamp(&searchx, 8, xdim-8-1);
|
|
inpclamp(&searchy, 8, ydim-8-1);
|
|
|
|
mainloop_move();
|
|
|
|
getpoint(searchx,searchy,&mousxplc,&mousyplc);
|
|
linehighlight = getlinehighlight(mousxplc, mousyplc, linehighlight, 0);
|
|
linehighlight2 = getlinehighlight(mousxplc, mousyplc, linehighlight, 1);
|
|
|
|
if (newnumwalls >= numwalls)
|
|
{
|
|
// if we're in the process of drawing a wall, set the end point's coordinates
|
|
dax = mousxplc;
|
|
day = mousyplc;
|
|
adjustmark(&dax,&day,numwalls+!ovh.split);
|
|
wall[newnumwalls].x = dax;
|
|
wall[newnumwalls].y = day;
|
|
}
|
|
|
|
ydim16 = ydim;// - STATUS2DSIZ2;
|
|
midydim16 = ydim>>1;
|
|
|
|
numwalls_bak = numwalls;
|
|
numwalls = newnumwalls;
|
|
if (numwalls < 0)
|
|
numwalls = numwalls_bak;
|
|
|
|
if ((getticks() - lastdraw) >= 5 || (vel|angvel|svel) || DOWN_BK(MOVEUP) || DOWN_BK(MOVEDOWN)
|
|
|| mousx || mousy || bstatus || keystatus[0x10] || keystatus[0x11]
|
|
|| newnumwalls>=0 || OSD_IsMoving())
|
|
{
|
|
lastdraw = getticks();
|
|
|
|
clear2dscreen();
|
|
|
|
setup_sideview_sincos();
|
|
|
|
if (graphicsmode && !m32_sideview)
|
|
{
|
|
Bmemset(show2dsector, 0, sizeof(show2dsector));
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
show2dsector[i>>3] |= (1<<(i&7));
|
|
}
|
|
|
|
setview(0, 0, xdim-1, ydim16-1);
|
|
|
|
if (graphicsmode == 2)
|
|
totalclocklock = totalclock;
|
|
|
|
drawmapview(pos.x, pos.y, zoom, 1536);
|
|
}
|
|
|
|
draw2dgrid(pos.x,pos.y,pos.z,cursectnum,ang,zoom,grid);
|
|
|
|
CallExtPreCheckKeys();
|
|
|
|
{
|
|
int32_t cx, cy;
|
|
|
|
// Draw brown arrow (start)
|
|
screencoords(&x2, &y2, startpos.x-pos.x,startpos.y-pos.y, zoom);
|
|
if (m32_sideview)
|
|
y2 += getscreenvdisp(startpos.z-pos.z, zoom);
|
|
|
|
cx = halfxdim16+x2;
|
|
cy = midydim16+y2;
|
|
if ((cx >= 2 && cx <= xdim-3) && (cy >= 2 && cy <= ydim16-3))
|
|
{
|
|
int16_t angofs = m32_sideview ? m32_sideang : 0;
|
|
x1 = mulscale11(sintable[(startang+angofs+2560)&2047],zoom) / 768;
|
|
y1 = mulscale11(sintable[(startang+angofs+2048)&2047],zoom) / 768;
|
|
i = scalescreeny(x1);
|
|
j = scalescreeny(y1);
|
|
begindrawing(); //{{{
|
|
drawline16base(cx,cy, x1,j, -x1,-j, editorcolors[2]);
|
|
drawline16base(cx,cy, x1,j, +y1,-i, editorcolors[2]);
|
|
drawline16base(cx,cy, x1,j, -y1,+i, editorcolors[2]);
|
|
enddrawing(); //}}}
|
|
}
|
|
}
|
|
|
|
draw2dscreen(&pos,cursectnum,ang,zoom,grid);
|
|
|
|
begindrawing(); //{{{
|
|
|
|
if (keystatus[0x2a] && (pointhighlight&16384) && highlightcnt<=0) // LShift
|
|
{
|
|
// draw lines to linking sprites
|
|
const int32_t refspritenum = pointhighlight&16383;
|
|
const int32_t reftag = select_sprite_tag(refspritenum);
|
|
|
|
if (reftag != INT32_MIN)
|
|
{
|
|
for (i=0; i<numsectors; i++)
|
|
for (SPRITES_OF_SECT(i, j))
|
|
if (reftag==select_sprite_tag(j))
|
|
drawlinebetween((vec3_t *)&sprite[refspritenum], (vec3_t *)&sprite[j],
|
|
editorcolors[12], 0x33333333);
|
|
}
|
|
}
|
|
|
|
if (showtags)
|
|
{
|
|
if (zoom >= 768)
|
|
{
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
int16_t secshort = i;
|
|
|
|
YAX_SKIPSECTOR(i);
|
|
|
|
dabuffer = CallExtGetSectorCaption(i);
|
|
if (dabuffer[0] == 0)
|
|
continue;
|
|
|
|
get_sectors_center(&secshort, 1, &dax, &day);
|
|
|
|
drawsmallabel(dabuffer, editorcolors[0], editorcolors[7],
|
|
dax, day, getflorzofslope(i,dax,day));
|
|
}
|
|
}
|
|
|
|
x3 = pos.x + divscale14(-halfxdim16,zoom);
|
|
y3 = pos.y + divscale14(-(midydim16-4),zoom);
|
|
// x4 = pos.x + divscale14(halfxdim16,zoom);
|
|
// y4 = pos.y + divscale14(ydim16-(midydim16-4),zoom);
|
|
|
|
if (newnumwalls >= 0)
|
|
{
|
|
for (i=newnumwalls; i>=numwalls_bak; i--)
|
|
wall[i].cstat |= (1<<14);
|
|
}
|
|
|
|
i = numwalls-1;
|
|
j = numsectors-1; // might be -1 if empty map!
|
|
if (newnumwalls >= 0)
|
|
i = newnumwalls-1;
|
|
for (; i>=0; i--)
|
|
{
|
|
walltype const * const wal = &wall[i];
|
|
|
|
if (j>=0 && sector[j].wallptr > i)
|
|
j--;
|
|
|
|
if (zoom < 768 && !(wal->cstat & (1<<14)))
|
|
continue;
|
|
|
|
YAX_SKIPWALL(i);
|
|
|
|
//Get average point of wall
|
|
// if ((dax > x3) && (dax < x4) && (day > y3) && (day < y4))
|
|
{
|
|
dabuffer = CallExtGetWallCaption(i);
|
|
if (dabuffer[0] == 0)
|
|
continue;
|
|
|
|
dax = (wal->x+wall[wal->point2].x)>>1;
|
|
day = (wal->y+wall[wal->point2].y)>>1;
|
|
drawsmallabel(dabuffer, editorcolors[0], editorcolors[31],
|
|
dax, day, (i >= numwalls || j<0) ? 0 : getflorzofslope(j, dax,day));
|
|
}
|
|
}
|
|
|
|
if (zoom >= 768)
|
|
{
|
|
int32_t alwaysshowgray = get_alwaysshowgray();
|
|
|
|
for (i=0, k=0; (m32_sideview && k<m32_swcnt) || (!m32_sideview && i<MAXSPRITES); i++, k++)
|
|
{
|
|
if (m32_sideview)
|
|
{
|
|
i = m32_wallsprite[k];
|
|
if (i<MAXWALLS)
|
|
continue;
|
|
i = i-MAXWALLS;
|
|
}
|
|
else
|
|
if (sprite[i].statnum == MAXSTATUS)
|
|
continue;
|
|
|
|
if ((!m32_sideview || !alwaysshowgray) && sprite[i].sectnum >= 0)
|
|
YAX_SKIPSECTOR(sprite[i].sectnum);
|
|
|
|
dabuffer = CallExtGetSpriteCaption(i);
|
|
if (dabuffer[0] != 0)
|
|
{
|
|
int32_t blocking = (sprite[i].cstat&1);
|
|
|
|
col = 3 + 2*blocking;
|
|
if (spritecol2d[sprite[i].picnum][blocking])
|
|
col = spritecol2d[sprite[i].picnum][blocking];
|
|
|
|
if ((i == pointhighlight-16384) && (totalclock & 32))
|
|
col += (2<<2);
|
|
|
|
drawsmallabel(dabuffer, editorcolors[0], editorcolors[col],
|
|
sprite[i].x, sprite[i].y, sprite[i].z);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// stick this event right between begin- end enddrawing()...
|
|
// also after the above label stuff so users can redefine them
|
|
VM_OnEvent(EVENT_DRAW2DSCREEN, -1);
|
|
|
|
printcoords16(pos.x,pos.y,ang);
|
|
|
|
numwalls = numwalls_bak;
|
|
|
|
if (highlightsectorcnt >= 0)
|
|
{
|
|
for (i=0; i<numsectors; i++)
|
|
if (hlsectorbitmap[i>>3]&(1<<(i&7)))
|
|
fillsector(i, -1);
|
|
}
|
|
|
|
if (keystatus[0x2a]) // LShift
|
|
{
|
|
if (m32_sideview || highlightcnt <= 0)
|
|
{
|
|
drawlinepat = 0x00ff00ff;
|
|
drawline16(searchx,0, searchx,ydim2d-1, editorcolors[15]);
|
|
drawline16(0,searchy, xdim2d-1,searchy, editorcolors[15]);
|
|
drawlinepat = 0xffffffff;
|
|
|
|
_printmessage16("(%d,%d)",mousxplc,mousyplc);
|
|
}
|
|
else
|
|
{
|
|
// do interactive scaling
|
|
if (!isc.active)
|
|
{
|
|
if (pointhighlight >= 0 && (bstatus&3))
|
|
{
|
|
// initialize by finding pivot point
|
|
int32_t minx=INT32_MAX, miny=INT32_MAX;
|
|
int32_t maxx=INT32_MIN, maxy=INT32_MIN;
|
|
|
|
isc.rotatep = ((bstatus&3)==2);
|
|
bstatus &= ~3;
|
|
|
|
for (i=0; i<highlightcnt; i++)
|
|
{
|
|
minx = min(minx, HLMEMBER(i, x));
|
|
miny = min(miny, HLMEMBER(i, y));
|
|
maxx = max(maxx, HLMEMBER(i, x));
|
|
maxy = max(maxy, HLMEMBER(i, y));
|
|
}
|
|
|
|
isc.pivx = (minx+maxx)/2;
|
|
isc.pivy = (miny+maxy)/2;
|
|
|
|
isc.dragx = HLMEMBERX(pointhighlight, x);
|
|
isc.dragy = HLMEMBERX(pointhighlight, y);
|
|
|
|
isc.xsc = isc.ysc = 1<<16;
|
|
isc.ang = 0;
|
|
|
|
isc.active = 1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (bstatus&3)
|
|
{
|
|
// drag/rotate the reference point
|
|
const int32_t pivx=isc.pivx, pivy=isc.pivy;
|
|
const int32_t dragx=isc.dragx, dragy=isc.dragy;
|
|
int32_t mxplc=mousxplc, myplc=mousyplc, xsc=1<<16, ysc=1<<16;
|
|
|
|
const int32_t dx=dragx-pivx, dy=dragy-pivy;
|
|
int32_t mdx, mdy;
|
|
|
|
bstatus &= ~3;
|
|
|
|
draw_cross(pivx, pivy, 3, editorcolors[14]);
|
|
|
|
adjustmark(&mxplc, &myplc, numwalls);
|
|
mdx = mxplc-pivx;
|
|
mdy = myplc-pivy;
|
|
|
|
if (!isc.rotatep)
|
|
{
|
|
if (mdx != 0 && dx != 0 && klabs(dx) >= 8)
|
|
xsc = min(klabs(divscale16(mdx, dx)), 1<<18);
|
|
if (mdy != 0 && dy != 0 && klabs(dy) >= 8)
|
|
ysc = min(klabs(divscale16(mdy, dy)), 1<<18);
|
|
|
|
if (eitherCTRL)
|
|
xsc = ysc = max(xsc, ysc);
|
|
|
|
isc.xsc = xsc;
|
|
isc.ysc = ysc;
|
|
|
|
printmessage16("scale x=%.3f y=%.3f", (double)xsc/65536, (double)ysc/65536);
|
|
}
|
|
else
|
|
{
|
|
isc.ang = getangle(mdx, mdy) - getangle(dx, dy);
|
|
printmessage16("rotate ang %d", isc.ang);
|
|
}
|
|
|
|
for (i=0; i<highlightcnt; i++)
|
|
{
|
|
int32_t x=HLMEMBER(i, x), y=HLMEMBER(i, y);
|
|
|
|
isc_transform(&x, &y);
|
|
|
|
draw_square(x, y, 2, editorcolors[15]);
|
|
|
|
if ((highlight[i]&16384)==0)
|
|
{
|
|
walltype const * const wal = &wall[highlight[i]];
|
|
const int32_t p2=wal->point2, hlp=(show2dwall[p2>>3]&(1<<(p2&7)));
|
|
vec3_t v1 = { x, y, 0 }, v2 = { wall[p2].x, wall[p2].y, 0 };
|
|
|
|
isc_transform(&v2.x, &v2.y);
|
|
if (!hlp)
|
|
{
|
|
v2.x = wall[p2].x;
|
|
v2.y = wall[p2].y;
|
|
}
|
|
|
|
drawlinebetween(&v1, &v2, !hlp ? 8 :
|
|
editorcolors[wal->nextwall >= 0 ? 33 : 7],
|
|
0x11111111);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// finish interactive scaling
|
|
isc.active = 0;
|
|
|
|
if ((!isc.rotatep && (isc.xsc!=1<<16 || isc.ysc!=1<<16)) ||
|
|
(isc.rotatep && (isc.ang!=0)))
|
|
{
|
|
for (i=0; i<highlightcnt; i++)
|
|
{
|
|
int32_t *x=&HLMEMBER(i, x), *y=&HLMEMBER(i, y);
|
|
|
|
isc_transform(x, y);
|
|
|
|
if (isc.rotatep && (highlight[i]&16384))
|
|
{
|
|
spritetype *spr = &sprite[highlight[i]&16383];
|
|
spr->ang = (spr->ang + isc.ang)&2047;
|
|
}
|
|
}
|
|
|
|
if (!isc.rotatep)
|
|
message("Highlights scaled by x=%.3f y=%.3f",
|
|
(double)isc.xsc/65536, (double)isc.ysc/65536);
|
|
else
|
|
message("Highlights rotated by %d BUILD degrees", isc.ang);
|
|
|
|
asksave = 1;
|
|
}
|
|
else
|
|
printmessage16(" ");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (isc.active)
|
|
{
|
|
isc.active = 0;
|
|
|
|
printmessage16("Aborted interactive %s.", isc.rotatep ? "rotation" : "scaling");
|
|
bstatus &= ~3;
|
|
mousewaitmask = 3;
|
|
pointhighlight = -1;
|
|
}
|
|
}
|
|
|
|
drawline16(searchx,0, searchx,8, editorcolors[15]);
|
|
drawline16(0,searchy, 8,searchy, editorcolors[15]);
|
|
|
|
////// draw mouse pointer
|
|
col = editorcolors[15 - 3*gridlock];
|
|
if (joinsector[0] >= 0)
|
|
col = editorcolors[11];
|
|
|
|
if (numcorruptthings>0)
|
|
{
|
|
static char cbuf[64];
|
|
|
|
if ((pointhighlight&16384)==0)
|
|
{
|
|
// If aiming at wall, check whether it is corrupt, and print a
|
|
// warning message near the mouse pointer if that is the case.
|
|
for (i=0; i<numcorruptthings; i++)
|
|
if ((corruptthings[i]&CORRUPT_MASK)==CORRUPT_WALL &&
|
|
(corruptthings[i]&(MAXWALLS-1))==pointhighlight)
|
|
{
|
|
col = editorcolors[13];
|
|
printext16(searchx+6,searchy-6-8,editorcolors[13],editorcolors[0],"corrupt wall",0);
|
|
break;
|
|
}
|
|
}
|
|
|
|
Bsprintf(cbuf, "Map corrupt (level %d): %s%d errors", corruptlevel,
|
|
numcorruptthings>=MAXCORRUPTTHINGS ? ">=":"", numcorruptthings);
|
|
printext16(8,8, editorcolors[13],editorcolors[0],cbuf,0);
|
|
}
|
|
|
|
if (highlightsectorcnt==0 || highlightcnt==0)
|
|
{
|
|
if (keystatus[0x27] || keystatus[0x28]) // ' and ;
|
|
{
|
|
col = editorcolors[14];
|
|
|
|
drawline16base(searchx+16, searchy-16, -4,0, +4,0, col);
|
|
if (keystatus[0x28])
|
|
drawline16base(searchx+16, searchy-16, 0,-4, 0,+4, col);
|
|
}
|
|
|
|
if (highlightsectorcnt == 0)
|
|
if (keystatus[0x36])
|
|
printext16(searchx+6, searchy-2+8,editorcolors[12],-1,"ALL",0);
|
|
|
|
if (highlightcnt == 0)
|
|
{
|
|
if (eitherCTRL && (highlightx1!=highlightx2 || highlighty1!=highlighty2))
|
|
printext16(searchx+6,searchy-6-8,editorcolors[12],-1,"SPR ONLY",0);
|
|
#ifdef YAX_ENABLE
|
|
if (keystatus[0xcf]) // End
|
|
printext16(searchx+6,searchy-2+8,editorcolors[12],-1,"ALL",0);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
drawline16base(searchx,searchy, +0,-8, +0,-1, col);
|
|
drawline16base(searchx,searchy, +1,-8, +1,-1, col);
|
|
drawline16base(searchx,searchy, +0,+2, +0,+9, col);
|
|
drawline16base(searchx,searchy, +1,+2, +1,+9, col);
|
|
drawline16base(searchx,searchy, -8,+0, -1,+0, col);
|
|
drawline16base(searchx,searchy, -8,+1, -1,+1, col);
|
|
drawline16base(searchx,searchy, +2,+0, +9,+0, col);
|
|
drawline16base(searchx,searchy, +2,+1, +9,+1, col);
|
|
|
|
////// Draw the white pixel closest to mouse cursor on linehighlight
|
|
if (linehighlight>=0)
|
|
{
|
|
char col = wall[linehighlight].nextsector >= 0 ? editorcolors[15] : editorcolors[5];
|
|
|
|
if (m32_sideview)
|
|
{
|
|
getclosestpointonwall(searchx,searchy, linehighlight, &dax,&day, 1);
|
|
drawline16base(dax,day, 0,0, 0,0, col);
|
|
}
|
|
else
|
|
{
|
|
getclosestpointonwall(mousxplc,mousyplc, linehighlight, &dax,&day, 0);
|
|
ovhscrcoords(dax, day, &x2, &y2);
|
|
drawline16base(x2, y2, 0,0, 0,0, col);
|
|
}
|
|
}
|
|
|
|
enddrawing(); //}}}
|
|
|
|
OSD_Draw();
|
|
}
|
|
|
|
inputchecked = 1;
|
|
|
|
VM_OnEvent(EVENT_PREKEYS2D, -1);
|
|
CallExtCheckKeys(); // TX 20050101, it makes more sense to have this here so keys can be overwritten with new functions in bstub.c
|
|
|
|
// Flip/mirror sector Ed Coolidge
|
|
if (keystatus[0x2d] || keystatus[0x15]) // X or Y (2D)
|
|
{
|
|
int32_t about_x=keystatus[0x2d];
|
|
int32_t doMirror = eitherALT; // mirror walls and wall/floor sprites
|
|
|
|
#ifdef YAX_ENABLE
|
|
if (highlightsectorcnt > 0 && !hl_all_bunch_sectors_p())
|
|
{
|
|
printmessage16("To flip extended sectors, all sectors of a bunch must be selected");
|
|
keystatus[0x2d] = keystatus[0x15] = 0;
|
|
}
|
|
else
|
|
#endif
|
|
if (highlightsectorcnt > 0)
|
|
{
|
|
int16_t *const otonwall = onextwall; // OK, since we make old-nextwalls invalid
|
|
|
|
mkonwinvalid();
|
|
|
|
keystatus[0x2d] = keystatus[0x15] = 0;
|
|
|
|
for (j=0; j<numwalls; j++)
|
|
otonwall[j] = j;
|
|
|
|
get_sectors_center(highlightsector, highlightsectorcnt, &dax, &day);
|
|
|
|
if (gridlock && grid > 0)
|
|
locktogrid(&dax, &day);
|
|
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
int32_t startofloop, endofloop;
|
|
int32_t numtoswap = -1;
|
|
int32_t w=0;
|
|
twalltype tempwall;
|
|
|
|
startofloop = startwall = sector[highlightsector[i]].wallptr;
|
|
endofloop = endwall = startwall+sector[highlightsector[i]].wallnum-1;
|
|
#if 0
|
|
if (doMirror)
|
|
{
|
|
//mirror sector textures
|
|
sector[highlightsector[i]].ceilingstat ^= 0x10;
|
|
sector[highlightsector[i]].floorstat ^= 0x10;
|
|
}
|
|
#endif
|
|
//save position of wall at start of loop
|
|
x3 = wall[startofloop].x;
|
|
y3 = wall[startofloop].y;
|
|
|
|
for (j=startwall; j<=endwall; j++)
|
|
{
|
|
//fix position of walls
|
|
if (about_x)
|
|
{
|
|
wall[j].x = dax-POINT2(j).x+dax; //flip wall.x about dax
|
|
wall[j].y = POINT2(j).y;
|
|
}
|
|
else
|
|
{
|
|
wall[j].x = POINT2(j).x;
|
|
wall[j].y = day-POINT2(j).y+day; //flip wall.y about day
|
|
}
|
|
|
|
if (doMirror)
|
|
wall[j].cstat ^= 8; //mirror walls about dax/day
|
|
|
|
if (wall[j].point2==startofloop) //check if j is end of loop
|
|
{
|
|
endofloop = j;
|
|
if (about_x)
|
|
{
|
|
wall[endofloop].x = dax-x3+dax; //flip wall.x about dax
|
|
wall[endofloop].y = y3;
|
|
}
|
|
else
|
|
{
|
|
wall[endofloop].x = x3;
|
|
wall[endofloop].y = day-y3+day; //flip wall.y about dax
|
|
}
|
|
|
|
//correct order of walls in loop to maintain player space (right-hand rule)
|
|
numtoswap = (endofloop-startofloop)>>1;
|
|
for (w=1; w<=numtoswap; w++)
|
|
{
|
|
Bmemcpy(&tempwall, &wall[startofloop+w], sizeof(walltype));
|
|
Bmemcpy(&wall[startofloop+w], &wall[endofloop-w+1], sizeof(walltype));
|
|
Bmemcpy(&wall[endofloop-w+1], &tempwall, sizeof(walltype));
|
|
|
|
otonwall[startofloop+w] = endofloop-w+1;
|
|
otonwall[endofloop-w+1] = startofloop+w;
|
|
}
|
|
|
|
//make point2 point to next wall in loop
|
|
for (w=startofloop; w<endofloop; w++)
|
|
wall[w].point2 = w+1;
|
|
wall[endofloop].point2 = startofloop;
|
|
|
|
startofloop = endofloop+1; //set first wall of next loop
|
|
//save position of wall at start of loop
|
|
x3 = wall[startofloop].x;
|
|
y3 = wall[startofloop].y;
|
|
}
|
|
}
|
|
|
|
j = headspritesect[highlightsector[i]];
|
|
while (j != -1)
|
|
{
|
|
if (about_x)
|
|
{
|
|
x3 = sprite[j].x;
|
|
sprite[j].x = dax-x3+dax; //flip sprite.x about dax
|
|
sprite[j].ang = (1024+2048-sprite[j].ang)&2047; //flip ang about 512
|
|
}
|
|
else
|
|
{
|
|
y3 = sprite[j].y;
|
|
sprite[j].y = day-y3+day; //flip sprite.y about day
|
|
sprite[j].ang = (2048-sprite[j].ang)&2047; //flip ang about 512
|
|
}
|
|
|
|
if (doMirror && (sprite[j].cstat & 0x30))
|
|
sprite[j].cstat ^= 4; // mirror sprites about dax/day (don't mirror monsters)
|
|
|
|
j = nextspritesect[j];
|
|
}
|
|
}
|
|
|
|
// finally, construct the nextwalls and yax-nextwalls
|
|
// for the new arrangement!
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
for (WALLS_OF_SECTOR(highlightsector[i], j))
|
|
{
|
|
if (wall[j].nextwall >= 0)
|
|
wall[j].nextwall = otonwall[wall[j].nextwall];
|
|
#ifdef YAX_ENABLE
|
|
{
|
|
int32_t cf, ynw;
|
|
for (cf=0; cf<2; cf++)
|
|
if ((ynw = yax_getnextwall(j, cf)) >= 0)
|
|
yax_setnextwall(j, cf, otonwall[ynw]);
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
printmessage16("Selected sector(s) flipped");
|
|
asksave = 1;
|
|
}
|
|
}
|
|
// end edit for sector flip
|
|
|
|
if (keystatus[88]) //F12
|
|
{
|
|
keystatus[88] = 0;
|
|
//__clearscreen_beforecapture__
|
|
|
|
Bsnprintf(tempbuf, sizeof(tempbuf), "%s %s", AppProperName, CallExtGetVer());
|
|
screencapture("captxxxx.tga", eitherSHIFT, tempbuf);
|
|
|
|
showframe(1);
|
|
}
|
|
if (keystatus[0x30]) // B (clip Blocking xor) (2D)
|
|
{
|
|
pointhighlight = getpointhighlight(mousxplc, mousyplc, pointhighlight);
|
|
linehighlight = getlinehighlight(mousxplc, mousyplc, linehighlight, 0);
|
|
|
|
if ((pointhighlight&0xc000) == 16384)
|
|
{
|
|
sprite[pointhighlight&16383].cstat ^= 1;
|
|
sprite[pointhighlight&16383].cstat &= ~256;
|
|
sprite[pointhighlight&16383].cstat |= ((sprite[pointhighlight&16383].cstat&1)<<8);
|
|
asksave = 1;
|
|
}
|
|
else if (linehighlight >= 0)
|
|
{
|
|
wall[linehighlight].cstat ^= 1;
|
|
wall[linehighlight].cstat &= ~64;
|
|
if ((wall[linehighlight].nextwall >= 0) && !eitherSHIFT)
|
|
{
|
|
NEXTWALL(linehighlight).cstat &= ~(1+64);
|
|
NEXTWALL(linehighlight).cstat |= (wall[linehighlight].cstat&1);
|
|
}
|
|
asksave = 1;
|
|
}
|
|
keystatus[0x30] = 0;
|
|
}
|
|
if (keystatus[0x21]) //F (F alone does nothing in 2D right now)
|
|
{
|
|
keystatus[0x21] = 0;
|
|
if (eitherALT) //ALT-F (relative alignmment flip)
|
|
{
|
|
linehighlight = getlinehighlight(mousxplc, mousyplc, linehighlight, 0);
|
|
if (linehighlight >= 0)
|
|
SetFirstWall(sectorofwall(linehighlight), linehighlight, 1);
|
|
}
|
|
}
|
|
|
|
if (keystatus[0x18]) // O (ornament onto wall) (2D)
|
|
{
|
|
keystatus[0x18] = 0;
|
|
if ((pointhighlight&0xc000) == 16384)
|
|
{
|
|
asksave = 1;
|
|
DoSpriteOrnament(pointhighlight&16383);
|
|
}
|
|
}
|
|
|
|
|
|
tsign = 0;
|
|
if (keystatus[0x33] || (bstatus&33)==33) // , (2D)
|
|
tsign = +1;
|
|
if (keystatus[0x34] || (bstatus&17)==17) // . (2D)
|
|
tsign = -1;
|
|
|
|
if (tsign)
|
|
{
|
|
#ifdef YAX_ENABLE
|
|
if (highlightsectorcnt > 0 && !hl_all_bunch_sectors_p())
|
|
{
|
|
printmessage16("To rotate ext. sectors, all sectors of a bunch must be selected");
|
|
}
|
|
else
|
|
#endif
|
|
if (highlightsectorcnt > 0)
|
|
{
|
|
int32_t smoothRotation = eitherSHIFT, manualAngle = eitherALT;
|
|
|
|
if (manualAngle)
|
|
{
|
|
tsign = getnumber16("Rotation BUILD angle: ", 0, 2047, 1);
|
|
if (tsign==0)
|
|
{
|
|
printmessage16(" ");
|
|
goto rotate_hlsect_out;
|
|
}
|
|
|
|
printmessage16("Rotated highlighted sectors by %d BUILD degrees", tsign);
|
|
tsign &= 2047;
|
|
smoothRotation = 1;
|
|
}
|
|
|
|
get_sectors_center(highlightsector, highlightsectorcnt, &dax, &day);
|
|
|
|
if (!smoothRotation)
|
|
{
|
|
if (gridlock && grid > 0)
|
|
locktogrid(&dax, &day);
|
|
|
|
tsign *= 512;
|
|
}
|
|
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
for (WALLS_OF_SECTOR(highlightsector[i], j))
|
|
rotatepoint(dax,day, wall[j].x,wall[j].y, tsign&2047, &wall[j].x,&wall[j].y);
|
|
|
|
for (j=headspritesect[highlightsector[i]]; j != -1; j=nextspritesect[j])
|
|
{
|
|
rotatepoint(dax,day, sprite[j].x,sprite[j].y, tsign&2047, &sprite[j].x,&sprite[j].y);
|
|
sprite[j].ang = (sprite[j].ang+tsign)&2047;
|
|
}
|
|
}
|
|
|
|
m32_rotateang += tsign;
|
|
m32_rotateang &= 2047;
|
|
asksave = 1;
|
|
rotate_hlsect_out:
|
|
if (!smoothRotation || manualAngle)
|
|
keystatus[0x33] = keystatus[0x34] = 0;
|
|
|
|
mouseb &= ~(16|32);
|
|
bstatus &= ~(16|32);
|
|
}
|
|
else
|
|
{
|
|
if (pointhighlight >= 16384)
|
|
{
|
|
i = pointhighlight-16384;
|
|
if (eitherSHIFT)
|
|
sprite[i].ang = (sprite[i].ang-tsign)&2047;
|
|
else
|
|
{
|
|
sprite[i].ang = (sprite[i].ang-128*tsign)&2047;
|
|
keystatus[0x33] = keystatus[0x34] = 0;
|
|
}
|
|
|
|
mouseb &= ~(16|32);
|
|
bstatus &= ~(16|32);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (keystatus[0x46]) //Scroll lock (set starting position)
|
|
{
|
|
startpos = pos;
|
|
startang = ang;
|
|
startsectnum = cursectnum;
|
|
keystatus[0x46] = 0;
|
|
asksave = 1;
|
|
|
|
printmessage16("Set starting position");
|
|
}
|
|
#if 1
|
|
if (keystatus[0x3f]) //F5
|
|
{
|
|
CallExtShowSectorData(0);
|
|
}
|
|
if (keystatus[0x40]) //F6
|
|
{
|
|
if (pointhighlight >= 16384)
|
|
CallExtShowSpriteData(pointhighlight-16384);
|
|
else if (linehighlight >= 0)
|
|
CallExtShowWallData(linehighlight);
|
|
else
|
|
CallExtShowWallData(0);
|
|
}
|
|
if (keystatus[0x41]) //F7
|
|
{
|
|
keystatus[0x41] = 0;
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
if (inside_editor_curpos(i) == 1)
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
|
|
CallExtEditSectorData(i);
|
|
break;
|
|
}
|
|
}
|
|
if (keystatus[0x42]) //F8
|
|
{
|
|
keystatus[0x42] = 0;
|
|
|
|
if (pointhighlight >= 16384)
|
|
CallExtEditSpriteData(pointhighlight-16384);
|
|
else if (linehighlight >= 0)
|
|
CallExtEditWallData(linehighlight);
|
|
}
|
|
#endif
|
|
|
|
if (keystatus[0x23]) //H (Hi 16 bits of tag)
|
|
{
|
|
keystatus[0x23] = 0;
|
|
if (eitherCTRL) //Ctrl-H
|
|
{
|
|
pointhighlight = getpointhighlight(mousxplc, mousyplc, pointhighlight);
|
|
linehighlight = getlinehighlight(mousxplc, mousyplc, linehighlight, 0);
|
|
|
|
if ((pointhighlight&0xc000) == 16384)
|
|
{
|
|
sprite[pointhighlight&16383].cstat ^= 256;
|
|
asksave = 1;
|
|
}
|
|
else if (linehighlight >= 0)
|
|
{
|
|
wall[linehighlight].cstat ^= 64;
|
|
if ((wall[linehighlight].nextwall >= 0) && !eitherSHIFT)
|
|
{
|
|
NEXTWALL(linehighlight).cstat &= ~64;
|
|
NEXTWALL(linehighlight).cstat |= (wall[linehighlight].cstat&64);
|
|
}
|
|
asksave = 1;
|
|
}
|
|
}
|
|
else if (eitherALT) //ALT
|
|
{
|
|
if (pointhighlight >= 16384)
|
|
{
|
|
i = pointhighlight-16384;
|
|
j = taglab_linktags(1, i);
|
|
j = 2*(j&2);
|
|
Bsprintf(buffer, "Sprite (%d) Hi-tag: ", i);
|
|
sprite[i].hitag = getnumber16(buffer, sprite[i].hitag, BTAG_MAX, 0+j);
|
|
}
|
|
else if (linehighlight >= 0)
|
|
{
|
|
i = linehighlight;
|
|
j = taglab_linktags(1, i);
|
|
j = 2*(j&2);
|
|
Bsprintf(buffer, "Wall (%d) Hi-tag: ", i);
|
|
wall[i].hitag = getnumber16(buffer, wall[i].hitag, BTAG_MAX, 0+j);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (i=0; i<numsectors; i++)
|
|
if (inside_editor_curpos(i) == 1)
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
|
|
Bsprintf(buffer, "Sector (%d) Hi-tag: ", i);
|
|
sector[i].hitag = getnumber16(buffer, sector[i].hitag, BTAG_MAX, 0);
|
|
break;
|
|
}
|
|
}
|
|
// printmessage16("");
|
|
}
|
|
if (keystatus[0x19]) // P (palookup #)
|
|
{
|
|
keystatus[0x19] = 0;
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
if (inside_editor_curpos(i) == 1)
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
|
|
Bsprintf(buffer, "Sector (%d) Ceilingpal: ", i);
|
|
sector[i].ceilingpal = getnumber16(buffer, sector[i].ceilingpal, M32_MAXPALOOKUPS, 0);
|
|
|
|
Bsprintf(buffer, "Sector (%d) Floorpal: ", i);
|
|
sector[i].floorpal = getnumber16(buffer, sector[i].floorpal, M32_MAXPALOOKUPS, 0);
|
|
break;
|
|
}
|
|
}
|
|
if (keystatus[0x12]) // E (status list)
|
|
{
|
|
keystatus[0x12] = 0;
|
|
|
|
if (!eitherCTRL)
|
|
{
|
|
if (pointhighlight >= 16384)
|
|
{
|
|
i = pointhighlight-16384;
|
|
Bsprintf(buffer, "Sprite (%d) Status list: ", i);
|
|
changespritestat(i, getnumber16(buffer, sprite[i].statnum, MAXSTATUS-1, 0));
|
|
}
|
|
}
|
|
#ifdef YAX_ENABLE
|
|
else if (highlightsectorcnt > 0 && newnumwalls < 0)
|
|
{
|
|
////////// YAX //////////
|
|
static const char *cfs[2] = {"ceiling", "floor"};
|
|
|
|
int32_t cf, thez, ulz[2]={0,0};
|
|
int16_t bn, sandwichbunch=-1;
|
|
|
|
if (numyaxbunches==YAX_MAXBUNCHES)
|
|
{
|
|
message("Bunch limit of %d reached, cannot extend", YAX_MAXBUNCHES);
|
|
goto end_yax;
|
|
}
|
|
|
|
if (highlighted_sectors_components(0,0) != 1)
|
|
{
|
|
message("Sectors to extend must be in one connected component");
|
|
goto end_yax;
|
|
}
|
|
|
|
cf = ask_above_or_below();
|
|
if (cf==-1)
|
|
goto end_yax;
|
|
|
|
thez = SECTORFLD(highlightsector[0],z, cf);
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
bn = yax_getbunch(highlightsector[i], cf);
|
|
|
|
if (sandwichbunch >= 0 && bn!=sandwichbunch)
|
|
{
|
|
message("When sandwiching extension, must select only sectors of one bunch");
|
|
goto end_yax;
|
|
}
|
|
|
|
if (bn >= 0)
|
|
{
|
|
if (cf==YAX_FLOOR)
|
|
{
|
|
if (sandwichbunch < 0 && i!=0)
|
|
{
|
|
message("When sandwiching extension, must select only sectors of the bunch");
|
|
goto end_yax;
|
|
}
|
|
sandwichbunch = bn;
|
|
}
|
|
else
|
|
{
|
|
message("Sector %d's %s is already extended", highlightsector[i], cfs[cf]);
|
|
goto end_yax;
|
|
}
|
|
}
|
|
|
|
if (SECTORFLD(highlightsector[i],z, cf) != thez)
|
|
{
|
|
message("Sector %d's %s height doesn't match sector %d's",
|
|
highlightsector[i], cfs[cf], highlightsector[0]);
|
|
goto end_yax;
|
|
}
|
|
|
|
if ((sandwichbunch>=0 || highlightsectorcnt>1) && SECTORFLD(highlightsector[i],stat, cf)&2)
|
|
{
|
|
message("Sector %ss must not be sloped%s", cfs[cf],
|
|
sandwichbunch>=0 ? "" : "if extending more than one");
|
|
goto end_yax;
|
|
}
|
|
}
|
|
|
|
if (sandwichbunch >= 0)
|
|
{
|
|
// cf==YAX_FLOOR here
|
|
|
|
int32_t tempz, oldfz, swsecheight = DEFAULT_YAX_HEIGHT/4;
|
|
// highest floor z of lower sectors, lowest ceiling z of these sectors
|
|
int32_t minfloorz = INT32_MAX, maxceilz = INT32_MIN;
|
|
|
|
// some preparation for making the sandwich
|
|
if (highlightsectorcnt != yax_numsectsinbunch(sandwichbunch, YAX_FLOOR))
|
|
{
|
|
message("When sandwiching extension, must select all sectors of the bunch");
|
|
goto end_yax;
|
|
}
|
|
|
|
// "for i in sectors of sandwichbunch(floor)" is now the same as
|
|
// "for i in highlighted sectors"
|
|
|
|
oldfz = sector[highlightsector[0]].floorz;
|
|
|
|
// check if enough room in z
|
|
for (SECTORS_OF_BUNCH(sandwichbunch, YAX_CEILING, i))
|
|
for (WALLS_OF_SECTOR(i, j))
|
|
{
|
|
tempz = getflorzofslope(i, wall[j].x, wall[j].y);
|
|
minfloorz = min(minfloorz, tempz);
|
|
}
|
|
for (SECTORS_OF_BUNCH(sandwichbunch, YAX_FLOOR, i))
|
|
for (WALLS_OF_SECTOR(i, j))
|
|
{
|
|
tempz = getceilzofslope(i, wall[j].x, wall[j].y);
|
|
maxceilz = max(maxceilz, tempz);
|
|
}
|
|
|
|
if (minfloorz - maxceilz < 2*swsecheight)
|
|
{
|
|
message("Too little z headroom for sandwiching, need at least %d",
|
|
2*swsecheight);
|
|
goto end_yax;
|
|
}
|
|
|
|
if (maxceilz >= oldfz || oldfz >= minfloorz)
|
|
{
|
|
message("Internal error while sandwiching: oldfz out of bounds");
|
|
goto end_yax;
|
|
}
|
|
|
|
// maxceilz ---|
|
|
// ^ |
|
|
// ulz[0] ^
|
|
// ^ oldfz
|
|
// ulz[1] ^
|
|
// ^ |
|
|
// minfloorz ---|
|
|
|
|
ulz[0] = (int32_t)(oldfz - swsecheight*((double)(oldfz-maxceilz)/(minfloorz-maxceilz)));
|
|
ulz[0] &= ~255;
|
|
ulz[1] = ulz[0] + swsecheight;
|
|
|
|
if (maxceilz >= ulz[0] || ulz[1] >= minfloorz)
|
|
{
|
|
message("Too little z headroom for sandwiching");
|
|
goto end_yax;
|
|
}
|
|
}
|
|
|
|
m = numwalls;
|
|
Bmemset(visited, 0, sizeof(visited));
|
|
// construct!
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
for (WALLS_OF_SECTOR(highlightsector[i], j))
|
|
{
|
|
k = trace_loop(j, visited, NULL, NULL, !cf);
|
|
if (k == 0)
|
|
continue;
|
|
else if (k < 0)
|
|
{
|
|
numwalls = m;
|
|
goto end_yax;
|
|
}
|
|
//message("loop");
|
|
wall[k-1].point2 = numwalls;
|
|
numwalls = k;
|
|
}
|
|
|
|
for (i=m; i<numwalls; i++) // try
|
|
{
|
|
j = YAX_NEXTWALL(i, !cf);
|
|
if (j < 0)
|
|
{
|
|
message("Internal error while constructing sector: "
|
|
"YAX_NEXTWALL(%d, %d)<0!", i, !cf);
|
|
numwalls = m;
|
|
goto end_yax;
|
|
}
|
|
if (sandwichbunch >= 0)
|
|
{
|
|
if (YAX_NEXTWALL(j, cf) < 0)
|
|
{
|
|
message("Internal error while sandwiching (2): "
|
|
"YAX_NEXTWALL(%d, %d)<0!", j, cf);
|
|
numwalls = m;
|
|
goto end_yax;
|
|
}
|
|
}
|
|
}
|
|
for (i=m; i<numwalls; i++) // do!
|
|
{
|
|
j = YAX_NEXTWALL(i, !cf);
|
|
|
|
if (sandwichbunch >= 0)
|
|
{
|
|
int16_t oynw = YAX_NEXTWALL(j, cf);
|
|
yax_setnextwall(j, cf, i);
|
|
yax_setnextwall(i, cf, oynw);
|
|
yax_setnextwall(oynw, !cf, i);
|
|
}
|
|
else
|
|
{
|
|
yax_setnextwall(j, cf, i);
|
|
}
|
|
}
|
|
|
|
// create new sector based on first highlighted one
|
|
i = highlightsector[0];
|
|
Bmemcpy(§or[numsectors], §or[i], sizeof(sectortype));
|
|
sector[numsectors].wallptr = m;
|
|
sector[numsectors].wallnum = numwalls-m;
|
|
|
|
if (sandwichbunch < 0)
|
|
{
|
|
if (SECTORFLD(i,stat, cf)&2)
|
|
setslope(numsectors, !cf, SECTORFLD(i,heinum, cf));
|
|
else
|
|
setslope(numsectors, !cf, 0);
|
|
setslope(numsectors, cf, 0);
|
|
|
|
SECTORFLD(numsectors,z, !cf) = SECTORFLD(i,z, cf);
|
|
SECTORFLD(numsectors,z, cf) = SECTORFLD(i,z, cf) - (1-2*cf)*DEFAULT_YAX_HEIGHT;
|
|
}
|
|
else
|
|
{
|
|
for (SECTORS_OF_BUNCH(sandwichbunch, cf, i))
|
|
sector[i].floorz = ulz[0];
|
|
sector[numsectors].ceilingz = ulz[0];
|
|
sector[numsectors].floorz = ulz[1];
|
|
for (SECTORS_OF_BUNCH(sandwichbunch, !cf, i))
|
|
sector[i].ceilingz = ulz[1];
|
|
}
|
|
|
|
newnumwalls = numwalls;
|
|
numwalls = m;
|
|
|
|
SECTORFLD(numsectors,stat, !cf) &= ~1; // no plax
|
|
|
|
// restore red walls of the selected sectors
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
SECTORFLD(highlightsector[i],stat, cf) &= ~1; // no plax
|
|
|
|
for (WALLS_OF_SECTOR(highlightsector[i], j))
|
|
if (wall[j].nextwall < 0)
|
|
checksectorpointer(j, highlightsector[i]);
|
|
}
|
|
|
|
// link
|
|
if (sandwichbunch < 0)
|
|
{
|
|
yax_setbunch(numsectors, !cf, numyaxbunches);
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
yax_setbunch(highlightsector[i], cf, numyaxbunches);
|
|
}
|
|
else
|
|
{
|
|
yax_setbunch(numsectors, !cf, sandwichbunch);
|
|
// also relink
|
|
yax_setbunch(numsectors, cf, numyaxbunches);
|
|
for (SECTORS_OF_BUNCH(sandwichbunch, !cf, i))
|
|
yax_setbunch(i, !cf, numyaxbunches);
|
|
}
|
|
|
|
numwalls = newnumwalls;
|
|
newnumwalls = -1;
|
|
|
|
numsectors++;
|
|
yax_update(0);
|
|
yax_updategrays(pos.z);
|
|
|
|
reset_highlightsector();
|
|
|
|
if (sandwichbunch < 0)
|
|
message("Extended %ss of highlighted sectors, creating bunch %d",
|
|
cfs[cf], numyaxbunches-1);
|
|
else
|
|
message("Sandwiched bunch %d, creating bunch %d",
|
|
sandwichbunch, numyaxbunches-1);
|
|
asksave = 1;
|
|
}
|
|
else if (highlightcnt > 0)
|
|
{
|
|
/// 'punch' wall loop through extension
|
|
|
|
int32_t loopstartwall = -1, numloopwalls, cf;
|
|
int32_t srcsect, dstsect, ofirstwallofs;
|
|
int16_t cb, fb, bunchnum;
|
|
|
|
if (EDITING_MAP_P())
|
|
{
|
|
printmessage16("Must not be editing map to punch loop");
|
|
goto end_yax;
|
|
}
|
|
|
|
if (numyaxbunches >= YAX_MAXBUNCHES)
|
|
{
|
|
message("TROR bunch limit reached, cannot punch loop");
|
|
goto end_yax;
|
|
}
|
|
|
|
// determine start wall
|
|
for (i=0; i<highlightcnt; i++)
|
|
{
|
|
j = highlight[i];
|
|
|
|
if (j&16384)
|
|
continue;
|
|
|
|
// we only want loop-starting walls
|
|
if (j>0 && wall[j-1].point2==j)
|
|
continue;
|
|
|
|
if (clockdir(j)==CLOCKDIR_CCW)
|
|
{
|
|
YAX_SKIPWALL(j);
|
|
|
|
if (loopstartwall >= 0)
|
|
{
|
|
message("Must have a unique highlighted CCW loop to punch");
|
|
goto end_yax;
|
|
}
|
|
|
|
loopstartwall = j;
|
|
}
|
|
}
|
|
|
|
if (loopstartwall == -1)
|
|
{
|
|
message("Didn't find any non-grayed out CCW loop start walls");
|
|
goto end_yax;
|
|
}
|
|
|
|
// determine sector
|
|
srcsect = sectorofwall(loopstartwall);
|
|
yax_getbunches(srcsect, &cb, &fb);
|
|
if (cb < 0 && fb < 0)
|
|
{
|
|
message("Ceiling or floor must be extended to punch loop");
|
|
goto end_yax;
|
|
}
|
|
|
|
/// determine c/f
|
|
cf = -1;
|
|
if (fb < 0)
|
|
cf = YAX_CEILING;
|
|
else if (cb < 0)
|
|
cf = YAX_FLOOR;
|
|
|
|
fade_editor_screen(-1);
|
|
|
|
// query top/bottom
|
|
if (cf == -1)
|
|
{
|
|
char dachars[2] = {'a', 'z'};
|
|
cf = editor_ask_function("Punch loop above (a) or below (z)?", dachars, 2);
|
|
if (cf == -1)
|
|
goto end_yax;
|
|
}
|
|
else
|
|
{
|
|
// ask even if only one choice -- I find it more
|
|
// consistent with 'extend sector' this way
|
|
if (-1 == editor_ask_function(cf==YAX_CEILING ? "Punch loop above (a)?" :
|
|
"Punch loop below (z)?", cf==YAX_CEILING?"a":"z", 1))
|
|
goto end_yax;
|
|
}
|
|
|
|
bunchnum = (cf==YAX_CEILING) ? cb : fb;
|
|
|
|
// check 1
|
|
j = loopstartwall; // will be real start wall of loop
|
|
numloopwalls = 1; // will be number of walls in loop
|
|
for (i=wall[loopstartwall].point2; i!=loopstartwall; i=wall[i].point2)
|
|
{
|
|
numloopwalls++;
|
|
if (i < j)
|
|
j = i;
|
|
|
|
if ((show2dwall[i>>3]&(1<<(i&7)))==0)
|
|
{
|
|
message("All loop points must be highlighted to punch");
|
|
goto end_yax;
|
|
}
|
|
|
|
if (yax_getnextwall(loopstartwall, cf) >= 0 || yax_getnextwall(i, cf) >= 0)
|
|
{
|
|
// somewhat redundant, since it would also be caught by check 2
|
|
message("Loop walls must not already have TROR neighbors");
|
|
goto end_yax;
|
|
}
|
|
|
|
if (wall[loopstartwall].nextwall < 0 || wall[i].nextwall < 0)
|
|
{
|
|
message("INTERNAL ERROR: All loop walls are expected to be red");
|
|
goto end_yax;
|
|
}
|
|
}
|
|
loopstartwall = j;
|
|
|
|
if (numwalls + 2*numloopwalls > MAXWALLS || numsectors+1 > MAXSECTORS)
|
|
{
|
|
message("Punching loop through extension would exceed limits");
|
|
goto end_yax;
|
|
}
|
|
|
|
// get other-side sector, j==loopstartwall
|
|
dstsect = yax_getneighborsect(wall[j].x, wall[j].y, srcsect, cf);
|
|
if (dstsect < 0)
|
|
{
|
|
message("Punch loop INTERNAL ERROR: dstsect < 0. Map corrupt?");
|
|
goto end_yax;
|
|
}
|
|
|
|
// check 2
|
|
i = loopstartwall;
|
|
do
|
|
{
|
|
j = wall[i].point2;
|
|
|
|
for (WALLS_OF_SECTOR(dstsect, k))
|
|
{
|
|
vec2_t pint;
|
|
if (lineintersect2v((vec2_t *)&wall[i], (vec2_t *)&wall[j],
|
|
(vec2_t *)&wall[k], (vec2_t *)&POINT2(k), &pint))
|
|
{
|
|
message("Loop lines must not intersect any destination sector's walls");
|
|
goto end_yax;
|
|
}
|
|
}
|
|
}
|
|
while ((i = j) != loopstartwall);
|
|
|
|
// construct new loop and (dummy yet) sector
|
|
Bmemcpy(&wall[numwalls], &wall[loopstartwall], numloopwalls*sizeof(walltype));
|
|
newnumwalls = numwalls+numloopwalls;
|
|
|
|
for (i=numwalls; i<newnumwalls; i++)
|
|
{
|
|
wall[i].point2 += (numwalls - loopstartwall);
|
|
wall[i].nextsector = wall[i].nextwall = -1;
|
|
}
|
|
|
|
sector[numsectors].wallptr = numwalls;
|
|
sector[numsectors].wallnum = numloopwalls;
|
|
numsectors++; // temp
|
|
|
|
// check 3
|
|
for (SECTORS_OF_BUNCH(bunchnum, !cf, i))
|
|
for (WALLS_OF_SECTOR(i, j))
|
|
{
|
|
if (inside(wall[j].x, wall[j].y, numsectors-1)==1)
|
|
{
|
|
numsectors--;
|
|
newnumwalls = -1;
|
|
message("A point of bunch %d's sectors lies inside the loop to punch",
|
|
bunchnum);
|
|
goto end_yax;
|
|
}
|
|
}
|
|
|
|
numsectors--;
|
|
|
|
// clear wall & sprite highlights
|
|
// TODO: see about consistency with update_highlight() after other ops
|
|
reset_highlight();
|
|
|
|
// construct the loop!
|
|
i = AddLoopToSector(dstsect, &ofirstwallofs);
|
|
|
|
if (i <= 0)
|
|
{
|
|
message("Punch loop INTERNAL ERROR with AddLoopToSector!");
|
|
}
|
|
else
|
|
{
|
|
int32_t oneinnersect = -1, innerdstsect = numsectors-1;
|
|
|
|
if (dstsect < srcsect)
|
|
loopstartwall += numloopwalls;
|
|
|
|
/// handle bunchnums! (specifically, create a new one)
|
|
|
|
// collect sectors inside source loop; for that, first break the
|
|
// inner->outer nextwall links
|
|
for (i=loopstartwall; i<loopstartwall+numloopwalls; i++)
|
|
{
|
|
// all src loop walls are red!
|
|
NEXTWALL(i).nextwall = NEXTWALL(i).nextsector = -1;
|
|
oneinnersect = wall[i].nextsector;
|
|
}
|
|
|
|
// vvv
|
|
// expect oneinnersect >= 0 here! Assumption: we collect exactly
|
|
// one connected component of sectors
|
|
collect_sectors1(collsectlist[0], collsectbitmap[0],
|
|
&collnumsects[0], oneinnersect, 0, 0);
|
|
|
|
// set new bunchnums
|
|
for (i=0; i<collnumsects[0]; i++)
|
|
yax_setbunch(collsectlist[0][i], cf, numyaxbunches);
|
|
yax_setbunch(innerdstsect, !cf, numyaxbunches);
|
|
// ^^^
|
|
|
|
// restore inner->outer nextwall links
|
|
for (i=loopstartwall; i<loopstartwall+numloopwalls; i++)
|
|
{
|
|
NEXTWALL(i).nextwall = i;
|
|
NEXTWALL(i).nextsector = srcsect;
|
|
|
|
// set yax-nextwalls!
|
|
j = (i-loopstartwall) + sector[dstsect].wallptr;
|
|
yax_setnextwall(i, cf, j);
|
|
yax_setnextwall(j, !cf, i);
|
|
|
|
yax_setnextwall(wall[i].nextwall, cf, wall[j].nextwall);
|
|
yax_setnextwall(wall[j].nextwall, !cf, wall[i].nextwall);
|
|
}
|
|
|
|
setfirstwall(dstsect, sector[dstsect].wallptr+ofirstwallofs);
|
|
|
|
message("Punched loop starting w/ wall %d into %s sector %d%s",
|
|
loopstartwall, cf==YAX_CEILING?"upper":"lower", dstsect,
|
|
(oneinnersect>=0) ? "" : " (ERRORS)");
|
|
}
|
|
|
|
mkonwinvalid();
|
|
asksave = 1;
|
|
|
|
yax_update(0);
|
|
yax_updategrays(pos.z);
|
|
}
|
|
end_yax: ;
|
|
#endif
|
|
}
|
|
|
|
if (highlightsectorcnt < 0)
|
|
{
|
|
if (keystatus[0x36]) //Right shift (point highlighting)
|
|
{
|
|
if (highlightcnt == 0)
|
|
{
|
|
int32_t xx[] = { highlightx1, highlightx1, searchx, searchx, highlightx1 };
|
|
int32_t yy[] = { highlighty1, searchy, searchy, highlighty1, highlighty1 };
|
|
|
|
highlightx2 = searchx;
|
|
highlighty2 = searchy;
|
|
ydim16 = ydim-STATUS2DSIZ2;
|
|
|
|
plotlines2d(xx, yy, 5, editorcolors[5]);
|
|
}
|
|
else
|
|
{
|
|
highlightcnt = 0;
|
|
|
|
highlightx1 = searchx;
|
|
highlighty1 = searchy;
|
|
highlightx2 = searchx;
|
|
highlighty2 = searchy;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (highlightcnt == 0)
|
|
{
|
|
int32_t add=keystatus[0x28], sub=(!add && keystatus[0x27]), setop=(add||sub);
|
|
|
|
if (!m32_sideview)
|
|
{
|
|
getpoint(highlightx1,highlighty1, &highlightx1,&highlighty1);
|
|
getpoint(highlightx2,highlighty2, &highlightx2,&highlighty2);
|
|
}
|
|
|
|
if (highlightx1 > highlightx2)
|
|
swaplong(&highlightx1, &highlightx2);
|
|
if (highlighty1 > highlighty2)
|
|
swaplong(&highlighty1, &highlighty2);
|
|
|
|
// Ctrl+RShift: select all wall-points of highlighted wall's loop:
|
|
if (eitherCTRL && highlightx1==highlightx2 && highlighty1==highlighty2)
|
|
{
|
|
if (!setop)
|
|
{
|
|
Bmemset(show2dwall, 0, sizeof(show2dwall));
|
|
Bmemset(show2dsprite, 0, sizeof(show2dsprite));
|
|
}
|
|
|
|
if (linehighlight >= 0 && linehighlight < MAXWALLS)
|
|
{
|
|
i = linehighlight;
|
|
do
|
|
{
|
|
if (!sub)
|
|
show2dwall[i>>3] |= (1<<(i&7));
|
|
else
|
|
show2dwall[i>>3] &= ~(1<<(i&7));
|
|
|
|
// XXX: this selects too many walls, need something more like
|
|
// those of dragpoint() -- could be still too many for
|
|
// loop punching though
|
|
for (j=0; j<numwalls; j++)
|
|
if (j!=i && wall[j].x==wall[i].x && wall[j].y==wall[i].y)
|
|
{
|
|
if (!sub)
|
|
show2dwall[j>>3] |= (1<<(j&7));
|
|
else
|
|
show2dwall[j>>3] &= ~(1<<(j&7));
|
|
}
|
|
|
|
i = wall[i].point2;
|
|
}
|
|
while (i != linehighlight);
|
|
}
|
|
|
|
update_highlight();
|
|
}
|
|
else
|
|
{
|
|
int32_t tx, ty, onlySprites=eitherCTRL;
|
|
int32_t accum_dragged_verts = 0;
|
|
|
|
if (!setop)
|
|
{
|
|
Bmemset(show2dwall, 0, sizeof(show2dwall));
|
|
Bmemset(show2dsprite, 0, sizeof(show2dsprite));
|
|
}
|
|
|
|
for (i=0; i<numwalls; i++)
|
|
wall[i].cstat &= ~(1<<14);
|
|
|
|
for (i=0; i<numwalls; i++)
|
|
{
|
|
if (onlySprites)
|
|
break;
|
|
|
|
YAX_SKIPWALL(i);
|
|
|
|
if (!m32_sideview)
|
|
{
|
|
tx = wall[i].x;
|
|
ty = wall[i].y;
|
|
// wall[i].cstat &= ~(1<<14);
|
|
}
|
|
else
|
|
{
|
|
screencoords(&tx,&ty, wall[i].x-pos.x,wall[i].y-pos.y, zoom);
|
|
ty += getscreenvdisp(
|
|
getflorzofslope(sectorofwall(i), wall[i].x,wall[i].y)-pos.z, zoom);
|
|
tx += halfxdim16;
|
|
ty += midydim16;
|
|
}
|
|
|
|
if (tx >= highlightx1 && tx <= highlightx2 &&
|
|
ty >= highlighty1 && ty <= highlighty2)
|
|
{
|
|
if (!sub)
|
|
{
|
|
if (numgraysects > 0 || m32_sideview)
|
|
{
|
|
// Only called to find out which walls would get dragged:
|
|
dragpoint(i, wall[i].x, wall[i].y, accum_dragged_verts);
|
|
accum_dragged_verts = 1;
|
|
}
|
|
else
|
|
show2dwall[i>>3] |= (1<<(i&7));
|
|
}
|
|
else
|
|
show2dwall[i>>3] &= ~(1<<(i&7));
|
|
}
|
|
}
|
|
|
|
if (!sub && (numgraysects > 0 || m32_sideview))
|
|
{
|
|
for (i=0; i<numwalls; i++)
|
|
if (wall[i].cstat&(1<<14))
|
|
show2dwall[i>>3] |= (1<<(i&7));
|
|
}
|
|
|
|
for (i=0; i<MAXSPRITES; i++)
|
|
{
|
|
if (sprite[i].statnum == MAXSTATUS)
|
|
continue;
|
|
|
|
// v v v: if END pressed, also permit sprites from grayed out sectors
|
|
if (!keystatus[0xcf] && (unsigned)sprite[i].sectnum < MAXSECTORS)
|
|
YAX_SKIPSECTOR(sprite[i].sectnum);
|
|
|
|
if (!m32_sideview)
|
|
{
|
|
tx = sprite[i].x;
|
|
ty = sprite[i].y;
|
|
}
|
|
else
|
|
{
|
|
screencoords(&tx,&ty, sprite[i].x-pos.x,sprite[i].y-pos.y, zoom);
|
|
ty += getscreenvdisp(sprite[i].z-pos.z, zoom);
|
|
tx += halfxdim16;
|
|
ty += midydim16;
|
|
}
|
|
|
|
if (tx >= highlightx1 && tx <= highlightx2 &&
|
|
ty >= highlighty1 && ty <= highlighty2)
|
|
{
|
|
if (!sub)
|
|
{
|
|
if (sprite[i].sectnum >= 0) // don't allow to select sprites in null space
|
|
show2dsprite[i>>3] |= (1<<(i&7));
|
|
}
|
|
else
|
|
show2dsprite[i>>3] &= ~(1<<(i&7));
|
|
}
|
|
}
|
|
|
|
update_highlight();
|
|
|
|
for (i=0; i<numwalls; i++)
|
|
wall[i].cstat &= ~(1<<14);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (highlightcnt < 0)
|
|
{
|
|
if (keystatus[0xb8]) //Right alt (sector highlighting)
|
|
{
|
|
if (highlightsectorcnt == 0)
|
|
{
|
|
if (!eitherCTRL)
|
|
{
|
|
int32_t xx[] = { highlightx1, highlightx1, searchx, searchx, highlightx1 };
|
|
int32_t yy[] = { highlighty1, searchy, searchy, highlighty1, highlighty1 };
|
|
|
|
highlightx2 = searchx;
|
|
highlighty2 = searchy;
|
|
ydim16 = ydim-STATUS2DSIZ2;
|
|
|
|
plotlines2d(xx, yy, 5, editorcolors[10]);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// didmakered: 'bad'!
|
|
int32_t didmakered = (highlightsectorcnt<0), hadouterpoint=0;
|
|
#ifdef YAX_ENABLE
|
|
for (i=0; i<MAXSECTORS>>3; i++)
|
|
hlorgraysectbitmap[i] = hlsectorbitmap[i]|graysectbitmap[i];
|
|
#endif
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
int16_t tmpsect = -1;
|
|
|
|
for (WALLS_OF_SECTOR(highlightsector[i], j))
|
|
{
|
|
// if (wall[j].nextwall >= 0)
|
|
// checksectorpointer(wall[j].nextwall,wall[j].nextsector);
|
|
if (wall[j].nextwall < 0)
|
|
didmakered |= !!checksectorpointer(j, highlightsector[i]);
|
|
|
|
if (!didmakered)
|
|
{
|
|
updatesectorexclude(wall[j].x, wall[j].y, &tmpsect, hlorgraysectbitmap);
|
|
if (tmpsect<0)
|
|
hadouterpoint = 1;
|
|
}
|
|
}
|
|
#ifdef YAX_ENABLE
|
|
{
|
|
int16_t cb, fb;
|
|
|
|
yax_getbunches(highlightsector[i], &cb, &fb);
|
|
if (cb>=0 || fb>=0)
|
|
{
|
|
// TROR stuff in the pasted sectors would really
|
|
// complicate things, so don't allow this
|
|
didmakered=1;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
if (!didmakered && !hadouterpoint && newnumwalls<0)
|
|
{
|
|
// fade the screen to have the user's attention
|
|
fade_editor_screen(-1);
|
|
|
|
didmakered |= !ask_if_sure("Insert outer loop and make red walls? (Y/N)", 0);
|
|
clearkeys();
|
|
}
|
|
|
|
if (!didmakered && !hadouterpoint && newnumwalls<0)
|
|
{
|
|
int16_t ignore, refsect;
|
|
int32_t n;
|
|
#ifdef YAX_ENABLE
|
|
int16_t refsectbn[2]={-1,-1};
|
|
int32_t refextcf=-1;
|
|
#endif
|
|
Bmemset(visited, 0, sizeof(visited));
|
|
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
for (WALLS_OF_SECTOR(highlightsector[i], j))
|
|
{
|
|
k = trace_loop(j, visited, &ignore, &refsect, -1);
|
|
if (k == 0)
|
|
continue;
|
|
else if (k < 0)
|
|
goto end_autoredwall;
|
|
|
|
if (ignore)
|
|
continue;
|
|
|
|
// done tracing one outer loop
|
|
#ifdef YAX_ENABLE
|
|
yax_getbunches(refsect, &refsectbn[0], &refsectbn[1]);
|
|
if (refsectbn[0]>=0 || refsectbn[1]>=0)
|
|
{
|
|
if (refsectbn[0]>=0 && refsectbn[1]>=0)
|
|
{
|
|
// at least one of ceiling/floor must be non-extended
|
|
didmakered = 1;
|
|
}
|
|
else
|
|
{
|
|
// ... and the other must be non-sloped
|
|
refextcf = (refsectbn[1]>=0);
|
|
if (SECTORFLD(refsect,stat, !refextcf)&2)
|
|
didmakered = 1;
|
|
}
|
|
}
|
|
|
|
if (didmakered)
|
|
goto end_autoredwall;
|
|
|
|
if (refextcf >= 0)
|
|
{
|
|
int32_t refz = SECTORFLD(refsect,z, refextcf), tmpsect;
|
|
int32_t neededzofs=0;
|
|
|
|
// the reference sector is extended on one side
|
|
// (given by refextcf) and non-sloped on the other
|
|
if (highlighted_sectors_components(0,0) != 1)
|
|
{
|
|
message("Highlighted sectors must be in one connected component");
|
|
goto end_autoredwall;
|
|
}
|
|
|
|
for (m=0; m<highlightsectorcnt; m++)
|
|
{
|
|
tmpsect = highlightsector[m];
|
|
yax_setbunch(tmpsect, refextcf, refsectbn[refextcf]);
|
|
// walls: not needed, since they're all inner to the bunch
|
|
|
|
SECTORFLD(tmpsect,z, refextcf) = refz;
|
|
setslope(tmpsect, refextcf, 0);
|
|
if (refextcf==0)
|
|
neededzofs = max(neededzofs, refz-sector[tmpsect].floorz);
|
|
else
|
|
neededzofs = max(neededzofs, sector[tmpsect].ceilingz-refz);
|
|
}
|
|
|
|
if (neededzofs > 0)
|
|
{
|
|
neededzofs += ksgn(neededzofs)*(512<<4);
|
|
neededzofs &= ~((256<<4)-1);
|
|
if (refextcf==1)
|
|
neededzofs *= -1;
|
|
for (m=0; m<highlightsectorcnt; m++)
|
|
SECTORFLD(highlightsector[m],z, !refextcf) += neededzofs;
|
|
}
|
|
}
|
|
#endif
|
|
wall[k-1].point2 = numwalls; // close the loop
|
|
newnumwalls = k;
|
|
n = (newnumwalls-numwalls); // number of walls in just constructed loop
|
|
|
|
if (clockdir(numwalls)==CLOCKDIR_CW)
|
|
{
|
|
int16_t begwalltomove = sector[refsect].wallptr+sector[refsect].wallnum;
|
|
int32_t onwwasvalid = onwisvalid();
|
|
|
|
flipwalls(numwalls, newnumwalls);
|
|
|
|
sector[refsect].wallnum += n;
|
|
if (refsect != numsectors-1)
|
|
{
|
|
twalltype *tmpwall = (twalltype *)Xmalloc(n * sizeof(walltype));
|
|
int16_t *tmponw = (int16_t *)Xmalloc(n * sizeof(int16_t));
|
|
|
|
for (m=0; m<numwalls; m++)
|
|
{
|
|
if (wall[m].nextwall >= begwalltomove)
|
|
wall[m].nextwall += n;
|
|
}
|
|
#ifdef YAX_ENABLE
|
|
yax_tweakwalls(begwalltomove, n);
|
|
#endif
|
|
for (m=refsect+1; m<numsectors; m++)
|
|
sector[m].wallptr += n;
|
|
for (m=begwalltomove; m<numwalls; m++)
|
|
wall[m].point2 += n;
|
|
for (m=numwalls; m<newnumwalls; m++)
|
|
wall[m].point2 += (begwalltomove-numwalls);
|
|
|
|
Bmemcpy(tmponw, &onextwall[numwalls], n*sizeof(int16_t));
|
|
Bmemmove(&onextwall[begwalltomove+n], &onextwall[begwalltomove],
|
|
(numwalls-begwalltomove)*sizeof(int16_t));
|
|
Bmemcpy(&onextwall[begwalltomove], tmponw, n*sizeof(int16_t));
|
|
|
|
Bmemcpy(tmpwall, &wall[numwalls], n*sizeof(walltype));
|
|
Bmemmove(&wall[begwalltomove+n], &wall[begwalltomove],
|
|
(numwalls-begwalltomove)*sizeof(walltype));
|
|
Bmemcpy(&wall[begwalltomove], tmpwall, n*sizeof(walltype));
|
|
|
|
Bfree(tmpwall);
|
|
Bfree(tmponw);
|
|
}
|
|
numwalls = newnumwalls;
|
|
newnumwalls = -1;
|
|
|
|
mkonwinvalid();
|
|
|
|
for (m=begwalltomove; m<begwalltomove+n; m++)
|
|
if (checksectorpointer(m, refsect) > 0)
|
|
if (onwwasvalid && onextwall[wall[m].nextwall]>=0)
|
|
{
|
|
//initprintf("%d %d\n", m, onextwall[wall[m].nextwall]);
|
|
copy_some_wall_members(m, onextwall[wall[m].nextwall], 0);
|
|
}
|
|
|
|
#ifndef YAX_ENABLE
|
|
message("Attached new inner loop to sector %d", refsect);
|
|
#else
|
|
{
|
|
const char *cfstr[2] = {"ceiling","floor"};
|
|
message("Attached new inner loop to %s%ssector %d",
|
|
refextcf>=0 ? cfstr[refextcf] : "",
|
|
refextcf>=0 ? "-extended " : "", refsect);
|
|
}
|
|
|
|
asksave = 1;
|
|
|
|
if (refextcf >= 0)
|
|
{
|
|
yax_update(0);
|
|
goto end_autoredwall;
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
end_autoredwall:
|
|
newnumwalls = -1;
|
|
#ifdef YAX_ENABLE
|
|
yax_updategrays(pos.z);
|
|
#endif
|
|
}
|
|
|
|
highlightx1 = searchx;
|
|
highlighty1 = searchy;
|
|
highlightx2 = searchx;
|
|
highlighty2 = searchy;
|
|
highlightsectorcnt = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (highlightsectorcnt == 0)
|
|
{
|
|
int32_t add=keystatus[0x28], sub=(!add && keystatus[0x27]), setop=(add||sub);
|
|
int32_t tx,ty, pointsel = eitherCTRL;
|
|
#ifdef YAX_ENABLE
|
|
// home: ceilings, end: floors
|
|
int32_t fb, bunchsel = keystatus[0xcf] ? 1 : (keystatus[0xc7] ? 0 : -1);
|
|
uint8_t bunchbitmap[YAX_MAXBUNCHES>>3];
|
|
Bmemset(bunchbitmap, 0, sizeof(bunchbitmap));
|
|
#endif
|
|
if (!m32_sideview)
|
|
{
|
|
getpoint(highlightx1,highlighty1, &highlightx1,&highlighty1);
|
|
getpoint(highlightx2,highlighty2, &highlightx2,&highlighty2);
|
|
}
|
|
|
|
if (!pointsel)
|
|
{
|
|
if (highlightx1 > highlightx2)
|
|
swaplong(&highlightx1, &highlightx2);
|
|
if (highlighty1 > highlighty2)
|
|
swaplong(&highlighty1, &highlighty2);
|
|
}
|
|
|
|
if (!setop)
|
|
Bmemset(hlsectorbitmap, 0, sizeof(hlsectorbitmap));
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
if (pointsel)
|
|
{
|
|
bad = (inside_editor(&pos, searchx,searchy, zoom, highlightx2, highlighty2, i)!=1);
|
|
}
|
|
else
|
|
{
|
|
bad = 0;
|
|
for (WALLS_OF_SECTOR(i, j))
|
|
{
|
|
if (!m32_sideview)
|
|
{
|
|
tx = wall[j].x;
|
|
ty = wall[j].y;
|
|
}
|
|
else
|
|
{
|
|
screencoords(&tx,&ty, wall[j].x-pos.x,wall[j].y-pos.y, zoom);
|
|
ty += getscreenvdisp(getflorzofslope(i, wall[j].x,wall[j].y)-pos.z, zoom);
|
|
tx += halfxdim16;
|
|
ty += midydim16;
|
|
}
|
|
|
|
if (tx < highlightx1 || tx > highlightx2) bad = 1;
|
|
if (ty < highlighty1 || ty > highlighty2) bad = 1;
|
|
if (bad == 1) break;
|
|
}
|
|
}
|
|
|
|
if (bad == 0)
|
|
{
|
|
#ifdef YAX_ENABLE
|
|
if (bunchsel!=-1 && (fb = yax_getbunch(i, YAX_FLOOR))>=0)
|
|
{
|
|
if ((sub || (graysectbitmap[i>>3]&(1<<(i&7)))==0) &&
|
|
(bunchbitmap[fb>>3]&(1<<(fb&7)))==0)
|
|
{
|
|
bunchbitmap[fb>>3] |= (1<<(fb&7));
|
|
for (SECTORS_OF_BUNCH(fb, bunchsel, j))
|
|
handlesecthighlight1(j, sub, 1);
|
|
}
|
|
}
|
|
else
|
|
#endif
|
|
handlesecthighlight1(i, sub, eitherSHIFT);
|
|
}
|
|
}
|
|
|
|
update_highlightsector();
|
|
ovh_whiteoutgrab(0);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (((bstatus&1) < (oldmousebstatus&1)) && highlightsectorcnt < 0) //after dragging
|
|
{
|
|
int32_t runi, numdelpoints=0;
|
|
int32_t havedrawnwalls = (newnumwalls!=-1), restorestat=1;
|
|
|
|
// restorestat is set to 2 whenever the drawn walls should NOT be
|
|
// restored afterwards
|
|
|
|
int32_t err = backup_drawn_walls(0);
|
|
|
|
if (err)
|
|
{
|
|
message("Error backing up drawn walls (code %d)!", err);
|
|
goto end_after_dragging;
|
|
}
|
|
|
|
j = 1;
|
|
for (i=0; i<highlightcnt; i++)
|
|
if (pointhighlight == highlight[i])
|
|
{
|
|
j = 0;
|
|
break;
|
|
}
|
|
|
|
if (j == 0)
|
|
{
|
|
for (i=0; i<highlightcnt; i++)
|
|
if ((highlight[i]&0xc000) == 16384)
|
|
updatesprite1(highlight[i]&16383);
|
|
}
|
|
else if ((pointhighlight&0xc000) == 16384)
|
|
{
|
|
updatesprite1(pointhighlight&16383);
|
|
}
|
|
|
|
if ((pointhighlight&0xc000) == 0)
|
|
{
|
|
dax = wall[pointhighlight].x;
|
|
day = wall[pointhighlight].y;
|
|
|
|
for (i=0; i<2; i++)
|
|
{
|
|
if (dragwall[i] < 0)
|
|
break;
|
|
|
|
if (keeptexturestretch && olen[i] != 0)
|
|
{
|
|
#ifndef YAX_ENABLE
|
|
j = dragwall[i];
|
|
#else
|
|
int32_t cf;
|
|
for (YAX_ITER_WALLS(dragwall[i], j, cf))
|
|
#endif
|
|
{
|
|
int32_t nw = wall[j].nextwall;
|
|
|
|
k = getlenbyrep(olen[i], wall[j].xrepeat);
|
|
fixxrepeat(j, k);
|
|
if (nw >= 0)
|
|
{
|
|
k = getlenbyrep(olen[i], wall[nw].xrepeat);
|
|
fixxrepeat(nw, k);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if ((pointhighlight&0xc000) == 16384)
|
|
{
|
|
dax = sprite[pointhighlight&16383].x;
|
|
day = sprite[pointhighlight&16383].y;
|
|
}
|
|
|
|
dragwall[0] = dragwall[1] = -1;
|
|
|
|
// attempt to delete some points
|
|
for (runi=0; runi<3; runi++) // check, tweak, carry out
|
|
for (i=numwalls-1; i>=0; i--)
|
|
{
|
|
if (runi==0)
|
|
wall[i].cstat &= ~(1<<14);
|
|
|
|
if (wall[i].x == POINT2(i).x && wall[i].y == POINT2(i).y)
|
|
{
|
|
if (havedrawnwalls)
|
|
{
|
|
if (i==ovh.suckwall || (ovh.split && i==ovh.splitstartwall))
|
|
{
|
|
// if we're about to delete a wall that participates
|
|
// in splitting, discard the already drawn walls
|
|
restorestat = 2;
|
|
}
|
|
else if (runi == 1)
|
|
{
|
|
// correct drawn wall anchors
|
|
if (ovh.suckwall > i)
|
|
ovh.suckwall--;
|
|
if (ovh.split && ovh.splitstartwall > i)
|
|
ovh.splitstartwall--;
|
|
}
|
|
}
|
|
|
|
if (runi == 0)
|
|
{
|
|
int32_t sectnum = sectorofwall(i);
|
|
if (sector[sectnum].wallnum <= 3)
|
|
{
|
|
message("Deleting wall %d would leave sector %d with %d walls.",
|
|
i, sectnum, sector[sectnum].wallnum-1);
|
|
goto end_after_dragging;
|
|
}
|
|
|
|
sectnum = wall[i].nextsector;
|
|
if (sectnum >= 0 && sector[sectnum].wallnum <= 3)
|
|
{
|
|
message("Deleting wall %d would leave sector %d with %d walls.",
|
|
i, sectnum, sector[sectnum].wallnum-1);
|
|
goto end_after_dragging;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
deletepoint(i, runi);
|
|
if (runi==2)
|
|
numdelpoints++;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (numdelpoints)
|
|
{
|
|
if (numdelpoints > 1)
|
|
message("Deleted %d points%s", numdelpoints,
|
|
(havedrawnwalls && restorestat==2) ? " and cleared drawn walls":"");
|
|
else
|
|
printmessage16("Point deleted%s", (havedrawnwalls && restorestat==2) ?
|
|
", cleared drawn walls":"");
|
|
asksave = 1;
|
|
}
|
|
else
|
|
{
|
|
for (i=0; i<numwalls; i++) //make new red lines?
|
|
{
|
|
YAX_SKIPWALL(i);
|
|
|
|
if ((wall[i].x == dax && wall[i].y == day)
|
|
|| (POINT2(i).x == dax && POINT2(i).y == day))
|
|
{
|
|
checksectorpointer(i, sectorofwall(i));
|
|
// fixrepeats(i);
|
|
asksave = 1;
|
|
}
|
|
}
|
|
}
|
|
#ifdef YAX_ENABLE
|
|
yax_update(0);
|
|
yax_updategrays(pos.z);
|
|
#endif
|
|
end_after_dragging:
|
|
backup_drawn_walls(restorestat);
|
|
}
|
|
|
|
if ((bstatus&1) > 0) //drag points
|
|
{
|
|
if (highlightsectorcnt > 0)
|
|
{
|
|
if ((bstatus&1) > (oldmousebstatus&1))
|
|
{
|
|
newnumwalls = -1;
|
|
sectorhighlightstat = -1;
|
|
|
|
// updatesector(mousxplc,mousyplc,&cursectorhighlight);
|
|
cursectorhighlight = -1;
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
if (inside_editor_curpos(highlightsector[i])==1)
|
|
{
|
|
cursectorhighlight = highlightsector[i];
|
|
break;
|
|
}
|
|
|
|
if (cursectorhighlight >= 0 && cursectorhighlight < numsectors)
|
|
{
|
|
//You clicked inside one of the flashing sectors!
|
|
sectorhighlightstat = 1;
|
|
|
|
dax = mousxplc;
|
|
day = mousyplc;
|
|
if (gridlock && grid > 0)
|
|
locktogrid(&dax, &day);
|
|
|
|
sectorhighlightx = dax;
|
|
sectorhighlighty = day;
|
|
}
|
|
}
|
|
else if (sectorhighlightstat == 1)
|
|
{
|
|
dax = mousxplc;
|
|
day = mousyplc;
|
|
if (gridlock && grid > 0)
|
|
locktogrid(&dax, &day);
|
|
|
|
dax -= sectorhighlightx;
|
|
day -= sectorhighlighty;
|
|
sectorhighlightx += dax;
|
|
sectorhighlighty += day;
|
|
#ifdef YAX_ENABLE
|
|
if (!hl_all_bunch_sectors_p())
|
|
printmessage16("To drag extended sectors, all sectors of a bunch must be selected");
|
|
else
|
|
#endif
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
for (WALLS_OF_SECTOR(highlightsector[i], j))
|
|
{ wall[j].x += dax; wall[j].y += day; }
|
|
|
|
for (j=headspritesect[highlightsector[i]]; j>=0; j=nextspritesect[j])
|
|
{ sprite[j].x += dax; sprite[j].y += day; }
|
|
}
|
|
|
|
//for(i=0;i<highlightsectorcnt;i++)
|
|
//{
|
|
// startwall = sector[highlightsector[i]].wallptr;
|
|
// endwall = startwall+sector[highlightsector[i]].wallnum-1;
|
|
// for(j=startwall;j<=endwall;j++)
|
|
// {
|
|
// if (wall[j].nextwall >= 0)
|
|
// checksectorpointer(wall[j].nextwall,wall[j].nextsector);
|
|
// checksectorpointer((short)j,highlightsector[i]);
|
|
// }
|
|
//}
|
|
asksave = 1;
|
|
}
|
|
|
|
}
|
|
else //if (highlightsectorcnt <= 0)
|
|
{
|
|
if ((bstatus&1) > (oldmousebstatus&1))
|
|
{
|
|
pointhighlight = getpointhighlight(mousxplc, mousyplc, pointhighlight);
|
|
|
|
if (pointhighlight >= 0 && (pointhighlight&0xc000)==0)
|
|
{
|
|
dragwall[0] = lastwall(pointhighlight);
|
|
dragwall[1] = pointhighlight;
|
|
olen[0] = wallength(dragwall[0]);
|
|
olen[1] = wallength(dragwall[1]);
|
|
}
|
|
}
|
|
|
|
if (pointhighlight >= 0 && (!m32_sideview || m32_sideelev>=32))
|
|
{
|
|
if (m32_sideview)
|
|
{
|
|
int32_t dz;
|
|
if (pointhighlight>=16384)
|
|
dz = sprite[pointhighlight&16383].z - pos.z;
|
|
else
|
|
dz = getflorzofslope(sectorofwall(pointhighlight),
|
|
wall[pointhighlight].x, wall[pointhighlight].y) - pos.z;
|
|
getinvdisplacement(&dax,&day, -dz);
|
|
dax += mousxplc;
|
|
day += mousyplc;
|
|
}
|
|
else
|
|
{
|
|
dax = mousxplc;
|
|
day = mousyplc;
|
|
}
|
|
|
|
if (gridlock && grid > 0)
|
|
locktogrid(&dax, &day);
|
|
|
|
j = 1;
|
|
if (highlightcnt > 0)
|
|
for (i=0; i<highlightcnt; i++)
|
|
if (pointhighlight == highlight[i])
|
|
{ j = 0; break; }
|
|
|
|
if (j == 0)
|
|
{
|
|
if ((pointhighlight&0xc000) == 0)
|
|
{
|
|
dax -= wall[pointhighlight].x;
|
|
day -= wall[pointhighlight].y;
|
|
}
|
|
else
|
|
{
|
|
dax -= sprite[pointhighlight&16383].x;
|
|
day -= sprite[pointhighlight&16383].y;
|
|
}
|
|
|
|
for (i=0; i<highlightcnt; i++)
|
|
{
|
|
if ((highlight[i]&0xc000) == 0)
|
|
{
|
|
wall[highlight[i]].x += dax;
|
|
wall[highlight[i]].y += day;
|
|
}
|
|
else
|
|
{
|
|
spritetype *daspr = &sprite[highlight[i]&16383];
|
|
|
|
daspr->x += dax;
|
|
daspr->y += day;
|
|
setspritez(daspr-sprite, (const vec3_t *)daspr);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ((pointhighlight&0xc000) == 0)
|
|
{
|
|
if (newnumwalls >= numwalls &&
|
|
wall[pointhighlight].x==firstx && wall[pointhighlight].y==firsty)
|
|
{
|
|
printmessage16("Can't drag point where drawing started.");
|
|
goto end_point_dragging;
|
|
}
|
|
|
|
dragpoint(pointhighlight,dax,day,2);
|
|
wall[lastwall(pointhighlight)].cstat |= (1<<14);
|
|
}
|
|
else if ((pointhighlight&0xc000) == 16384)
|
|
{
|
|
int32_t daspr=pointhighlight&16383;
|
|
int16_t osec=sprite[daspr].sectnum, nsec=osec;
|
|
vec3_t vec, ovec;
|
|
|
|
Bmemcpy(&ovec, (vec3_t *)&sprite[daspr], sizeof(vec3_t));
|
|
vec.x = dax;
|
|
vec.y = day;
|
|
vec.z = sprite[daspr].z;
|
|
if (setspritez(daspr, &vec) == -1 && osec>=0)
|
|
{
|
|
updatesectorbreadth(dax, day, &nsec);
|
|
|
|
if (nsec >= 0)
|
|
{
|
|
sprite[daspr].x = dax;
|
|
sprite[daspr].y = day;
|
|
// z updating is after we released the mouse button
|
|
if (sprite[daspr].sectnum != nsec)
|
|
changespritesect(daspr, nsec);
|
|
}
|
|
else
|
|
Bmemcpy(&sprite[daspr], &ovec, sizeof(vec3_t));
|
|
}
|
|
}
|
|
}
|
|
|
|
asksave = 1;
|
|
}
|
|
}
|
|
}
|
|
else //if ((bstatus&1) == 0)
|
|
{
|
|
pointhighlight = getpointhighlight(mousxplc, mousyplc, pointhighlight);
|
|
sectorhighlightstat = -1;
|
|
}
|
|
end_point_dragging:
|
|
|
|
if (bstatus&(2|4)) // change arrow position
|
|
{
|
|
if (eitherCTRL)
|
|
{
|
|
int16_t cursectornum;
|
|
|
|
for (cursectornum=0; cursectornum<numsectors; cursectornum++)
|
|
if (inside_editor_curpos(cursectornum) == 1)
|
|
break;
|
|
|
|
if (cursectornum < numsectors)
|
|
{
|
|
if (pointhighlight >= 16384)
|
|
CallExtEditSpriteData(pointhighlight-16384);
|
|
else if ((linehighlight >= 0) && ((bstatus&1) || sectorofwall(linehighlight) == cursectornum))
|
|
CallExtEditWallData(linehighlight);
|
|
else if (cursectornum >= 0)
|
|
CallExtEditSectorData(cursectornum);
|
|
}
|
|
|
|
bstatus &= ~6;
|
|
}
|
|
else
|
|
{
|
|
if (m32_sideview && (bstatus&4))
|
|
{
|
|
pos.z += divscale18(searchy-midydim16,zoom);
|
|
getpoint(searchx,midydim16, &pos.x, &pos.y);
|
|
#ifdef YAX_ENABLE
|
|
yax_updategrays(pos.z);
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
pos.x = mousxplc;
|
|
pos.y = mousyplc;
|
|
}
|
|
|
|
if (m32_sideview)
|
|
{
|
|
int32_t opat=drawlinepat;
|
|
|
|
y1 = INT32_MAX;
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
if (inside(pos.x, pos.y, i)==1)
|
|
{
|
|
day = getscreenvdisp(getflorzofslope(i, pos.x, pos.y)-pos.z, zoom);
|
|
|
|
x2 = max(4, mulscale14(64,zoom));
|
|
y2 = scalescreeny(x2);
|
|
|
|
if (klabs(day) < y1)
|
|
y1 = day;
|
|
|
|
drawline16base(halfxdim16, midydim16+day, -x2,-y2, x2,y2, editorcolors[14]);
|
|
drawline16base(halfxdim16, midydim16+day, -x2,y2, x2,-y2, editorcolors[14]);
|
|
}
|
|
}
|
|
|
|
drawlinepat = 0x11111111;
|
|
if (y1 != INT32_MAX)
|
|
drawline16base(halfxdim16,midydim16, 0,0, 0,y1, editorcolors[14]);
|
|
// else
|
|
// drawline16base(halfxdim16,midydim16, 0,0, 0,getscreenvdisp(-pos.z, zoom), editorcolors[14]);
|
|
drawlinepat = opat;
|
|
}
|
|
|
|
searchx = halfxdim16;
|
|
searchy = midydim16;
|
|
}
|
|
}
|
|
else if ((oldmousebstatus&6) > 0)
|
|
updatesectorz(pos.x,pos.y,pos.z,&cursectnum);
|
|
|
|
if (circlewall != -1 && (keystatus[0x4a] || ((bstatus&32) && !eitherCTRL))) // -, mousewheel down
|
|
{
|
|
if (circlepoints > 1)
|
|
circlepoints--;
|
|
keystatus[0x4a] = 0;
|
|
mouseb &= ~32;
|
|
bstatus &= ~32;
|
|
}
|
|
if (circlewall != -1 && (keystatus[0x4e] || ((bstatus&16) && !eitherCTRL))) // +, mousewheel up
|
|
{
|
|
if (circlepoints < 63)
|
|
circlepoints++;
|
|
keystatus[0x4e] = 0;
|
|
mouseb &= ~16;
|
|
bstatus &= ~16;
|
|
}
|
|
|
|
if (keystatus[0x3d]) // F3
|
|
{
|
|
keystatus[0x3d]=0;
|
|
if (!m32_sideview && EDITING_MAP_P())
|
|
message("Must not be editing map while switching to side view mode.");
|
|
else
|
|
{
|
|
m32_sideview = !m32_sideview;
|
|
printmessage16("Side view %s", m32_sideview?"enabled":"disabled");
|
|
}
|
|
}
|
|
|
|
if (m32_sideview && (keystatus[0x10] || keystatus[0x11]))
|
|
{
|
|
if (eitherCTRL)
|
|
{
|
|
if (m32_sideang&63)
|
|
{
|
|
m32_sideang += (1-2*keystatus[0x10])*(1-2*sideview_reversehrot)*32;
|
|
m32_sideang &= (2047&~63);
|
|
}
|
|
else
|
|
{
|
|
m32_sideang += (1-2*keystatus[0x10])*(1-2*sideview_reversehrot)*64;
|
|
m32_sideang &= 2047;
|
|
}
|
|
|
|
keystatus[0x10] = keystatus[0x11] = 0;
|
|
}
|
|
else
|
|
{
|
|
m32_sideang += (1-2*keystatus[0x10])*(1-2*sideview_reversehrot)*synctics<<(eitherSHIFT*2);
|
|
m32_sideang &= 2047;
|
|
}
|
|
_printmessage16("Sideview angle: %d", (int32_t)m32_sideang);
|
|
}
|
|
|
|
if (m32_sideview && (eitherSHIFT || (bstatus&(16|32))))
|
|
{
|
|
if ((DOWN_BK(MOVEUP) || (bstatus&16)) && m32_sideelev < 512)
|
|
{
|
|
if (DOWN_BK(MOVEUP))
|
|
m32_sideelev += synctics<<1;
|
|
if (bstatus&16)
|
|
m32_sideelev += 4<<1;
|
|
|
|
if (m32_sideelev > 512)
|
|
m32_sideelev = 512;
|
|
_printmessage16("Sideview elevation: %d", m32_sideelev);
|
|
}
|
|
if ((DOWN_BK(MOVEDOWN) || (bstatus&32)) && m32_sideelev > 0)
|
|
{
|
|
if (DOWN_BK(MOVEDOWN))
|
|
m32_sideelev -= synctics<<1;
|
|
if (bstatus&32)
|
|
m32_sideelev -= 4<<1;
|
|
|
|
if (m32_sideelev < 0)
|
|
m32_sideelev = 0;
|
|
_printmessage16("Sideview elevation: %d", m32_sideelev);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int32_t didzoom=0;
|
|
|
|
if ((DOWN_BK(MOVEUP) || (bstatus&16)) && zoom < 65536)
|
|
{
|
|
if (DOWN_BK(MOVEUP))
|
|
zoom += synctics*(zoom>>4);
|
|
if (bstatus&16)
|
|
zoom += 4*(zoom>>4);
|
|
|
|
if (zoom < 24) zoom += 2;
|
|
didzoom = 1;
|
|
}
|
|
if ((DOWN_BK(MOVEDOWN) || (bstatus&32)) && zoom > 8)
|
|
{
|
|
if (DOWN_BK(MOVEDOWN))
|
|
zoom -= synctics*(zoom>>4);
|
|
if (bstatus&32)
|
|
zoom -= 4*(zoom>>4);
|
|
|
|
didzoom = 1;
|
|
}
|
|
|
|
if (didzoom)
|
|
{
|
|
if (eitherALT)
|
|
{
|
|
searchx = halfxdim16;
|
|
searchy = midydim16;
|
|
pos.x = mousxplc;
|
|
pos.y = mousyplc;
|
|
}
|
|
zoom = clamp(zoom, 8, 65536);
|
|
_printmessage16("Zoom: %d",zoom);
|
|
}
|
|
}
|
|
|
|
if (keystatus[0x22]) // G (grid on/off)
|
|
{
|
|
keystatus[0x22] = 0;
|
|
grid++;
|
|
if (grid == 7) grid = 0;
|
|
}
|
|
if (keystatus[0x26]) // L (grid lock)
|
|
{
|
|
keystatus[0x26] = 0;
|
|
gridlock = !gridlock;
|
|
printmessage16("Grid locking %s", gridlock?"on":"off");
|
|
}
|
|
|
|
if (keystatus[0x24]) // J (join sectors)
|
|
{
|
|
keystatus[0x24] = 0;
|
|
|
|
if (newnumwalls >= 0)
|
|
{
|
|
printmessage16("Can't join sectors while editing.");
|
|
goto end_join_sectors;
|
|
}
|
|
|
|
#ifdef YAX_ENABLE
|
|
if (highlightsectorcnt > 0 && eitherCTRL)
|
|
{
|
|
// [component][ceiling(0) or floor(1)]
|
|
// compstat: &1: "has extension", &2: "differ in z", &4: "sloped", -1: "uninited"
|
|
int32_t cf, comp, compstat[2][2]={{-1,-1},{-1,-1}}, compcfz[2][2];
|
|
|
|
// joinstat: join what to what?
|
|
// &1: ceil(comp 0) <-> flor(comp 1), &2: flor(comp 0) <-> ceil(comp 1)
|
|
// (doesn't yet say which is stationary)
|
|
// movestat: which component can be displaced?
|
|
// &1: first, &2: second
|
|
int32_t askres, joinstat, needsdisp, moveonwp;
|
|
int32_t movestat, dx=0,dy=0,dz, delayerr=0;
|
|
|
|
int32_t numouterwalls[2] = {0,0}, numowals;
|
|
static int16_t outerwall[2][MAXWALLS];
|
|
const twalltype *wal0, *wal1, *wal0p2, *wal1p2;
|
|
|
|
// join sector ceilings/floors to a new bunch
|
|
if (numyaxbunches==YAX_MAXBUNCHES)
|
|
{
|
|
message("Bunch limit of %d reached, cannot join", YAX_MAXBUNCHES);
|
|
goto end_join_sectors;
|
|
}
|
|
|
|
// first, see whether we have exactly two connected components
|
|
// wrt wall[].nextsector
|
|
if (highlighted_sectors_components(0,0) != 2)
|
|
{
|
|
message("Sectors must be partitioned in two components to join");
|
|
goto end_join_sectors;
|
|
}
|
|
|
|
for (k=0; k<highlightsectorcnt; k++)
|
|
{
|
|
j = highlightsector[k];
|
|
comp = !!(collsectbitmap[1][j>>3]&(1<<(j&7)));
|
|
|
|
for (cf=0; cf<2; cf++)
|
|
{
|
|
if (compstat[comp][cf]==-1)
|
|
{
|
|
compstat[comp][cf] = 0;
|
|
compcfz[comp][cf] = SECTORFLD(j,z, cf);
|
|
}
|
|
|
|
if (yax_getbunch(j, cf)>=0)
|
|
compstat[comp][cf] |= 1;
|
|
if (SECTORFLD(j,z, cf) != compcfz[comp][cf])
|
|
compstat[comp][cf] |= 2;
|
|
if (SECTORFLD(j,stat, cf)&2)
|
|
compstat[comp][cf] |= 4;
|
|
|
|
compcfz[comp][cf] = SECTORFLD(j,z, cf);
|
|
}
|
|
}
|
|
|
|
// check for consistency
|
|
joinstat = 0;
|
|
if (!compstat[0][YAX_CEILING] && !compstat[1][YAX_FLOOR])
|
|
joinstat |= 1;
|
|
if (!compstat[0][YAX_FLOOR] && !compstat[1][YAX_CEILING])
|
|
joinstat |= 2;
|
|
|
|
if (joinstat==0)
|
|
{
|
|
message("No consistent joining combination found");
|
|
OSD_Printf("comp0: c=%d,f=%d; comp1: c=%d,f=%d (1:extended, 2:z mismatch, 4:sloped)\n",
|
|
compstat[0][YAX_CEILING], compstat[0][YAX_FLOOR],
|
|
compstat[1][YAX_CEILING], compstat[1][YAX_FLOOR]);
|
|
//for (i=0; i<2; i++) for (j=0; j<2; j++) message("%d", compstat[i][j]);
|
|
goto end_join_sectors;
|
|
}
|
|
if (joinstat==3)
|
|
{
|
|
if (compcfz[0][YAX_CEILING] != compstat[1][YAX_FLOOR])
|
|
joinstat &= 1;
|
|
if (compcfz[0][YAX_CEILING] != compstat[1][YAX_FLOOR])
|
|
joinstat &= 2;
|
|
|
|
if (joinstat == 0)
|
|
joinstat = 3; // we couldn't disambiguate
|
|
}
|
|
|
|
for (comp=0; comp<2; comp++)
|
|
for (k=0; k<collnumsects[comp]; k++)
|
|
{
|
|
i = collsectlist[comp][k];
|
|
for (WALLS_OF_SECTOR(i, j))
|
|
if (wall[j].nextwall < 0)
|
|
outerwall[comp][numouterwalls[comp]++] = j;
|
|
}
|
|
|
|
if (numouterwalls[0] != numouterwalls[1])
|
|
{
|
|
message("Number of outer walls must be equal for both components"
|
|
" (have %d and %d)", numouterwalls[0], numouterwalls[1]);
|
|
if (numouterwalls[0]>0 && numouterwalls[1]>0)
|
|
delayerr = 1;
|
|
else
|
|
goto end_join_sectors;
|
|
}
|
|
numowals = min(numouterwalls[0], numouterwalls[1]);
|
|
|
|
// now sort outer walls 'geometrically'
|
|
for (comp=0; comp<2; comp++)
|
|
sort_walls_geometrically(outerwall[comp], numouterwalls[comp]);
|
|
|
|
for (k=0; k<numowals; k++)
|
|
{
|
|
wal0 = (twalltype *)&wall[outerwall[0][k]];
|
|
wal1 = (twalltype *)&wall[outerwall[1][k]];
|
|
|
|
wal0p2 = (twalltype *)&wall[wal0->point2];
|
|
wal1p2 = (twalltype *)&wall[wal1->point2];
|
|
|
|
if (k==0)
|
|
{
|
|
dx = wal1->x - wal0->x;
|
|
dy = wal1->y - wal0->y;
|
|
}
|
|
|
|
if (wal1->x - wal0->x != dx || wal1->y - wal0->y != dy ||
|
|
wal1p2->x - wal0p2->x != dx || wal1p2->y - wal0p2->y != dy)
|
|
{
|
|
pos.x = wal0->x + (wal0p2->x - wal0->x)/4;
|
|
pos.y = wal0->y + (wal0p2->y - wal0->y)/4;
|
|
pos.z = getflorzofslope(sectorofwall(wal0-(twalltype *)wall), pos.x, pos.y);
|
|
|
|
if (!delayerr)
|
|
message("Outer wall coordinates must coincide for both components");
|
|
OSD_Printf("wal0:%d (%d,%d)--(%d,%d)\n",(int)(wal0-(twalltype *)wall),
|
|
wal0->x,wal0->y, wal0p2->x,wal0p2->y);
|
|
OSD_Printf("wal1:%d (%d,%d)--(%d,%d)\n",(int)(wal1-(twalltype *)wall),
|
|
wal1->x,wal1->y, wal1p2->x,wal1p2->y);
|
|
|
|
goto end_join_sectors;
|
|
}
|
|
}
|
|
|
|
if (delayerr)
|
|
goto end_join_sectors;
|
|
|
|
if (joinstat == 3)
|
|
{
|
|
char askchars[2] = {'1', 'v'};
|
|
|
|
// now is a good time to ask...
|
|
for (comp=0; comp<2; comp++)
|
|
for (k=0; k<collnumsects[comp]; k++)
|
|
fillsector(collsectlist[comp][k], comp==0 ? 159 : editorcolors[11]);
|
|
|
|
fade_editor_screen(editorcolors[11] | (159<<8));
|
|
askres = editor_ask_function("Connect yellow ceil w/ blue floor (1) or (v)ice versa?", askchars, 2);
|
|
if (askres==-1)
|
|
goto end_join_sectors;
|
|
joinstat &= (1<<askres);
|
|
}
|
|
|
|
joinstat--; // 0:ceil(0)<->flor(1), 1:ceil(1)<->flor(0)
|
|
|
|
dz = compcfz[1][!joinstat] - compcfz[0][joinstat];
|
|
needsdisp = (dx || dy || dz);
|
|
|
|
if (needsdisp)
|
|
{
|
|
// a component is more likely to be displaced if it's not
|
|
// extended on the non-joining side
|
|
movestat = (!(compstat[0][!joinstat]&1)) | ((!(compstat[1][joinstat]&1))<<1);
|
|
if (!movestat)
|
|
{
|
|
movestat = 3;
|
|
// message("Internal error while TROR-joining: movestat inconsistent!");
|
|
// goto end_join_sectors;
|
|
}
|
|
|
|
if (movestat==3)
|
|
{
|
|
char askchars[2] = {'y', 'b'};
|
|
|
|
for (comp=0; comp<2; comp++)
|
|
for (k=0; k<collnumsects[comp]; k++)
|
|
fillsector(collsectlist[comp][k], comp==0 ? 159 : editorcolors[11]);
|
|
|
|
fade_editor_screen(editorcolors[11] | (159<<8));
|
|
askres = editor_ask_function("Move (y)ellow or (b)lue component?", askchars, 2);
|
|
if (askres==-1)
|
|
goto end_join_sectors;
|
|
movestat &= (1<<askres);
|
|
}
|
|
|
|
movestat--; // 0:move 1st, 1:move 2nd component
|
|
if (movestat==1)
|
|
dx*=-1, dy*=-1, dz*=-1;
|
|
|
|
moveonwp = 0;
|
|
if (onwisvalid())
|
|
{
|
|
static int16_t ocollsectlist[MAXSECTORS];
|
|
static uint8_t tcollbitmap[MAXSECTORS>>3];
|
|
int16_t ocollnumsects=collnumsects[movestat], tmpsect;
|
|
|
|
Bmemcpy(ocollsectlist, collsectlist[movestat], ocollnumsects*sizeof(int16_t));
|
|
Bmemset(tcollbitmap, 0, sizeof(tcollbitmap));
|
|
|
|
for (k=0; k<ocollnumsects; k++)
|
|
for (WALLS_OF_SECTOR(ocollsectlist[k], j))
|
|
{
|
|
if (onextwall[j] < 0)
|
|
continue;
|
|
|
|
tmpsect = sectorofwall(onextwall[j]);
|
|
sectors_components(1, &tmpsect, 1,0);
|
|
|
|
for (m=0; m<(numsectors+7)>>3; m++)
|
|
tcollbitmap[m] |= collsectbitmap[0][m];
|
|
moveonwp = 1;
|
|
}
|
|
|
|
if (moveonwp)
|
|
{
|
|
int32_t movecol = movestat==0 ? 159 : editorcolors[11];
|
|
for (i=0; i<numsectors; i++)
|
|
if (tcollbitmap[i>>3]&(1<<(i&7)))
|
|
fillsector(i, editorcolors[12]);
|
|
|
|
fade_editor_screen(editorcolors[12] | (movecol<<8));
|
|
moveonwp = ask_if_sure("Also move formerly wall-connected sectors?",0);
|
|
if (moveonwp==-1)
|
|
goto end_join_sectors;
|
|
}
|
|
}
|
|
|
|
// now need to collect them wrt. the nextsector but also
|
|
// the yax-nextsector relation
|
|
if (highlighted_sectors_components(1,moveonwp) != 2)
|
|
{
|
|
message("Must not have TROR connections between the two components");
|
|
goto end_join_sectors;
|
|
}
|
|
|
|
// displace!
|
|
for (k=0; k<collnumsects[movestat]; k++)
|
|
{
|
|
i = collsectlist[movestat][k];
|
|
|
|
sector[i].floorz += dz;
|
|
sector[i].ceilingz += dz;
|
|
|
|
for (WALLS_OF_SECTOR(i, j))
|
|
{
|
|
wall[j].x += dx;
|
|
wall[j].y += dy;
|
|
}
|
|
|
|
for (j=headspritesect[i]; j>=0; j=nextspritesect[j])
|
|
{
|
|
sprite[j].x += dx;
|
|
sprite[j].y += dy;
|
|
sprite[j].z += dz;
|
|
}
|
|
}
|
|
|
|
// restore old components, i.e. only the bunch sectors
|
|
highlighted_sectors_components(0,0);
|
|
|
|
} // end if (needsdisp)
|
|
|
|
/*** construct the YAX connection! ***/
|
|
for (comp=0; comp<2; comp++)
|
|
{
|
|
// walls
|
|
for (j=0; j<numowals; j++)
|
|
yax_setnextwall(outerwall[comp][j], comp^joinstat, outerwall[!comp][j]);
|
|
|
|
// sectors
|
|
for (k=0; k<collnumsects[comp]; k++)
|
|
{
|
|
i = collsectlist[comp][k];
|
|
yax_setbunch(i, comp^joinstat, numyaxbunches);
|
|
SECTORFLD(i,stat, comp^joinstat) &= ~1; // no plax
|
|
|
|
// restore red walls AFTER setting nextwalls
|
|
// (see checksectorpointer() for why)
|
|
for (WALLS_OF_SECTOR(i, j))
|
|
if (wall[j].nextwall < 0)
|
|
checksectorpointer(j, i);
|
|
}
|
|
}
|
|
|
|
reset_highlightsector();
|
|
|
|
yax_update(0);
|
|
yax_updategrays(pos.z);
|
|
|
|
message("Joined highlighted sectors to new bunch %d", numyaxbunches);
|
|
asksave = 1;
|
|
}
|
|
else
|
|
#endif // defined YAX_ENABLE
|
|
if (joinsector[0] < 0)
|
|
{
|
|
int32_t numjoincandidates = 0;
|
|
char *origframe=NULL;
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
numjoincandidates += (inside_editor_curpos(i) == 1);
|
|
}
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
if (inside_editor_curpos(i) == 1)
|
|
{
|
|
if (numjoincandidates > 1)
|
|
{
|
|
if (!bakframe_fillandfade(&origframe, i, "Use this as first joining sector? (Y/N)"))
|
|
continue;
|
|
}
|
|
|
|
joinsector[0] = i;
|
|
printmessage16("Join sector - press J again on sector to join with.");
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (origframe)
|
|
Bfree(origframe);
|
|
}
|
|
else
|
|
{
|
|
char *origframe = NULL;
|
|
int32_t numjoincandidates = 0;
|
|
|
|
joinsector[1] = -1;
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
numjoincandidates += (inside_editor_curpos(i) == 1);
|
|
}
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
|
|
if (inside_editor_curpos(i) == 1)
|
|
{
|
|
if (numjoincandidates > 1)
|
|
{
|
|
if (!bakframe_fillandfade(&origframe, i, "Use this as second joining sector? (Y/N)"))
|
|
continue;
|
|
|
|
if (origframe)
|
|
{
|
|
Bfree(origframe);
|
|
origframe = NULL;
|
|
}
|
|
}
|
|
|
|
joinsector[1] = i;
|
|
|
|
const int s1to0wall = find_nextwall(i, joinsector[0]);
|
|
const int s0to1wall = s1to0wall == -1 ? -1 : wall[s1to0wall].nextwall;
|
|
#ifdef YAX_ENABLE
|
|
int16_t jbn[2][2]; // [join index][c/f]
|
|
|
|
for (k=0; k<2; k++)
|
|
yax_getbunches(joinsector[k], &jbn[k][YAX_CEILING], &jbn[k][YAX_FLOOR]);
|
|
#endif
|
|
// pressing J into the same sector is the same as saying 'no'
|
|
// v----------------v
|
|
if (s1to0wall == -1 && i != joinsector[0])
|
|
{
|
|
int32_t good = 1;
|
|
#ifdef YAX_ENABLE
|
|
if (jbn[0][0]>=0 || jbn[0][1]>=0 || jbn[1][0]>=0 || jbn[1][1]>=0)
|
|
{
|
|
message("Joining non-adjacent extended sectors not allowed!");
|
|
good = 0;
|
|
}
|
|
#endif
|
|
if (!m32_script_expertmode)
|
|
{
|
|
message("Joining non-adjacent disabled in non-expert mode");
|
|
good = 0;
|
|
}
|
|
|
|
if (!good)
|
|
{
|
|
joinsector[0] = joinsector[1] = -1;
|
|
goto end_join_sectors;
|
|
}
|
|
|
|
{
|
|
fillsector(i, editorcolors[9]);
|
|
fillsector(joinsector[0], editorcolors[9]);
|
|
fade_editor_screen(editorcolors[9]);
|
|
|
|
if (!ask_if_sure("Really join non-adjacent sectors? (Y/N)", 0))
|
|
joinsector[1] = joinsector[0];
|
|
}
|
|
}
|
|
#ifdef YAX_ENABLE
|
|
// unequal bunchnums (bitmap): 1:above, 2:below
|
|
int uneqbn =
|
|
(jbn[0][YAX_CEILING] != jbn[1][YAX_CEILING]) |
|
|
((jbn[0][YAX_FLOOR] != jbn[1][YAX_FLOOR])<<1);
|
|
|
|
if (uneqbn)
|
|
{
|
|
const int32_t cf=YAX_FLOOR;
|
|
int32_t whybad=0;
|
|
|
|
if (uneqbn == 1)
|
|
{
|
|
OSD_Printf("Can't join two sectors with different ceiling bunchnums."
|
|
" To make them equal, join their upper neighbor's floors.\n");
|
|
printmessage16("Can't join two sectors with different ceiling bunchnums. See OSD");
|
|
joinsector[0] = joinsector[1] = -1;
|
|
goto end_join_sectors;
|
|
}
|
|
if (s0to1wall < 0)
|
|
{
|
|
printmessage16("INTERNAL ERROR: nextwalls inconsistent!");
|
|
joinsector[0] = joinsector[1] = -1;
|
|
goto end_join_sectors;
|
|
}
|
|
|
|
// both must be extended
|
|
if (jbn[0][cf]<0 || jbn[1][cf]<0)
|
|
uneqbn &= ~(1<<cf), whybad|=1;
|
|
// if any sloped, can't join
|
|
if ((SECTORFLD(joinsector[0],stat, cf)&2) || (SECTORFLD(joinsector[1],stat, cf)&2))
|
|
uneqbn &= ~(1<<cf), whybad|=2;
|
|
// if on unequal heights, can't join either
|
|
if (SECTORFLD(joinsector[0],z, cf) != SECTORFLD(joinsector[1],z, cf))
|
|
uneqbn &= ~(1<<cf), whybad|=4;
|
|
|
|
// check whether the lower neighbors have a red-wall link to each other
|
|
const int jsynw[2] = {
|
|
yax_getnextwall(s0to1wall, cf),
|
|
yax_getnextwall(s1to0wall, cf)
|
|
};
|
|
|
|
if (jsynw[0]<0 || jsynw[1]<0) // this shouldn't happen
|
|
{
|
|
uneqbn &= ~(1<<cf), whybad|=8;
|
|
}
|
|
else if (wall[jsynw[1]].nextwall != jsynw[0])
|
|
{
|
|
// if (find_nextwall(sectorofwall(jsynw[1]), sectorofwall(jsynw[0])) < 0)
|
|
uneqbn &= ~(1<<cf), whybad|=16;
|
|
}
|
|
|
|
if ((uneqbn&2)==0)
|
|
{
|
|
#if 0
|
|
if (whybad==1+8 && jbn[0][cf]>=0 && jbn[1][cf]<0)
|
|
{
|
|
// 1st join sector extended, 2nd not... let's see
|
|
// if the latter is inner to the former one
|
|
|
|
int32_t lowerstartsec = yax_vnextsec(s0to1wall, cf);
|
|
|
|
m = (lowerstartsec < 0)<<1;
|
|
for (WALLS_OF_SECTOR(joinsector[1], k))
|
|
{
|
|
if (m) break;
|
|
|
|
m |= (wall[k].nextsector>=0 && wall[k].nextsector != joinsector[0]);
|
|
m |= (wall[k].nextwall>=0 && yax_vnextsec(wall[k].nextwall, cf)!=lowerstartsec)<<1;
|
|
}
|
|
|
|
if (m==0)
|
|
{
|
|
yax_setbunch(joinsector[1], YAX_FLOOR, jbn[0][cf]);
|
|
yax_update(0);
|
|
yax_updategrays(pos.z);
|
|
asksave = 1;
|
|
|
|
printmessage16("Added sector %d's floor to bunch %d",
|
|
joinsector[1], jbn[0][cf]);
|
|
}
|
|
else
|
|
{
|
|
if (m&1)
|
|
{
|
|
message("Can't add sector %d's floor to bunch %d: not inner to sector %d",
|
|
joinsector[1], jbn[0][cf], joinsector[0]);
|
|
}
|
|
else // if (m&2)
|
|
{
|
|
message("Can't add sector %d's floor to bunch %d: must have lower neighbor",
|
|
joinsector[1], jbn[0][cf]);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
if (whybad&1)
|
|
message("Can't make floor bunchnums equal: both floors must be extended");
|
|
else if (whybad&2)
|
|
message("Can't make floor bunchnums equal: both floors must be non-sloped");
|
|
else if (whybad&4)
|
|
message("Can't make floor bunchnums equal: both floors must have equal height");
|
|
else if (whybad&8)
|
|
message("Can't make floor bunchnums equal: INTERNAL ERROR");
|
|
else if (whybad&16)
|
|
message("Can't make floor bunchnums equal: lower neighbors must be linked");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int32_t vcf, newbn, ynw;
|
|
|
|
// we're good to go for making floor bunchnums equal
|
|
for (SECTORS_OF_BUNCH(jbn[1][cf], YAX_FLOOR, k))
|
|
yax_setbunch(k, YAX_FLOOR, jbn[0][cf]);
|
|
for (SECTORS_OF_BUNCH(jbn[1][cf], YAX_CEILING, k))
|
|
yax_setbunch(k, YAX_CEILING, jbn[0][cf]);
|
|
|
|
yax_update(0);
|
|
// now we can iterate the sectors with the new bunchnums
|
|
newbn = yax_getbunch(joinsector[0], cf);
|
|
|
|
// clear all yax-nextwall links on walls that are inside the bunch
|
|
for (vcf=0; vcf<2; vcf++)
|
|
for (SECTORS_OF_BUNCH(newbn, vcf, k))
|
|
for (WALLS_OF_SECTOR(k, m))
|
|
{
|
|
ynw = yax_getnextwall(m, vcf);
|
|
if (ynw < 0 || wall[m].nextsector < 0)
|
|
continue;
|
|
|
|
if (yax_getbunch(wall[m].nextsector, vcf) == newbn)
|
|
{
|
|
yax_setnextwall(ynw, !vcf, -1);
|
|
yax_setnextwall(m, vcf, -1);
|
|
}
|
|
}
|
|
|
|
// shouldn't be needed again for the editor, but can't harm either:
|
|
yax_update(0);
|
|
yax_updategrays(pos.z);
|
|
|
|
printmessage16("Made sector %d and %d floor bunchnums equal",
|
|
joinsector[0], joinsector[1]);
|
|
asksave = 1;
|
|
}
|
|
|
|
joinsector[0] = joinsector[1] = -1;
|
|
goto end_join_sectors;
|
|
}
|
|
#endif
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (joinsector[1] < 0 || joinsector[0] == joinsector[1])
|
|
{
|
|
printmessage16("No sectors joined.");
|
|
joinsector[0] = -1;
|
|
goto end_join_sectors;
|
|
}
|
|
|
|
|
|
for (i=0; i<numwalls; i++)
|
|
wall[i].cstat &= ~(1<<14);
|
|
|
|
newnumwalls = numwalls;
|
|
|
|
for (k=0; k<2; k++)
|
|
{
|
|
for (WALLS_OF_SECTOR(joinsector[k], j))
|
|
{
|
|
int32_t loopnum=MAXWALLS*2;
|
|
|
|
if (wall[j].cstat & (1<<14))
|
|
continue;
|
|
|
|
if (wall[j].nextsector == joinsector[1-k])
|
|
{
|
|
wall[j].cstat |= (1<<14);
|
|
continue;
|
|
}
|
|
|
|
i = j;
|
|
m = newnumwalls;
|
|
int joink = k;
|
|
do
|
|
{
|
|
if (newnumwalls >= MAXWALLS + M32_FIXME_WALLS)
|
|
{
|
|
message("Joining sectors failed: not enough space beyond wall[]");
|
|
joinsector[0] = -1;
|
|
newnumwalls = -1;
|
|
|
|
for (i=0; i<numwalls; i++)
|
|
wall[i].cstat &= ~(1<<14);
|
|
|
|
goto end_join_sectors;
|
|
}
|
|
|
|
Bmemcpy(&wall[newnumwalls], &wall[i], sizeof(walltype));
|
|
#ifdef YAX_ENABLE
|
|
yax_fixreverselinks(newnumwalls, newnumwalls);
|
|
#endif
|
|
wall[newnumwalls].point2 = newnumwalls+1;
|
|
newnumwalls++;
|
|
|
|
wall[i].cstat |= (1<<14);
|
|
|
|
i = wall[i].point2;
|
|
if (wall[i].nextsector == joinsector[1-joink])
|
|
{
|
|
i = NEXTWALL(i).point2;
|
|
joink = 1 - joink;
|
|
}
|
|
|
|
loopnum--;
|
|
}
|
|
while (loopnum>0 && ((wall[i].cstat & (1<<14))==0)
|
|
&& (wall[i].nextsector != joinsector[1-joink]));
|
|
|
|
wall[newnumwalls-1].point2 = m;
|
|
|
|
if (loopnum==0)
|
|
{
|
|
message("internal error while joining sectors: infloop!");
|
|
newnumwalls = -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (newnumwalls > numwalls)
|
|
{
|
|
Bmemcpy(§or[numsectors], §or[joinsector[0]], sizeof(sectortype));
|
|
sector[numsectors].wallptr = numwalls;
|
|
sector[numsectors].wallnum = newnumwalls-numwalls;
|
|
|
|
//fix sprites
|
|
for (i=0; i<2; i++)
|
|
{
|
|
j = headspritesect[joinsector[i]];
|
|
while (j != -1)
|
|
{
|
|
k = nextspritesect[j];
|
|
changespritesect(j, numsectors);
|
|
j = k;
|
|
}
|
|
}
|
|
|
|
numsectors++;
|
|
|
|
for (i=numwalls; i<newnumwalls; i++)
|
|
{
|
|
if (wall[i].nextwall >= 0)
|
|
{
|
|
NEXTWALL(i).nextwall = i;
|
|
NEXTWALL(i).nextsector = numsectors-1;
|
|
}
|
|
}
|
|
|
|
numwalls = newnumwalls;
|
|
newnumwalls = -1;
|
|
|
|
// clean out nextwall links for deletesector
|
|
for (k=0; k<2; k++)
|
|
for (WALLS_OF_SECTOR(joinsector[k], j))
|
|
{
|
|
wall[j].nextwall = wall[j].nextsector = -1;
|
|
#ifdef YAX_ENABLE
|
|
yax_setnextwall(j, YAX_CEILING, -1);
|
|
yax_setnextwall(j, YAX_FLOOR, -1);
|
|
#endif
|
|
}
|
|
|
|
deletesector(joinsector[0]);
|
|
if (joinsector[0] < joinsector[1])
|
|
joinsector[1]--;
|
|
deletesector(joinsector[1]);
|
|
|
|
printmessage16("Sectors joined.");
|
|
mkonwinvalid();
|
|
asksave = 1;
|
|
#ifdef YAX_ENABLE
|
|
yax_update(0);
|
|
yax_updategrays(pos.z);
|
|
#endif
|
|
}
|
|
|
|
joinsector[0] = -1;
|
|
}
|
|
}
|
|
end_join_sectors:
|
|
|
|
// PK
|
|
for (i=0x02; i<=0x0b; i++) // keys '1' to '0' on the upper row
|
|
if (keystatus[i])
|
|
{
|
|
prefixarg = prefixtiles[i-2];
|
|
break;
|
|
}
|
|
|
|
if (eitherALT && keystatus[0x1f]) //ALT-S
|
|
{
|
|
keystatus[0x1f] = 0;
|
|
|
|
if (linehighlight >= 0 && wall[linehighlight].nextwall == -1)
|
|
{
|
|
newnumwalls = whitelinescan(sectorofwall(linehighlight), linehighlight);
|
|
if (newnumwalls < numwalls)
|
|
{
|
|
printmessage16("Can't make a sector out there.");
|
|
newnumwalls = -1;
|
|
}
|
|
else if (newnumwalls > MAXWALLS)
|
|
{
|
|
printmessage16("Making new sector from inner loop would exceed wall limits.");
|
|
newnumwalls = -1;
|
|
}
|
|
else
|
|
{
|
|
for (i=numwalls; i<newnumwalls; i++)
|
|
{
|
|
NEXTWALL(i).nextwall = i;
|
|
NEXTWALL(i).nextsector = numsectors;
|
|
#ifdef YAX_ENABLE
|
|
yax_setnextwall(i, YAX_CEILING, -1);
|
|
yax_setnextwall(i, YAX_FLOOR, -1);
|
|
#endif
|
|
}
|
|
#ifdef YAX_ENABLE
|
|
yax_setbunches(numsectors, -1, -1);
|
|
yax_update(0);
|
|
yax_updategrays(pos.z);
|
|
#endif
|
|
numwalls = newnumwalls;
|
|
newnumwalls = -1;
|
|
numsectors++;
|
|
|
|
asksave = 1;
|
|
printmessage16("Inner loop made into new sector.");
|
|
}
|
|
}
|
|
}
|
|
else if (keystatus[0x1f]) //S
|
|
{
|
|
int16_t sucksect = -1;
|
|
|
|
keystatus[0x1f] = 0;
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
if (inside_editor_curpos(i) == 1)
|
|
{
|
|
sucksect = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (sucksect >= 0)
|
|
{
|
|
dax = mousxplc;
|
|
day = mousyplc;
|
|
if (gridlock && grid > 0)
|
|
locktogrid(&dax, &day);
|
|
|
|
i = insert_sprite_common(sucksect, dax, day);
|
|
|
|
if (i < 0)
|
|
printmessage16("Couldn't insert sprite.");
|
|
else
|
|
{
|
|
sprite[i].z = getflorzofslope(sucksect,dax,day);
|
|
// PK
|
|
if (prefixarg)
|
|
{
|
|
sprite[i].picnum = prefixarg;
|
|
sprite[i].xrepeat = sprite[i].yrepeat = 48;
|
|
prefixarg = 0;
|
|
}
|
|
else handle_sprite_in_clipboard(i);
|
|
|
|
if (tilesiz[sprite[i].picnum].y >= 32)
|
|
sprite[i].cstat |= 1;
|
|
|
|
correct_sprite_yoffset(i);
|
|
|
|
printmessage16("Sprite inserted.");
|
|
|
|
asksave = 1;
|
|
|
|
VM_OnEvent(EVENT_INSERTSPRITE2D, i);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (keystatus[0x2e]) // C (make circle of points)
|
|
{
|
|
if (highlightsectorcnt > 0)
|
|
duplicate_selected_sectors();
|
|
else if (highlightcnt > 0)
|
|
duplicate_selected_sprites();
|
|
else if (circlewall >= 0)
|
|
{
|
|
circlewall = -1;
|
|
}
|
|
else if (!m32_sideview)
|
|
{
|
|
if (linehighlight >= 0)
|
|
{
|
|
#if 0 //def YAX_ENABLE
|
|
j = linehighlight;
|
|
|
|
if (yax_islockedwall(j) ||
|
|
(wall[j].nextwall >= 0 && yax_islockedwall(wall[j].nextwall)))
|
|
printmessage16("Can't make circle in wall constrained by sector extension.");
|
|
else
|
|
#endif
|
|
circlewall = linehighlight;
|
|
}
|
|
}
|
|
|
|
keystatus[0x2e] = 0;
|
|
}
|
|
|
|
bad = keystatus[0x39] && !m32_sideview; //Gotta do this to save lots of 3 spaces!
|
|
|
|
if (circlewall >= 0)
|
|
{
|
|
int32_t tempint1, tempint2;
|
|
|
|
x1 = wall[circlewall].x;
|
|
y1 = wall[circlewall].y;
|
|
x2 = POINT2(circlewall).x;
|
|
y2 = POINT2(circlewall).y;
|
|
x3 = mousxplc;
|
|
y3 = mousyplc;
|
|
adjustmark(&x3,&y3,newnumwalls);
|
|
tempint1 = dmulscale4(x3-x2,x1-x3, y1-y3,y3-y2);
|
|
tempint2 = dmulscale4(y1-y2,x1-x3, y1-y3,x2-x1);
|
|
|
|
if (tempint2 != 0)
|
|
{
|
|
int32_t goodtogo, err=0;
|
|
|
|
const int32_t centerx = ((x1+x2) + scale(y1-y2,tempint1,tempint2))>>1;
|
|
const int32_t centery = ((y1+y2) + scale(x2-x1,tempint1,tempint2))>>1;
|
|
const int32_t circlerad = ksqrt(dmulscale4(centerx-x1,centerx-x1, centery-y1,centery-y1))<<2;
|
|
|
|
const int32_t circleang1 = getangle(x1-centerx,y1-centery);
|
|
const int32_t circleang2 = getangle(x2-centerx,y2-centery);
|
|
|
|
const int32_t redw = (int32_t)(wall[circlewall].nextwall >= 0);
|
|
int32_t insdpoints = 0;
|
|
|
|
draw_cross(centerx, centery, 2, editorcolors[14]);
|
|
|
|
k = ((circleang2-circleang1)&2047);
|
|
if (mulscale4(x3-x1,y2-y1) < mulscale4(x2-x1,y3-y1))
|
|
{
|
|
k = -((circleang1-circleang2)&2047);
|
|
}
|
|
|
|
// XXX: Still too permissive for TROR insertion
|
|
goodtogo = (numwalls+(1+redw)*circlepoints <= MAXWALLS);
|
|
|
|
if (bad > 0 && goodtogo)
|
|
{
|
|
err = backup_drawn_walls(0);
|
|
|
|
if (err)
|
|
{
|
|
message("Error backing up drawn walls (code %d)!", err);
|
|
goodtogo = 0;
|
|
}
|
|
}
|
|
|
|
for (i=circlepoints; i>0; i--)
|
|
{
|
|
const int32_t ps = 2;
|
|
|
|
j = (circleang1 + scale(i,k,circlepoints+1))&2047;
|
|
dax = centerx + mulscale14(sintable[(j+512)&2047],circlerad);
|
|
day = centery + mulscale14(sintable[j],circlerad);
|
|
|
|
inpclamp(&dax, -editorgridextent, editorgridextent);
|
|
inpclamp(&day, -editorgridextent, editorgridextent);
|
|
|
|
if (bad > 0 && goodtogo)
|
|
{
|
|
int32_t inspts = M32_InsertPoint(circlewall, dax,day, -1, &circlewall);
|
|
|
|
if (inspts==0)
|
|
{
|
|
message("Wall limit exceeded while inserting points.");
|
|
goto end_circle_insertion;
|
|
}
|
|
else if (inspts >= 65536)
|
|
{
|
|
message("ERR: Inserted %d points for constr. wall (exp. %d; %d already ins'd)",
|
|
inspts&65535, inspts>>16, insdpoints);
|
|
goto end_circle_insertion;
|
|
}
|
|
|
|
insdpoints += inspts;
|
|
}
|
|
|
|
draw_square(dax, day, ps, editorcolors[14]);
|
|
}
|
|
|
|
if (bad > 0 && goodtogo)
|
|
backup_drawn_walls(1);
|
|
|
|
if (bad > 0)
|
|
{
|
|
if (goodtogo)
|
|
{
|
|
asksave = 1;
|
|
printmessage16("Circle points inserted.");
|
|
end_circle_insertion:
|
|
circlewall = -1;
|
|
mkonwinvalid();
|
|
}
|
|
else
|
|
printmessage16("Inserting circle points would exceed wall limit.");
|
|
}
|
|
}
|
|
|
|
bad = 0;
|
|
keystatus[0x39] = 0;
|
|
}
|
|
|
|
if (bad > 0) //Space bar test
|
|
{
|
|
keystatus[0x39] = 0;
|
|
adjustmark(&mousxplc,&mousyplc,newnumwalls);
|
|
|
|
if (checkautoinsert(mousxplc,mousyplc,newnumwalls) == 1)
|
|
{
|
|
printmessage16("You must insert a point there first.");
|
|
bad = 0;
|
|
}
|
|
}
|
|
|
|
if (bad > 0) //Space
|
|
{
|
|
if (newnumwalls < numwalls) // starting wall drawing
|
|
{
|
|
if (numwalls >= MAXWALLS-1)
|
|
{
|
|
// whatever we do, we will need at least two new walls
|
|
printmessage16("Can't start sector drawing: wall limit reached.");
|
|
goto end_space_handling;
|
|
}
|
|
|
|
if (numsectors >= MAXSECTORS)
|
|
{
|
|
printmessage16("Can't start sector drawing: sector limit reached.");
|
|
goto end_space_handling;
|
|
}
|
|
|
|
firstx = mousxplc;
|
|
firsty = mousyplc; //Make first point
|
|
newnumwalls = numwalls;
|
|
ovh.suckwall = -1;
|
|
ovh.split = 0;
|
|
|
|
init_new_wall1(&ovh.suckwall, mousxplc, mousyplc);
|
|
|
|
printmessage16("Sector drawing started.");
|
|
}
|
|
else // 2nd point and up...
|
|
{
|
|
//if not back to first point
|
|
if (firstx != mousxplc || firsty != mousyplc) //nextpoint
|
|
{
|
|
if (newnumwalls>=MAXWALLS)
|
|
{
|
|
printmessage16("Inserting another point would exceed wall limit.");
|
|
goto end_space_handling;
|
|
}
|
|
|
|
j = 0;
|
|
for (i=numwalls; i<newnumwalls; i++)
|
|
if (mousxplc == wall[i].x && mousyplc == wall[i].y)
|
|
j = 1;
|
|
|
|
if (j == 0) // if new point is not on a position of already drawn points
|
|
{
|
|
// on the second point insertion, check if starting to split a sector
|
|
if (newnumwalls == numwalls+1)
|
|
{
|
|
dax = (wall[numwalls].x+mousxplc)>>1;
|
|
day = (wall[numwalls].y+mousyplc)>>1;
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
if (inside(dax,day,i) != 1)
|
|
continue;
|
|
|
|
//check if first point at point of sector
|
|
m = -1;
|
|
for (WALLS_OF_SECTOR(i, k))
|
|
if (wall[k].x==wall[numwalls].x && wall[k].y==wall[numwalls].y)
|
|
{
|
|
YAX_SKIPWALL(k);
|
|
|
|
m = k;
|
|
break;
|
|
}
|
|
|
|
// if the second insertion is not on a neighboring point of the first one...
|
|
if (m>=0 && (POINT2(k).x != mousxplc || POINT2(k).y != mousyplc))
|
|
if (wall[lastwall(k)].x != mousxplc || wall[lastwall(k)].y != mousyplc)
|
|
{
|
|
ovh.split = 1;
|
|
ovh.splitsect = i;
|
|
ovh.splitstartwall = m;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//make new point
|
|
|
|
//make sure not drawing over old red line
|
|
bad = 0;
|
|
for (i=0; i<numwalls; i++)
|
|
{
|
|
YAX_SKIPWALL(i);
|
|
|
|
if (wall[i].nextwall >= 0)
|
|
{
|
|
int32_t lastwalx = wall[newnumwalls-1].x;
|
|
int32_t lastwaly = wall[newnumwalls-1].y;
|
|
|
|
YAX_SKIPWALL(wall[i].nextwall);
|
|
|
|
if (wall[i].x == mousxplc && wall[i].y == mousyplc)
|
|
if (POINT2(i).x == lastwalx && POINT2(i).y == lastwaly)
|
|
bad = 1;
|
|
if (wall[i].x == lastwalx && wall[i].y == lastwaly)
|
|
if (POINT2(i).x == mousxplc && POINT2(i).y == mousyplc)
|
|
bad = 1;
|
|
}
|
|
}
|
|
|
|
if (bad == 0)
|
|
{
|
|
init_new_wall1(&ovh.suckwall, mousxplc, mousyplc);
|
|
}
|
|
else
|
|
{
|
|
printmessage16("You can't draw new lines over red lines.");
|
|
goto end_space_handling;
|
|
}
|
|
}
|
|
}
|
|
|
|
////////// newnumwalls is at most MAXWALLS here //////////
|
|
|
|
//if not split and back to first point
|
|
if (!ovh.split && newnumwalls >= numwalls+3
|
|
&& firstx==mousxplc && firsty==mousyplc)
|
|
{
|
|
wall[newnumwalls-1].point2 = numwalls;
|
|
|
|
if (ovh.suckwall == -1) //if no connections to other sectors
|
|
{
|
|
k = -1;
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
|
|
if (inside(firstx,firsty,i) == 1)
|
|
{
|
|
// if all points inside that one sector i,
|
|
// will add an inner loop
|
|
for (j=numwalls+1; j<newnumwalls; j++)
|
|
{
|
|
if (inside(wall[j].x, wall[j].y, i) != 1)
|
|
goto check_next_sector;
|
|
}
|
|
|
|
k = i;
|
|
break;
|
|
}
|
|
check_next_sector: ;
|
|
}
|
|
|
|
if (k == -1) //if not inside another sector either
|
|
{
|
|
//add island sector
|
|
if (clockdir(numwalls) == CLOCKDIR_CCW)
|
|
flipwalls(numwalls,newnumwalls);
|
|
|
|
Bmemset(§or[numsectors], 0, sizeof(sectortype));
|
|
sector[numsectors].extra = -1;
|
|
|
|
sector[numsectors].wallptr = numwalls;
|
|
sector[numsectors].wallnum = newnumwalls-numwalls;
|
|
sector[numsectors].ceilingz = -(32<<8);
|
|
sector[numsectors].floorz = (32<<8);
|
|
|
|
for (i=numwalls; i<newnumwalls; i++)
|
|
copy_some_wall_members(i, -1, 1);
|
|
#ifdef YAX_ENABLE
|
|
yax_setbunches(numsectors, -1, -1);
|
|
#endif
|
|
headspritesect[numsectors] = -1;
|
|
numsectors++;
|
|
|
|
numwalls = newnumwalls;
|
|
newnumwalls = -1;
|
|
|
|
printmessage16("Created new sector %d", numsectors-1);
|
|
}
|
|
else //else add loop to sector
|
|
{
|
|
int32_t ret = AddLoopToSector(k, NULL);
|
|
|
|
if (ret < 0)
|
|
goto end_space_handling;
|
|
#ifdef YAX_ENABLE
|
|
else if (ret > 0)
|
|
printmessage16("Added inner loop to sector %d and made new inner sector", k);
|
|
else
|
|
#endif
|
|
printmessage16("Added inner loop to sector %d", k);
|
|
mkonwinvalid();
|
|
asksave = 1;
|
|
}
|
|
}
|
|
else // if connected to at least one other sector
|
|
{
|
|
int16_t sucksect;
|
|
|
|
//add new sector with connections
|
|
|
|
if (clockdir(numwalls) == CLOCKDIR_CCW)
|
|
flipwalls(numwalls,newnumwalls);
|
|
|
|
for (i=numwalls; i<newnumwalls; i++)
|
|
{
|
|
copy_some_wall_members(i, ovh.suckwall, 1);
|
|
wall[i].cstat &= ~(1+16+32+64);
|
|
|
|
if (checksectorpointer(i, numsectors) > 0)
|
|
{
|
|
// if new red line, prefer the other-side wall as base
|
|
ovh.suckwall = wall[i].nextwall;
|
|
}
|
|
}
|
|
sucksect = sectorofwall(ovh.suckwall);
|
|
|
|
if (numsectors != sucksect)
|
|
Bmemcpy(§or[numsectors], §or[sucksect], sizeof(sectortype));
|
|
|
|
sector[numsectors].wallptr = numwalls;
|
|
sector[numsectors].wallnum = newnumwalls-numwalls;
|
|
|
|
sector[numsectors].extra = -1;
|
|
sector[numsectors].lotag = sector[numsectors].hitag = 0;
|
|
|
|
setslope(numsectors, YAX_CEILING, 0);
|
|
setslope(numsectors, YAX_FLOOR, 0);
|
|
|
|
sector[numsectors].ceilingpal = sector[numsectors].floorpal = 0;
|
|
#ifdef YAX_ENABLE
|
|
yax_setbunches(numsectors, -1, -1);
|
|
#endif
|
|
headspritesect[numsectors] = -1;
|
|
numsectors++;
|
|
|
|
numwalls = newnumwalls;
|
|
newnumwalls = -1;
|
|
|
|
message("Created new sector %d based on sector %d", numsectors-1, sucksect);
|
|
}
|
|
|
|
asksave = 1;
|
|
#ifdef YAX_ENABLE
|
|
yax_update(0);
|
|
yax_updategrays(pos.z);
|
|
#endif
|
|
|
|
goto end_space_handling;
|
|
}
|
|
////////// split sector //////////
|
|
else if (ovh.split == 1)
|
|
{
|
|
int16_t danumwalls, splitendwall, doSectorSplit;
|
|
int16_t secondstartwall=-1; // used only with splitting
|
|
int32_t expectedNumwalls = numwalls+2*(newnumwalls-numwalls-1), loopnum;
|
|
int32_t firstwallflag;
|
|
#ifdef YAX_ENABLE
|
|
int16_t cb, fb;
|
|
#endif
|
|
startwall = sector[ovh.splitsect].wallptr;
|
|
endwall = startwall + sector[ovh.splitsect].wallnum - 1;
|
|
|
|
firstwallflag = (startwall==ovh.splitstartwall || startwall==lastwall(ovh.splitstartwall));
|
|
|
|
// OSD_Printf("numwalls: %d, newnumwalls: %d\n", numwalls, newnumwalls);
|
|
i = -1;
|
|
for (k=startwall; k<=endwall; k++)
|
|
if (wall[k].x == wall[newnumwalls-1].x && wall[k].y == wall[newnumwalls-1].y)
|
|
{
|
|
i = k;
|
|
break;
|
|
}
|
|
// vvvv shouldn't happen, but you never know...
|
|
if (i==-1 || k==ovh.splitstartwall)
|
|
goto end_space_handling;
|
|
|
|
splitendwall = k;
|
|
doSectorSplit = (loopnumofsector(ovh.splitsect,ovh.splitstartwall)
|
|
== loopnumofsector(ovh.splitsect,splitendwall));
|
|
|
|
if (expectedNumwalls > MAXWALLS)
|
|
{
|
|
printmessage16("%s would exceed wall limit.", bad==0 ?
|
|
"Splitting sector" : "Joining sector loops");
|
|
newnumwalls--;
|
|
goto end_space_handling;
|
|
}
|
|
#ifdef YAX_ENABLE
|
|
yax_getbunches(ovh.splitsect, &cb, &fb);
|
|
|
|
if ((cb>=0 && (sector[ovh.splitsect].ceilingstat&2))
|
|
|| (fb>=0 && (sector[ovh.splitsect].floorstat&2)))
|
|
{
|
|
printmessage16("Sloped extended sectors cannot be split.");
|
|
newnumwalls--;
|
|
goto end_space_handling;
|
|
}
|
|
#endif
|
|
////////// common code for splitting/loop joining //////////
|
|
|
|
newnumwalls--; //first fix up the new walls
|
|
for (i=numwalls; i<newnumwalls; i++)
|
|
{
|
|
copy_some_wall_members(i, startwall, 1);
|
|
wall[i].point2 = i+1;
|
|
}
|
|
|
|
danumwalls = newnumwalls; //where to add more walls
|
|
|
|
if (doSectorSplit)
|
|
{
|
|
// Copy outer loop of first sector
|
|
if (do_while_copyloop1(splitendwall, ovh.splitstartwall, &danumwalls, numwalls))
|
|
goto split_not_enough_walls;
|
|
|
|
//Add other loops for 1st sector
|
|
i = loopnum = loopnumofsector(ovh.splitsect,ovh.splitstartwall);
|
|
|
|
for (j=startwall; j<=endwall; j++)
|
|
{
|
|
k = loopnumofsector(ovh.splitsect,j);
|
|
if (k == i)
|
|
continue;
|
|
|
|
if (k == loopnum)
|
|
continue;
|
|
|
|
i = k;
|
|
|
|
if (loopinside(wall[j].x,wall[j].y, numwalls) != 1)
|
|
continue;
|
|
|
|
if (do_while_copyloop1(j, j, &danumwalls, danumwalls))
|
|
goto split_not_enough_walls;
|
|
}
|
|
|
|
secondstartwall = danumwalls;
|
|
}
|
|
else
|
|
{
|
|
if (do_while_copyloop1(splitendwall, splitendwall, &danumwalls, -1))
|
|
goto split_not_enough_walls;
|
|
}
|
|
|
|
//copy split points for other sector backwards
|
|
for (j=newnumwalls; j>numwalls; j--)
|
|
{
|
|
Bmemcpy(&wall[danumwalls], &wall[j], sizeof(walltype));
|
|
wall[danumwalls].nextwall = -1;
|
|
wall[danumwalls].nextsector = -1;
|
|
wall[danumwalls].point2 = danumwalls+1;
|
|
#ifdef YAX_ENABLE
|
|
yax_setnextwall(danumwalls,YAX_CEILING, -1);
|
|
yax_setnextwall(danumwalls,YAX_FLOOR, -1);
|
|
#endif
|
|
danumwalls++;
|
|
}
|
|
|
|
//copy rest of loop next
|
|
if (doSectorSplit)
|
|
{
|
|
if (do_while_copyloop1(ovh.splitstartwall, splitendwall, &danumwalls, secondstartwall))
|
|
goto split_not_enough_walls;
|
|
}
|
|
else
|
|
{
|
|
if (do_while_copyloop1(ovh.splitstartwall, ovh.splitstartwall, &danumwalls, numwalls))
|
|
goto split_not_enough_walls;
|
|
}
|
|
|
|
//Add other loops for 2nd sector
|
|
i = loopnum = loopnumofsector(ovh.splitsect,ovh.splitstartwall);
|
|
|
|
for (j=startwall; j<=endwall; j++)
|
|
{
|
|
k = loopnumofsector(ovh.splitsect, j);
|
|
if (k==i)
|
|
continue;
|
|
|
|
if (doSectorSplit && k==loopnum)
|
|
continue;
|
|
if (!doSectorSplit && (k == loopnumofsector(ovh.splitsect,ovh.splitstartwall) ||
|
|
k == loopnumofsector(ovh.splitsect,splitendwall)))
|
|
continue;
|
|
|
|
i = k;
|
|
|
|
// was loopinside(... , secondstartwall) != 1, but this way there are
|
|
// no duplicate or left-out loops (can happen with convoluted geometry)
|
|
if (doSectorSplit && (loopinside(wall[j].x,wall[j].y, numwalls) != 0))
|
|
continue;
|
|
|
|
if (do_while_copyloop1(j, j, &danumwalls, danumwalls))
|
|
goto split_not_enough_walls;
|
|
}
|
|
|
|
//fix all next pointers on old sector line
|
|
for (j=numwalls; j<danumwalls; j++)
|
|
{
|
|
#ifdef YAX_ENABLE
|
|
// if (doSectorSplit || (j!=numwalls && j!=danumwalls-1))
|
|
yax_fixreverselinks(j, j);
|
|
#endif
|
|
if (wall[j].nextwall >= 0)
|
|
{
|
|
NEXTWALL(j).nextwall = j;
|
|
|
|
if (!doSectorSplit || j < secondstartwall)
|
|
NEXTWALL(j).nextsector = numsectors;
|
|
else
|
|
NEXTWALL(j).nextsector = numsectors+1;
|
|
}
|
|
}
|
|
|
|
//copy sector attributes & fix wall pointers
|
|
Bmemcpy(§or[numsectors], §or[ovh.splitsect], sizeof(sectortype));
|
|
sector[numsectors].wallptr = numwalls;
|
|
sector[numsectors].wallnum = (doSectorSplit?secondstartwall:danumwalls) - numwalls;
|
|
|
|
if (doSectorSplit)
|
|
{
|
|
//set all next pointers on split
|
|
for (j=numwalls; j<newnumwalls; j++)
|
|
{
|
|
m = secondstartwall+(newnumwalls-1-j);
|
|
wall[j].nextwall = m;
|
|
wall[j].nextsector = numsectors+1;
|
|
wall[m].nextwall = j;
|
|
wall[m].nextsector = numsectors;
|
|
}
|
|
|
|
Bmemcpy(§or[numsectors+1], §or[ovh.splitsect], sizeof(sectortype));
|
|
sector[numsectors+1].wallptr = secondstartwall;
|
|
sector[numsectors+1].wallnum = danumwalls-secondstartwall;
|
|
}
|
|
|
|
//fix sprites
|
|
j = headspritesect[ovh.splitsect];
|
|
while (j != -1)
|
|
{
|
|
k = nextspritesect[j];
|
|
if (!doSectorSplit || loopinside(sprite[j].x,sprite[j].y,numwalls) == 1)
|
|
changespritesect(j,numsectors);
|
|
//else if (loopinside(sprite[j].x,sprite[j].y,secondstartwall) == 1)
|
|
else //Make sure no sprites get left out & deleted!
|
|
changespritesect(j,numsectors+1);
|
|
j = k;
|
|
}
|
|
|
|
numsectors += 1 + doSectorSplit;
|
|
|
|
k = danumwalls-numwalls; //Back up number of walls of new sector for later
|
|
numwalls = danumwalls;
|
|
|
|
//clear out old sector's next pointers for clean deletesector
|
|
for (j=startwall; j<=endwall; j++)
|
|
{
|
|
#ifdef YAX_ENABLE
|
|
// same thing for yax-nextwalls (only forward links!)
|
|
yax_setnextwall(j, YAX_CEILING, -1);
|
|
yax_setnextwall(j, YAX_FLOOR, -1);
|
|
#endif
|
|
wall[j].nextwall = wall[j].nextsector = -1;
|
|
}
|
|
deletesector(ovh.splitsect);
|
|
|
|
//Check pointers
|
|
for (j=numwalls-k; j<numwalls; j++)
|
|
{
|
|
if (wall[j].nextwall >= 0)
|
|
checksectorpointer(wall[j].nextwall, wall[j].nextsector);
|
|
checksectorpointer(j, sectorofwall(j));
|
|
}
|
|
|
|
//k now safe to use as temp
|
|
|
|
if (numwalls==expectedNumwalls)
|
|
{
|
|
if (doSectorSplit && cb<0 && fb<0)
|
|
{
|
|
if (firstwallflag)
|
|
{
|
|
int32_t rhsnew1stwall = sector[numsectors-2].wallptr;
|
|
int32_t lhsotherwall = wall[rhsnew1stwall].nextwall;
|
|
|
|
Bassert(lhsotherwall >= 0);
|
|
|
|
setfirstwall(numsectors-2, lastwall(rhsnew1stwall));
|
|
setfirstwall(numsectors-1, wall[lhsotherwall].point2);
|
|
}
|
|
}
|
|
|
|
message("%s", doSectorSplit ? "Sector split." : "Loops joined.");
|
|
}
|
|
else
|
|
{
|
|
message("%s WARNING: CREATED %d MORE WALLS THAN EXPECTED!",
|
|
doSectorSplit ? "Sector split." : "Loops joined.",
|
|
numwalls-expectedNumwalls);
|
|
// this would display 'num* out of bounds' but this corruption
|
|
// is almost as bad... (shouldn't happen anymore)
|
|
if (numcorruptthings < MAXCORRUPTTHINGS)
|
|
corruptthings[numcorruptthings++] = 0;
|
|
corruptlevel = 5;
|
|
}
|
|
|
|
if (0)
|
|
{
|
|
split_not_enough_walls:
|
|
message("%s failed: not enough space beyond wall[]",
|
|
doSectorSplit ? "Splitting sectors" : "Joining loops");
|
|
}
|
|
|
|
newnumwalls = -1;
|
|
asksave = 1;
|
|
|
|
mkonwinvalid();
|
|
#ifdef YAX_ENABLE
|
|
yax_update(0);
|
|
yax_updategrays(pos.z);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
end_space_handling:
|
|
|
|
if (keystatus[0x1c]) //Left Enter
|
|
{
|
|
keystatus[0x1c] = 0;
|
|
if (keystatus[0x2a] && keystatus[0x1d]) // LCtrl+LShift
|
|
{
|
|
#ifdef YAX_ENABLE
|
|
if (numyaxbunches == 0 ||
|
|
(fade_editor_screen(-1), ask_if_sure("Really check all wall pointers in TROR map?", 0)))
|
|
#endif
|
|
{
|
|
printmessage16("CHECKING ALL POINTERS!");
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
startwall = sector[i].wallptr;
|
|
for (j=startwall; j<numwalls; j++)
|
|
if (startwall > wall[j].point2)
|
|
startwall = wall[j].point2;
|
|
sector[i].wallptr = startwall;
|
|
}
|
|
for (i=numsectors-2; i>=0; i--)
|
|
sector[i].wallnum = sector[i+1].wallptr-sector[i].wallptr;
|
|
sector[numsectors-1].wallnum = numwalls-sector[numsectors-1].wallptr;
|
|
|
|
for (i=0; i<numwalls; i++)
|
|
{
|
|
wall[i].nextsector = -1;
|
|
wall[i].nextwall = -1;
|
|
}
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
for (WALLS_OF_SECTOR(i, j))
|
|
checksectorpointer(j, i);
|
|
}
|
|
printmessage16("ALL POINTERS CHECKED!");
|
|
asksave = 1;
|
|
}
|
|
}
|
|
else // NOT (LCtrl + LShift)
|
|
{
|
|
if (newnumwalls > numwalls) // batch insert points
|
|
{
|
|
const int32_t numdrawnwalls = newnumwalls-numwalls;
|
|
vec2_t *point = (vec2_t *)tempxyar; // [MAXWALLS][2]
|
|
int32_t insdpoints = 0;
|
|
|
|
// back up the points of the line strip
|
|
for (i=0; i<numdrawnwalls+1; i++)
|
|
{
|
|
point[i].x = wall[numwalls+i].x;
|
|
point[i].y = wall[numwalls+i].y;
|
|
}
|
|
|
|
newnumwalls = -1;
|
|
|
|
for (i=0; i<numdrawnwalls; i++)
|
|
{
|
|
for (j=numwalls-1; j>=0; j--) /* j may be modified in loop */
|
|
{
|
|
vec2_t pint;
|
|
int32_t inspts;
|
|
|
|
YAX_SKIPWALL(j);
|
|
|
|
if (!lineintersect2v((vec2_t *)&wall[j], (vec2_t *)&POINT2(j),
|
|
&point[i], &point[i+1], &pint))
|
|
continue;
|
|
|
|
if (vec2eq(&pint, (vec2_t *)&wall[j]) || vec2eq(&pint, (vec2_t *)&POINT2(j)))
|
|
continue;
|
|
|
|
inspts = M32_InsertPoint(j, pint.x, pint.y, -1, &j); /* maybe modify j */
|
|
|
|
if (inspts==0)
|
|
{
|
|
printmessage16("Wall limit exceeded while inserting points.");
|
|
goto end_batch_insert_points;
|
|
}
|
|
else if (inspts >= 65536)
|
|
{
|
|
message("ERR: Inserted %d points for constr. wall (exp. %d; %d already ins'd)",
|
|
inspts&65535, inspts>>16, insdpoints);
|
|
goto end_batch_insert_points;
|
|
}
|
|
|
|
insdpoints += inspts;
|
|
}
|
|
}
|
|
|
|
message("Batch-inserted %d points in total", insdpoints);
|
|
end_batch_insert_points:
|
|
|
|
if (insdpoints != 0)
|
|
{
|
|
#ifdef YAX_ENABLE
|
|
yax_updategrays(pos.z);
|
|
#endif
|
|
mkonwinvalid_keeptempsect();
|
|
asksave = 1;
|
|
}
|
|
}
|
|
else if (linehighlight >= 0)
|
|
{
|
|
checksectorpointer(linehighlight,sectorofwall(linehighlight));
|
|
printmessage16("Checked pointers of highlighted line.");
|
|
asksave = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
{
|
|
static int32_t backspace_last = 0;
|
|
|
|
if (keystatus[0x0e]) //Backspace
|
|
{
|
|
keystatus[0x0e] = 0;
|
|
|
|
if (newnumwalls >= numwalls)
|
|
{
|
|
backspace_last = 1;
|
|
|
|
if (newnumwalls == numwalls+1 || keystatus[0x1d]) // LCtrl: delete all newly drawn walls
|
|
newnumwalls = -1;
|
|
else
|
|
newnumwalls--;
|
|
}
|
|
else if (backspace_last==0)
|
|
{
|
|
graphicsmode += (1-2*(DOWN_BK(RUN) || keystatus[0x36]))+3;
|
|
graphicsmode %= 3;
|
|
printmessage16("2D mode textures %s",
|
|
(graphicsmode == 2)?"enabled w/ animation":graphicsmode?"enabled":"disabled");
|
|
}
|
|
}
|
|
else
|
|
backspace_last = 0;
|
|
}
|
|
|
|
if (keystatus[0xd3] && eitherCTRL && numwalls > 0) //sector delete
|
|
{
|
|
int32_t numdelsectors = 0;
|
|
char *origframe=NULL;
|
|
|
|
#ifdef YAX_ENABLE
|
|
int16_t cb, fb;
|
|
uint8_t bunchbitmap[YAX_MAXBUNCHES>>3];
|
|
Bmemset(bunchbitmap, 0, sizeof(bunchbitmap));
|
|
#endif
|
|
keystatus[0xd3] = 0;
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
numdelsectors += (inside_editor_curpos(i) == 1);
|
|
}
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
if (highlightsectorcnt <= 0 || !keystatus[0x2a]) // LShift
|
|
{
|
|
YAX_SKIPSECTOR(i);
|
|
|
|
if (inside_editor_curpos(i) != 1)
|
|
continue;
|
|
}
|
|
|
|
k = 0;
|
|
if (highlightsectorcnt > 0)
|
|
{
|
|
// LShift: force highlighted sector deleting
|
|
if (keystatus[0x2a] || (hlsectorbitmap[i>>3]&(1<<(i&7))))
|
|
{
|
|
for (j=highlightsectorcnt-1; j>=0; j--)
|
|
{
|
|
#ifdef YAX_ENABLE
|
|
yax_getbunches(highlightsector[j], &cb, &fb);
|
|
if (cb>=0) bunchbitmap[cb>>3] |= (1<<(cb&7));
|
|
if (fb>=0) bunchbitmap[fb>>3] |= (1<<(fb&7));
|
|
#endif
|
|
deletesector(highlightsector[j]);
|
|
for (k=j-1; k>=0; k--)
|
|
if (highlightsector[k] >= highlightsector[j])
|
|
highlightsector[k]--;
|
|
}
|
|
|
|
printmessage16("Highlighted sectors deleted.");
|
|
mkonwinvalid();
|
|
k = 1;
|
|
}
|
|
}
|
|
|
|
if (k == 0)
|
|
{
|
|
if (numdelsectors > 1)
|
|
{
|
|
if (!bakframe_fillandfade(&origframe, i, "Delete this sector? (Y/N)"))
|
|
continue;
|
|
}
|
|
|
|
#ifdef YAX_ENABLE
|
|
yax_getbunches(i, &cb, &fb);
|
|
if (cb>=0) bunchbitmap[cb>>3] |= (1<<(cb&7));
|
|
if (fb>=0) bunchbitmap[fb>>3] |= (1<<(fb&7));
|
|
#endif
|
|
deletesector(i);
|
|
mkonwinvalid();
|
|
printmessage16("Sector deleted.");
|
|
}
|
|
|
|
if (origframe)
|
|
Bfree(origframe);
|
|
|
|
#ifdef YAX_ENABLE
|
|
for (j=0; j<numsectors; j++)
|
|
{
|
|
yax_getbunches(j, &cb, &fb);
|
|
if (cb>=0 && (bunchbitmap[cb>>3] & (1<<(cb&7))))
|
|
yax_setbunch(j, YAX_CEILING, -1);
|
|
if (fb>=0 && (bunchbitmap[fb>>3] & (1<<(fb&7))))
|
|
yax_setbunch(j, YAX_FLOOR, -1);
|
|
}
|
|
#endif
|
|
reset_highlightsector();
|
|
reset_highlight();
|
|
|
|
newnumwalls = -1;
|
|
asksave = 1;
|
|
#ifdef YAX_ENABLE
|
|
yax_update(0);
|
|
yax_updategrays(pos.z);
|
|
#endif
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (keystatus[0xd3] && (pointhighlight >= 0))
|
|
{
|
|
if ((pointhighlight&0xc000) == 16384) //Sprite Delete
|
|
{
|
|
deletesprite(pointhighlight&16383);
|
|
printmessage16("Sprite deleted.");
|
|
|
|
update_highlight();
|
|
asksave = 1;
|
|
}
|
|
keystatus[0xd3] = 0;
|
|
}
|
|
|
|
if (keystatus[0xd2] || keystatus[0x17]) //InsertPoint
|
|
{
|
|
if (highlightsectorcnt > 0)
|
|
duplicate_selected_sectors();
|
|
else if (highlightcnt > 0)
|
|
duplicate_selected_sprites();
|
|
else if (linehighlight2 >= 0)
|
|
{
|
|
int32_t onewnumwalls = newnumwalls;
|
|
int32_t wallis2sided = (wall[linehighlight2].nextwall>=0);
|
|
|
|
int32_t err = backup_drawn_walls(0);
|
|
|
|
if (err)
|
|
{
|
|
message("Error backing up drawn walls (code %d)!", err);
|
|
}
|
|
else if (max(numwalls,onewnumwalls) >= MAXWALLS-wallis2sided)
|
|
{
|
|
printmessage16("Inserting point would exceed wall limit.");
|
|
}
|
|
else
|
|
{
|
|
getclosestpointonwall(m32_sideview?searchx:mousxplc, m32_sideview?searchy:mousyplc,
|
|
linehighlight2, &dax,&day, 1);
|
|
i = linehighlight2;
|
|
if (m32_sideview)
|
|
{
|
|
int32_t y_p, d, dx, dy, frac;
|
|
|
|
dx = dax - m32_wallscreenxy[i][0];
|
|
dy = day - m32_wallscreenxy[i][1];
|
|
d = max(dx, dy);
|
|
y_p = (dy>dx);
|
|
|
|
if (d==0)
|
|
goto point_not_inserted;
|
|
|
|
frac = divscale24(d, m32_wallscreenxy[wall[i].point2][y_p]-m32_wallscreenxy[i][y_p]);
|
|
dax = POINT2(i).x - wall[i].x;
|
|
day = POINT2(i).y - wall[i].y;
|
|
dax = wall[i].x + mulscale24(dax,frac);
|
|
day = wall[i].y + mulscale24(day,frac);
|
|
}
|
|
|
|
adjustmark(&dax,&day, newnumwalls);
|
|
if ((wall[i].x == dax && wall[i].y == day) || (POINT2(i).x == dax && POINT2(i).y == day))
|
|
{
|
|
point_not_inserted:
|
|
printmessage16("Point not inserted.");
|
|
}
|
|
else
|
|
{
|
|
int32_t insdpoints = M32_InsertPoint(linehighlight2, dax, day, onewnumwalls, NULL);
|
|
|
|
if (insdpoints == 0)
|
|
{
|
|
printmessage16("Inserting points would exceed wall limit.");
|
|
goto end_insert_points;
|
|
}
|
|
else if (insdpoints == 1)
|
|
{
|
|
printmessage16("Point inserted.");
|
|
}
|
|
else if (insdpoints > 1 && insdpoints < 65536)
|
|
{
|
|
message("Inserted %d points for constrained wall.", insdpoints);
|
|
}
|
|
else // insdpoints >= 65536
|
|
{
|
|
message("Inserted %d points for constrained wall (expected %d, WTF?).",
|
|
insdpoints&65535, insdpoints>>16);
|
|
}
|
|
|
|
#ifdef YAX_ENABLE
|
|
yax_updategrays(pos.z);
|
|
#endif
|
|
mkonwinvalid_keeptempsect();
|
|
}
|
|
}
|
|
|
|
end_insert_points:
|
|
backup_drawn_walls(1);
|
|
|
|
asksave = 1;
|
|
}
|
|
keystatus[0xd2] = keystatus[0x17] = 0;
|
|
}
|
|
|
|
/*j = 0;
|
|
for(i=22-1;i>=0;i--) updatecrc16(j,kensig[i]);
|
|
if ((j&0xffff) != 0xebf)
|
|
{
|
|
printf("Don't screw with my name.\n");
|
|
Bexit(0);
|
|
}*/
|
|
//printext16(9L,336+9L,4,-1,kensig,0);
|
|
//printext16(8L,336+8L,12,-1,kensig,0);
|
|
|
|
showframe(1);
|
|
synctics = totalclock-lockclock;
|
|
lockclock += synctics;
|
|
|
|
if (keystatus[buildkeys[BK_MODE2D_3D]])
|
|
{
|
|
updatesector(pos.x,pos.y,&cursectnum);
|
|
if (cursectnum >= 0)
|
|
keystatus[buildkeys[BK_MODE2D_3D]] = 2;
|
|
else
|
|
printmessage16("Arrow must be inside a sector before entering 3D mode.");
|
|
}
|
|
|
|
// vvv PK ------------------------------------ (LShift) Ctrl-X: (prev) next map
|
|
// this is copied from 'L' (load map), but without copying the highlighted sectors
|
|
|
|
if (quickmapcycling && keystatus[0x2d]) //X
|
|
{
|
|
if (eitherCTRL) //Ctrl
|
|
{
|
|
nextmap:
|
|
// bad = 0;
|
|
i = menuselect_auto(keystatus[0x2a] ? 0:1); // LShift: prev map
|
|
if (i < 0)
|
|
{
|
|
if (i == -1)
|
|
message("No more map files.");
|
|
else if (i == -2)
|
|
message("No .MAP files found.");
|
|
}
|
|
else
|
|
{
|
|
if (LoadBoard(NULL, 4))
|
|
goto nextmap;
|
|
|
|
RESET_EDITOR_VARS();
|
|
oposz = pos.z;
|
|
}
|
|
showframe(1);
|
|
keystatus[0x1c] = 0;
|
|
|
|
keystatus[0x2d]=keystatus[0x13]=0;
|
|
|
|
}
|
|
}
|
|
// ^^^ PK ------------------------------------
|
|
|
|
if (keystatus[1] && joinsector[0] >= 0)
|
|
{
|
|
keystatus[1]=0;
|
|
joinsector[0]=-1;
|
|
printmessage16("No sectors joined.");
|
|
}
|
|
|
|
CANCEL:
|
|
if (keystatus[1])
|
|
{
|
|
keystatus[1] = 0;
|
|
#if M32_UNDO
|
|
_printmessage16("(N)ew, (L)oad, (S)ave, save (A)s, (T)est map, (U)ndo, (R)edo, (Q)uit");
|
|
#else
|
|
_printmessage16("(N)ew, (L)oad, (S)ave, save (A)s, (T)est map, (Q)uit");
|
|
#endif
|
|
printext16(16*8, ydim-STATUS2DSIZ2-12, editorcolors[15], -1, GetSaveBoardFilename(NULL), 0);
|
|
|
|
showframe(1);
|
|
bflushchars();
|
|
bad = 1;
|
|
while (bad == 1)
|
|
{
|
|
char ch;
|
|
|
|
if (handleevents())
|
|
{
|
|
if (quitevent)
|
|
quitevent = 0;
|
|
}
|
|
idle();
|
|
|
|
ch = bgetchar();
|
|
|
|
if (keystatus[1])
|
|
{
|
|
keystatus[1] = 0;
|
|
bad = 0;
|
|
// printmessage16("");
|
|
}
|
|
else if (ch == 'n' || ch == 'N') //N
|
|
{
|
|
bad = 0;
|
|
|
|
if (!ask_if_sure("Are you sure you want to start a new board? (Y/N)", 0))
|
|
break;
|
|
else
|
|
{
|
|
int32_t bakstat=-1;
|
|
mapinfofull_t bakmap;
|
|
|
|
if (highlightsectorcnt > 0)
|
|
bakstat = backup_highlighted_map(&bakmap);
|
|
|
|
// Bmemset(hlsectorbitmap, 0, sizeof(hlsectorbitmap));
|
|
// highlightsectorcnt = -1;
|
|
|
|
highlightcnt = -1;
|
|
//Clear all highlights
|
|
Bmemset(show2dwall, 0, sizeof(show2dwall));
|
|
Bmemset(show2dsprite, 0, sizeof(show2dsprite));
|
|
|
|
for (i=0; i<MAXSECTORS; i++) sector[i].extra = -1;
|
|
for (i=0; i<MAXWALLS; i++) wall[i].extra = -1;
|
|
for (i=0; i<MAXSPRITES; i++) sprite[i].extra = -1;
|
|
|
|
RESET_EDITOR_VARS();
|
|
mkonwinvalid();
|
|
|
|
reset_default_mapstate();
|
|
|
|
Bstrcpy(boardfilename,"newboard.map");
|
|
CallExtLoadMap(boardfilename);
|
|
#if M32_UNDO
|
|
map_undoredo_free();
|
|
#endif
|
|
if (bakstat==0)
|
|
{
|
|
bakstat = restore_highlighted_map(&bakmap, 1);
|
|
if (bakstat == -1)
|
|
{
|
|
message("Can't copy highlighted portion of old map: limits exceeded.");
|
|
reset_highlightsector();
|
|
}
|
|
}
|
|
|
|
CheckMapCorruption(4, 0);
|
|
|
|
break;
|
|
}
|
|
|
|
// printmessage16("");
|
|
showframe(1);
|
|
}
|
|
else if (ch == 'l' || ch == 'L') //L
|
|
{
|
|
bad = 0;
|
|
i = menuselect();
|
|
if (i < 0)
|
|
{
|
|
if (i == -2)
|
|
printmessage16("No .MAP files found.");
|
|
}
|
|
else
|
|
{
|
|
int32_t bakstat=-1, ret;
|
|
mapinfofull_t bakmap;
|
|
|
|
if (highlightsectorcnt > 0)
|
|
bakstat = backup_highlighted_map(&bakmap);
|
|
|
|
ret = LoadBoard(NULL, 4);
|
|
if (ret)
|
|
{
|
|
message("^13Invalid map format, nothing loaded (code %d).", ret);
|
|
if (bakstat==0)
|
|
mapinfofull_free(&bakmap);
|
|
}
|
|
else
|
|
{
|
|
RESET_EDITOR_VARS();
|
|
oposz = pos.z;
|
|
|
|
if (bakstat==0)
|
|
{
|
|
bakstat = restore_highlighted_map(&bakmap, 1);
|
|
if (bakstat == -1)
|
|
{
|
|
message("Can't copy highlighted portion of old map: limits exceeded.");
|
|
reset_highlightsector();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
showframe(1);
|
|
keystatus[0x1c] = 0;
|
|
}
|
|
else if (ch == 'a' || ch == 'A') //A
|
|
{
|
|
int32_t corrupt = CheckMapCorruption(4, 0);
|
|
|
|
bad = 0;
|
|
|
|
// Back up full name.
|
|
Bstrcpy(selectedboardfilename, boardfilename);
|
|
|
|
// Get base name.
|
|
{
|
|
const char *basefn = getbasefn(boardfilename);
|
|
|
|
if (basefn != boardfilename)
|
|
Bmemmove(boardfilename, basefn, Bstrlen(basefn)+1);
|
|
}
|
|
|
|
i = 0;
|
|
while (boardfilename[i] != 0 && i < 64)
|
|
i++;
|
|
if (i >= 4 && boardfilename[i-4] == '.')
|
|
i -= 4;
|
|
boardfilename[i] = 0;
|
|
|
|
bflushchars();
|
|
while (bad == 0)
|
|
{
|
|
_printmessage16("%sSave as: ^011%s%s", corrupt>=4?"(map corrupt) ":"",
|
|
boardfilename, (totalclock&32)?"_":"");
|
|
showframe(1);
|
|
|
|
if (handleevents())
|
|
quitevent = 0;
|
|
|
|
idle();
|
|
|
|
ch = bgetchar();
|
|
|
|
if (keystatus[1]) bad = 1;
|
|
else if (ch == 13) bad = 2;
|
|
else if (ch > 0)
|
|
{
|
|
if (i > 0 && (ch == 8 || ch == 127))
|
|
{
|
|
i--;
|
|
boardfilename[i] = 0;
|
|
}
|
|
else if (i < 40 && ch > 32 && ch < 128)
|
|
{
|
|
boardfilename[i++] = ch;
|
|
boardfilename[i] = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (bad == 1)
|
|
{
|
|
Bstrcpy(boardfilename, selectedboardfilename);
|
|
keystatus[1] = 0;
|
|
printmessage16("Operation cancelled");
|
|
showframe(1);
|
|
}
|
|
else if (bad == 2)
|
|
{
|
|
char *slash;
|
|
|
|
keystatus[0x1c] = 0;
|
|
|
|
Bstrcpy(&boardfilename[i], ".map");
|
|
|
|
// Update full name with new basename.
|
|
slash = Bstrrchr(selectedboardfilename,'/');
|
|
Bstrcpy(slash ? slash+1 : selectedboardfilename, boardfilename);
|
|
|
|
SaveBoardAndPrintMessage(selectedboardfilename);
|
|
|
|
Bstrcpy(boardfilename, selectedboardfilename);
|
|
CallExtSetupMapFilename(boardfilename);
|
|
}
|
|
bad = 0;
|
|
}
|
|
else if (ch == 's' || ch == 'S') //S
|
|
{
|
|
bad = 0;
|
|
|
|
if (CheckMapCorruption(4, 0)>=4)
|
|
{
|
|
fade_editor_screen(-1);
|
|
if (!ask_if_sure("Map is corrupt. Are you sure you want to save? (Y/N)", 0))
|
|
break;
|
|
}
|
|
|
|
SaveBoardAndPrintMessage(NULL);
|
|
|
|
showframe(1);
|
|
}
|
|
else if (ch == 't' || ch == 'T')
|
|
{
|
|
test_map(0);
|
|
}
|
|
#if M32_UNDO
|
|
else if (ch == 'u' || ch == 'U')
|
|
{
|
|
bad = 0;
|
|
if (map_undoredo(0)) printmessage16("Nothing to undo!");
|
|
else printmessage16("Revision %d undone",map_revision);
|
|
}
|
|
else if (ch == 'r' || ch == 'R')
|
|
{
|
|
bad = 0;
|
|
if (map_undoredo(1)) printmessage16("Nothing to redo!");
|
|
else printmessage16("Restored revision %d",map_revision-1);
|
|
}
|
|
#endif
|
|
else if (ch == 'q' || ch == 'Q') //Q
|
|
{
|
|
bad = 0;
|
|
|
|
if (ask_if_sure("Are you sure you want to quit?", 0))
|
|
{
|
|
//QUIT!
|
|
|
|
int32_t corrupt = CheckMapCorruption(4, 0);
|
|
|
|
if (ask_if_sure(corrupt<4?"Save changes?":"Map corrupt. Save changes?", 2+(corrupt>=4)))
|
|
SaveBoard(NULL, M32_SB_ASKOV);
|
|
|
|
while (keystatus[1] || keystatus[0x2e])
|
|
{
|
|
keystatus[1] = 0;
|
|
keystatus[0x2e] = 0;
|
|
quitevent = 0;
|
|
printmessage16("Operation cancelled");
|
|
showframe(1);
|
|
goto CANCEL;
|
|
}
|
|
|
|
CallExtUnInit();
|
|
// clearfilenames();
|
|
uninitengine();
|
|
|
|
Bexit(0);
|
|
}
|
|
|
|
// printmessage16("");
|
|
showframe(1);
|
|
}
|
|
}
|
|
|
|
clearkeys();
|
|
}
|
|
|
|
VM_OnEvent(EVENT_KEYS2D, -1);
|
|
|
|
//nextpage();
|
|
}
|
|
|
|
for (i=0; i<highlightsectorcnt; i++)
|
|
{
|
|
for (WALLS_OF_SECTOR(highlightsector[i], j))
|
|
{
|
|
if (wall[j].nextwall >= 0)
|
|
checksectorpointer(wall[j].nextwall, wall[j].nextsector);
|
|
checksectorpointer(j, highlightsector[i]);
|
|
}
|
|
}
|
|
mkonwinvalid_keeptempsect();
|
|
|
|
fixspritesectors();
|
|
|
|
if (setgamemode(fullscreen,xdimgame,ydimgame,bppgame) < 0)
|
|
{
|
|
initprintf("%d * %d not supported in this graphics mode\n",xdim,ydim);
|
|
CallExtUnInit();
|
|
// clearfilenames();
|
|
uninitengine();
|
|
Bexit(1);
|
|
}
|
|
|
|
setbrightness(GAMMA_CALC,0,0);
|
|
|
|
pos.z = oposz;
|
|
|
|
searchx = clamp(scale(searchx,xdimgame,xdim2d), 8, xdimgame-8-1);
|
|
searchy = clamp(scale(searchy,ydimgame,ydim2d-STATUS2DSIZ), 8, ydimgame-8-1);
|
|
|
|
VM_OnEvent(EVENT_ENTER3DMODE, -1);
|
|
}
|
|
|
|
// flags:
|
|
// 1: quit_is_yes
|
|
// 2: don't clear keys on return
|
|
int32_t ask_if_sure(const char *query, uint32_t flags)
|
|
{
|
|
char ch;
|
|
int32_t ret=-1;
|
|
|
|
if (!query)
|
|
_printmessage16("Are you sure?");
|
|
else
|
|
_printmessage16("%s", query);
|
|
showframe(1);
|
|
bflushchars();
|
|
|
|
while ((keystatus[1]|keystatus[0x2e]) == 0 && ret==-1)
|
|
{
|
|
if (handleevents())
|
|
{
|
|
if (quitevent)
|
|
{
|
|
if (flags&1)
|
|
return 1;
|
|
else
|
|
quitevent = 0;
|
|
}
|
|
}
|
|
idle();
|
|
|
|
ch = bgetchar();
|
|
|
|
if (ch == 'y' || ch == 'Y')
|
|
ret = 1;
|
|
else if (ch == 'n' || ch == 'N' || ch == 13 || ch == ' ')
|
|
ret = 0;
|
|
}
|
|
|
|
if ((flags&2)==0)
|
|
clearkeys();
|
|
|
|
if (ret >= 0)
|
|
return ret;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int32_t editor_ask_function(const char *question, const char *dachars, int32_t numchars)
|
|
{
|
|
char ch;
|
|
int32_t i, ret=-1;
|
|
|
|
_printmessage16("%s", question);
|
|
|
|
showframe(1);
|
|
bflushchars();
|
|
|
|
// 'c' is cancel too, but can be overridden
|
|
while ((keystatus[1]|keystatus[0x2e]) == 0 && ret==-1)
|
|
{
|
|
if (handleevents())
|
|
quitevent = 0;
|
|
|
|
idle();
|
|
ch = bgetchar();
|
|
|
|
for (i=0; i<numchars; i++)
|
|
if (ch==Btolower(dachars[i]) || ch==Btoupper(dachars[i]))
|
|
ret = i;
|
|
}
|
|
|
|
clearkeys();
|
|
|
|
return ret;
|
|
}
|
|
|
|
#ifdef YAX_ENABLE
|
|
static int32_t ask_above_or_below(void)
|
|
{
|
|
char dachars[2] = {'a', 'z'};
|
|
return editor_ask_function("Extend above (a) or below (z)?", dachars, 2);
|
|
}
|
|
#endif
|
|
|
|
static void SaveBoardAndPrintMessage(const char *fn)
|
|
{
|
|
const char *f;
|
|
|
|
_printmessage16("Saving board...");
|
|
showframe(1);
|
|
|
|
f = SaveBoard(fn, M32_SB_ASKOV);
|
|
|
|
if (f)
|
|
{
|
|
if (saveboard_fixedsprites)
|
|
message("Saved board %sto %s (changed sectnums of %d sprites).",
|
|
saveboard_savedtags?"and tags ":"", f, saveboard_fixedsprites);
|
|
else
|
|
message("Saved board %sto %s.", saveboard_savedtags?"and tags ":"", f);
|
|
}
|
|
else
|
|
{
|
|
if (!saveboard_canceled)
|
|
{
|
|
if (saveboard_fixedsprites)
|
|
message("^13SAVING BOARD FAILED (changed sectnums of %d sprites).",
|
|
saveboard_fixedsprites);
|
|
else
|
|
message("^13SAVING BOARD FAILED.");
|
|
}
|
|
}
|
|
}
|
|
|
|
// get the file name of the file that would be written if SaveBoard(fn, 0) was called
|
|
const char *GetSaveBoardFilename(const char *fn)
|
|
{
|
|
if (!fn)
|
|
fn = boardfilename;
|
|
|
|
if (pathsearchmode)
|
|
return fn;
|
|
|
|
// virtual filesystem mode can't save to directories so drop the file into
|
|
// the current directory
|
|
return getbasefn(fn);
|
|
}
|
|
|
|
// flags: see enum SaveBoardFlags.
|
|
// returns: NULL on failure, file name on success.
|
|
const char *SaveBoard(const char *fn, uint32_t flags)
|
|
{
|
|
int32_t ret;
|
|
const char *f = GetSaveBoardFilename(fn);
|
|
|
|
saveboard_canceled = 0;
|
|
#ifdef NEW_MAP_FORMAT
|
|
if ((flags&M32_SB_ASKOV) && mapversion>=10 &&
|
|
g_loadedMapVersion != -1 && g_loadedMapVersion < mapversion)
|
|
{
|
|
char question[128];
|
|
// XXX: This message is potentially confusing if the user is "Saving
|
|
// As" to a new file name.
|
|
Bsnprintf(question, sizeof(question), "Are you sure to overwrite a version "
|
|
"V%d map with a V%d map-text one?", g_loadedMapVersion, mapversion);
|
|
|
|
if (AskIfSure(question))
|
|
{
|
|
message("Cancelled saving board");
|
|
saveboard_canceled = 1;
|
|
return NULL;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
saveboard_savedtags = 0;
|
|
saveboard_fixedsprites = CallExtPreSaveMap();
|
|
|
|
ret = saveboard(f, &startpos, startang, startsectnum);
|
|
if ((flags&M32_SB_NOEXT)==0)
|
|
{
|
|
CallExtSaveMap(f);
|
|
saveboard_savedtags = !taglab_save(f);
|
|
}
|
|
|
|
return (ret==0) ? f : NULL;
|
|
}
|
|
|
|
// flags: 1: for running on Mapster32 init
|
|
// 4: passed to loadboard flags (no polymer_loadboard); implies no maphack loading
|
|
// returns:
|
|
// 0 on success,
|
|
// <0 on failure.
|
|
int32_t LoadBoard(const char *filename, uint32_t flags)
|
|
{
|
|
int32_t i, tagstat;
|
|
const int32_t loadingflags = (!pathsearchmode && grponlymode) ? 2 : 0;
|
|
|
|
if (!filename)
|
|
filename = selectedboardfilename;
|
|
|
|
editorzrange[0] = INT32_MIN;
|
|
editorzrange[1] = INT32_MAX;
|
|
|
|
CallExtPreLoadMap();
|
|
i = loadboard(filename, (flags&4)|loadingflags, &pos, &ang, &cursectnum);
|
|
if (i == -2)
|
|
i = loadoldboard(filename,loadingflags, &pos, &ang, &cursectnum);
|
|
|
|
if (i < 0)
|
|
{
|
|
// printmessage16("Invalid map format.");
|
|
return i;
|
|
}
|
|
|
|
// Success, so copy the file name.
|
|
if (filename != boardfilename)
|
|
Bstrcpy(boardfilename, filename);
|
|
|
|
mkonwinvalid();
|
|
|
|
highlightcnt = -1;
|
|
Bmemset(show2dwall, 0, sizeof(show2dwall)); //Clear all highlights
|
|
Bmemset(show2dsprite, 0, sizeof(show2dsprite));
|
|
|
|
if ((flags&4)==0)
|
|
loadmhk(0);
|
|
|
|
tagstat = taglab_load(boardfilename, loadingflags);
|
|
CallExtLoadMap(boardfilename);
|
|
|
|
{
|
|
char msgtail[64];
|
|
const int32_t ci = CheckMapCorruption(4, 0);
|
|
|
|
if (ci == 5)
|
|
Bstrcpy(msgtail, "^12(EXTREME corruption)");
|
|
else if (ci == 4)
|
|
Bstrcpy(msgtail, "^12(HEAVY corruption)");
|
|
else if (i > 0)
|
|
Bsprintf(msgtail, "^14(removed %d sprites)", i);
|
|
else if (ci >= 1 && ci < 4)
|
|
Bstrcpy(msgtail, "^14(moderate corruption)");
|
|
else
|
|
Bstrcpy(msgtail, "successfully");
|
|
|
|
message("Loaded V%d map %s%s %s", g_loadedMapVersion,
|
|
boardfilename, tagstat==0?" w/tags":"", msgtail);
|
|
}
|
|
|
|
startpos = pos; //this is same
|
|
startang = ang;
|
|
startsectnum = cursectnum;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void getpoint(int32_t searchxe, int32_t searchye, int32_t *x, int32_t *y)
|
|
{
|
|
inpclamp(&pos.x, -editorgridextent, editorgridextent);
|
|
inpclamp(&pos.y, -editorgridextent, editorgridextent);
|
|
|
|
searchxe -= halfxdim16;
|
|
searchye -= midydim16;
|
|
|
|
if (m32_sideview)
|
|
{
|
|
if (m32_sidesin!=0)
|
|
searchye = divscale14(searchye, m32_sidesin);
|
|
rotatepoint(0,0, searchxe,searchye, -m32_sideang, &searchxe,&searchye);
|
|
}
|
|
|
|
*x = pos.x + divscale14(searchxe,zoom);
|
|
*y = pos.y + divscale14(searchye,zoom);
|
|
|
|
inpclamp(x, -editorgridextent, editorgridextent);
|
|
inpclamp(y, -editorgridextent, editorgridextent);
|
|
}
|
|
|
|
static int32_t getlinehighlight(int32_t xplc, int32_t yplc, int32_t line, int8_t ignore_pointhighlight)
|
|
{
|
|
int32_t i, j, dst, dist, closest, x1, y1, x2, y2, nx, ny;
|
|
int32_t daxplc, dayplc;
|
|
|
|
if (numwalls == 0)
|
|
return -1;
|
|
|
|
if (mouseb & 1)
|
|
return line;
|
|
|
|
if (!ignore_pointhighlight && (pointhighlight&0xc000) == 16384)
|
|
return -1;
|
|
|
|
dist = 1024;
|
|
if (m32_sideview)
|
|
{
|
|
daxplc = searchx;
|
|
dayplc = searchy;
|
|
dist = mulscale14(dist, zoom);
|
|
}
|
|
else
|
|
{
|
|
daxplc = xplc;
|
|
dayplc = yplc;
|
|
}
|
|
|
|
closest = -1;
|
|
for (i=0; i<numwalls; i++)
|
|
{
|
|
if (!m32_sideview)
|
|
YAX_SKIPWALL(i);
|
|
|
|
getclosestpointonwall(daxplc,dayplc, i, &nx,&ny, 1);
|
|
dst = klabs(daxplc-nx) + klabs(dayplc-ny);
|
|
if (dst <= dist)
|
|
{
|
|
dist = dst;
|
|
closest = i;
|
|
}
|
|
}
|
|
|
|
if (closest>=0 && (j = wall[closest].nextwall) >= 0)
|
|
#ifdef YAX_ENABLE
|
|
if (m32_sideview || ((graywallbitmap[j>>3]&(1<<(j&7)))==0))
|
|
#endif
|
|
{
|
|
//if red line, allow highlighting of both sides
|
|
if (m32_sideview)
|
|
{
|
|
x1 = m32_wallscreenxy[closest][0];
|
|
y1 = m32_wallscreenxy[closest][1];
|
|
x2 = m32_wallscreenxy[wall[closest].point2][0];
|
|
y2 = m32_wallscreenxy[wall[closest].point2][1];
|
|
}
|
|
else
|
|
{
|
|
x1 = wall[closest].x;
|
|
y1 = wall[closest].y;
|
|
x2 = POINT2(closest).x;
|
|
y2 = POINT2(closest).y;
|
|
}
|
|
|
|
i = wall[closest].nextwall;
|
|
if (!m32_sideview ||
|
|
((B_UNBUF64(m32_wallscreenxy[closest]) == B_UNBUF64(m32_wallscreenxy[wall[j].point2])) &&
|
|
(B_UNBUF64(m32_wallscreenxy[wall[closest].point2]) == B_UNBUF64(m32_wallscreenxy[j]))))
|
|
if (dmulscale32(daxplc-x1,y2-y1,-(x2-x1),dayplc-y1) >= 0)
|
|
closest = j;
|
|
}
|
|
|
|
return closest;
|
|
}
|
|
|
|
int32_t getpointhighlight(int32_t xplc, int32_t yplc, int32_t point)
|
|
{
|
|
int32_t i, j, dst, dist = 512, closest = -1;
|
|
int32_t dax,day;
|
|
int32_t alwaysshowgray = get_alwaysshowgray();
|
|
|
|
if (numwalls == 0)
|
|
return -1;
|
|
|
|
if (mouseb & 1)
|
|
return point;
|
|
|
|
if (grid < 1)
|
|
dist = 0;
|
|
|
|
for (i=0; i<numsectors; i++)
|
|
{
|
|
if (!m32_sideview || !alwaysshowgray)
|
|
YAX_SKIPSECTOR(i);
|
|
|
|
for (j=sector[i].wallptr; j<sector[i].wallptr+sector[i].wallnum; j++)
|
|
{
|
|
if (!m32_sideview)
|
|
dst = klabs(xplc-wall[j].x) + klabs(yplc-wall[j].y);
|
|
else
|
|
{
|
|
screencoords(&dax,&day, wall[j].x-pos.x,wall[j].y-pos.y, zoom);
|
|
day += getscreenvdisp(getflorzofslope(i, wall[j].x,wall[j].y)-pos.z, zoom);
|
|
|
|
if (halfxdim16+dax < 0 || halfxdim16+dax >= xdim || midydim16+day < 0 || midydim16+day >= ydim)
|
|
continue;
|
|
|
|
dst = klabs(halfxdim16+dax-searchx) + klabs(midydim16+day-searchy);
|
|
}
|
|
|
|
if (dst <= dist)
|
|
{
|
|
// prefer white walls
|
|
if (dst<dist || closest==-1 || (wall[j].nextwall>=0)-(wall[closest].nextwall>=0) <= 0)
|
|
dist = dst, closest = j;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (zoom >= 256)
|
|
for (i=0; i<MAXSPRITES; i++)
|
|
if (sprite[i].statnum < MAXSTATUS)
|
|
{
|
|
if ((!m32_sideview || !alwaysshowgray) && sprite[i].sectnum >= 0)
|
|
YAX_SKIPSECTOR(sprite[i].sectnum);
|
|
|
|
if (!m32_sideview)
|
|
{
|
|
dst = klabs(xplc-sprite[i].x) + klabs(yplc-sprite[i].y);
|
|
}
|
|
else
|
|
{
|
|
screencoords(&dax,&day, sprite[i].x-pos.x,sprite[i].y-pos.y, zoom);
|
|
day += getscreenvdisp(sprite[i].z-pos.z, zoom);
|
|
|
|
if (halfxdim16+dax < 0 || halfxdim16+dax >= xdim || midydim16+day < 0 || midydim16+day >= ydim)
|
|
continue;
|
|
|
|
dst = klabs(halfxdim16+dax-searchx) + klabs(midydim16+day-searchy);
|
|
}
|
|
|
|
// was (dst <= dist), but this way, when duplicating sprites,
|
|
// the selected ones are dragged first
|
|
if (dst < dist || (dst == dist && (show2dsprite[i>>3]&(1<<(i&7)))))
|
|
dist = dst, closest = i+16384;
|
|
}
|
|
|
|
return closest;
|
|
}
|
|
|
|
static void locktogrid(int32_t *dax, int32_t *day)
|
|
{
|
|
*dax = ((*dax+(1024>>grid))&(0xffffffff<<(11-grid)));
|
|
*day = ((*day+(1024>>grid))&(0xffffffff<<(11-grid)));
|
|
}
|
|
|
|
static int32_t adjustmark(int32_t *xplc, int32_t *yplc, int16_t danumwalls)
|
|
{
|
|
int32_t i, dst, dist, dax, day, pointlockdist;
|
|
|
|
if (danumwalls < 0)
|
|
danumwalls = numwalls;
|
|
|
|
pointlockdist = 0;
|
|
if (grid > 0 && gridlock)
|
|
pointlockdist = (256>>grid);
|
|
|
|
dist = pointlockdist;
|
|
dax = *xplc;
|
|
day = *yplc;
|
|
|
|
for (i=0; i<danumwalls; i++)
|
|
{
|
|
YAX_SKIPWALL(i);
|
|
|
|
dst = klabs((*xplc)-wall[i].x) + klabs((*yplc)-wall[i].y);
|
|
if (dst < dist)
|
|
{
|
|
dist = dst;
|
|
dax = wall[i].x;
|
|
day = wall[i].y;
|
|
}
|
|
}
|
|
if (dist == pointlockdist)
|
|
if (gridlock && grid > 0)
|
|
locktogrid(&dax, &day);
|
|
|
|
*xplc = dax;
|
|
*yplc = day;
|
|
|
|
return(0);
|
|
}
|
|
|
|
static int32_t checkautoinsert(int32_t dax, int32_t day, int16_t danumwalls)
|
|
{
|
|
int32_t i, x1, y1, x2, y2;
|
|
|
|
if (danumwalls < 0)
|
|
danumwalls = numwalls;
|
|
|
|
for (i=0; i<danumwalls; i++) // Check if a point should be inserted
|
|
{
|
|
YAX_SKIPWALL(i);
|
|
|
|
x1 = wall[i].x;
|
|
y1 = wall[i].y;
|
|
x2 = POINT2(i).x;
|
|
y2 = POINT2(i).y;
|
|
|
|
if ((x1 != dax || y1 != day) && (x2 != dax || y2 != day))
|
|
if ((x1 <= dax && dax <= x2) || (x2 <= dax && dax <= x1))
|
|
if ((y1 <= day && day <= y2) || (y2 <= day && day <= y1))
|
|
if ((dax-x1)*(y2-y1) == (day-y1)*(x2-x1))
|
|
return 1; //insertpoint((short)i,dax,day,NULL);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// <wallstart> has to be the starting (i.e. least index) wall of a loop!
|
|
// Returns: CLOCKDIR_CW or CLOCKDIR_CCW.
|
|
int32_t clockdir(int32_t wallstart)
|
|
{
|
|
int32_t tempint, x0, x1, x2, y0, y1, y2;
|
|
|
|
int32_t minx = 0x7fffffff;
|
|
int32_t themin = -1;
|
|
int32_t i = wallstart-1;
|
|
|
|
do
|
|
{
|
|
i++;
|
|
if (POINT2(i).x < minx)
|
|
{
|
|
minx = POINT2(i).x;
|
|
themin = i;
|
|
}
|
|
}
|
|
while (wall[i].point2 != wallstart && i < MAXWALLS-1);
|
|
// NOTE: the i < MAXWALLS-1 check is really only safety against either
|
|
// - a really corrupt map, or
|
|
// - misuse of clockdir() where <wallstart> is not the starting wall of
|
|
// the very last loop
|
|
|
|
x0 = wall[themin].x;
|
|
y0 = wall[themin].y;
|
|
x1 = POINT2(themin).x;
|
|
y1 = POINT2(themin).y;
|
|
x2 = POINT2(wall[themin].point2).x;
|
|
y2 = POINT2(wall[themin].point2).y;
|
|
|
|
if (y2 <= y1 && y1 <= y0)
|
|
return CLOCKDIR_CW;
|
|
if (y0 <= y1 && y1 <= y2)
|
|
return CLOCKDIR_CCW;
|
|
|
|
tempint = (x0-x1)*(y2-y1) - (x2-x1)*(y0-y1);
|
|
if (tempint < 0)
|
|
return CLOCKDIR_CW;
|
|
else
|
|
return CLOCKDIR_CCW;
|
|
}
|
|
|
|
static void flipwalls(int16_t numwalls, int16_t newnumwalls)
|
|
{
|
|
int32_t i, j, nume;
|
|
|
|
nume = newnumwalls-numwalls;
|
|
|
|
for (i=numwalls; i<numwalls+(nume>>1); i++)
|
|
{
|
|
j = numwalls+newnumwalls-i-1;
|
|
|
|
swapshort(&onextwall[i], &onextwall[j]);
|
|
|
|
swaplong(&wall[i].x, &wall[j].x);
|
|
swaplong(&wall[i].y, &wall[j].y);
|
|
}
|
|
}
|
|
|
|
static void do_insertpoint(int32_t w, int32_t dax, int32_t day, int32_t *mapwallnum)
|
|
{
|
|
int32_t i;
|
|
const int32_t sucksect = sectorofwall(w);
|
|
const uint32_t lenbyrep = getlenbyrep(wallength(w), wall[w].xrepeat);
|
|
|
|
sector[sucksect].wallnum++;
|
|
for (i=sucksect+1; i<numsectors; i++)
|
|
sector[i].wallptr++;
|
|
|
|
if (mapwallnum && *mapwallnum >= w+1)
|
|
(*mapwallnum)++;
|
|
|
|
movewalls(w+1, +1);
|
|
Bmemcpy(&wall[w+1], &wall[w], sizeof(walltype));
|
|
#ifdef YAX_ENABLE
|
|
wall[w+1].cstat &= ~(1<<14);
|
|
#endif
|
|
wall[w].point2 = w+1;
|
|
wall[w+1].x = dax;
|
|
wall[w+1].y = day;
|
|
|
|
fixxrepeat(w, lenbyrep);
|
|
AlignWallPoint2(w);
|
|
fixxrepeat(w+1, lenbyrep);
|
|
}
|
|
|
|
// Returns number of points inserted (1; or 2 if wall had a nextwall).
|
|
// *mapwallnum is set to the new wallnum of the former (pre-insertpoint) *mapwallnum
|
|
// (the new one can only be >= than the old one; ptr may be NULL if we don't care)
|
|
static int32_t insertpoint(int16_t linehighlight, int32_t dax, int32_t day, int32_t *mapwallnum)
|
|
{
|
|
int32_t j = linehighlight;
|
|
|
|
do_insertpoint(j, dax, day, mapwallnum);
|
|
|
|
if (wall[j].nextwall >= 0)
|
|
{
|
|
int32_t k = wall[j].nextwall;
|
|
|
|
do_insertpoint(k, dax, day, mapwallnum);
|
|
|
|
j = wall[k].nextwall;
|
|
wall[j].nextwall = k+1;
|
|
wall[j+1].nextwall = k;
|
|
wall[k].nextwall = j+1;
|
|
wall[k+1].nextwall = j;
|
|
|
|
return 2;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
// runi: 0=check (forbidden), 1=prepare, 2=do!
|
|
static void deletepoint(int16_t point, int32_t runi)
|
|
{
|
|
int32_t i, j, sucksect;
|
|
|
|
Bassert(runi != 0);
|
|
|
|
if (runi==1)
|
|
{
|
|
i = wall[point].nextwall;
|
|
if (i >= 0)
|
|
{
|
|
NEXTWALL(i).nextwall = NEXTWALL(i).nextsector = -1;
|
|
wall[i].nextwall = wall[i].nextsector = -1;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
sucksect = sectorofwall(point);
|
|
|
|
sector[sucksect].wallnum--;
|
|
for (i=sucksect+1; i<numsectors; i++)
|
|
sector[i].wallptr--;
|
|
|
|
j = lastwall(point);
|
|
wall[j].point2 = wall[point].point2;
|
|
|
|
#if 0
|
|
if (wall[j].nextwall >= 0)
|
|
{
|
|
NEXTWALL(j).nextwall = -1;
|
|
NEXTWALL(j).nextsector = -1;
|
|
}
|
|
if (wall[point].nextwall >= 0)
|
|
{
|
|
NEXTWALL(point).nextwall = -1;
|
|
NEXTWALL(point).nextsector = -1;
|
|
}
|
|
#endif
|
|
movewalls(point, -1);
|
|
|
|
// checksectorpointer(j, sucksect);
|
|
|
|
return;
|
|
}
|
|
|
|
static int32_t deletesector(int16_t sucksect)
|
|
{
|
|
int32_t i, j, k, nextk, startwall, endwall;
|
|
|
|
while (headspritesect[sucksect] >= 0)
|
|
deletesprite(headspritesect[sucksect]);
|
|
|
|
startwall = sector[sucksect].wallptr;
|
|
endwall = startwall + sector[sucksect].wallnum - 1;
|
|
j = sector[sucksect].wallnum;
|
|
|
|
#ifdef YAX_ENABLE
|
|
yax_setbunches(sucksect, -1, -1);
|
|
#endif
|
|
|
|
for (i=sucksect; i<numsectors-1; i++)
|
|
{
|
|
k = headspritesect[i+1];
|
|
while (k != -1)
|
|
{
|
|
nextk = nextspritesect[k];
|
|
changespritesect(k, i);
|
|
k = nextk;
|
|
}
|
|
|
|
Bmemcpy(§or[i], §or[i+1], sizeof(sectortype));
|
|
sector[i].wallptr -= j;
|
|
}
|
|
numsectors--;
|
|
|
|
for (i=startwall; i<=endwall; i++)
|
|
{
|
|
if (wall[i].nextwall >= 0)
|
|
{
|
|
NEXTWALL(i).nextwall = -1;
|
|
NEXTWALL(i).nextsector = -1;
|
|
}
|
|
}
|
|
|
|
movewalls(startwall, -j);
|
|
for (i=0; i<numwalls; i++)
|
|
if (wall[i].nextwall >= startwall)
|
|
wall[i].nextsector--;
|
|
|
|
return(0);
|
|
}
|
|
|
|
int32_t fixspritesectors(void)
|
|
{
|
|
int32_t i;
|
|
int32_t numfixedsprites = 0, printfirsttime = 0;
|
|
|
|
for (i=numsectors-1; i>=0; i--)
|
|
if (sector[i].wallnum <= 0 || sector[i].wallptr >= numwalls)
|
|
{
|
|
// XXX: This is not the best course of action for
|
|
// such great corruption.
|
|
deletesector(i);
|
|
mkonwinvalid();
|
|
initprintf("NOTE: Deleted sector %d which had corrupt .wallnum or .wallptr\n", i);
|
|
}
|
|
|
|
if (m32_script_expertmode)
|
|
return 0;
|
|
|
|
for (i=0; i<MAXSPRITES; i++)
|
|
if (sprite[i].statnum < MAXSTATUS)
|
|
{
|
|
const int32_t dax=sprite[i].x, day=sprite[i].y;
|
|
|
|
if (inside(dax,day,sprite[i].sectnum) != 1)
|
|
{
|
|
int32_t j, cz, fz;
|
|
|
|
spriteoncfz(i, &cz, &fz);
|
|
|
|
for (j=0; j<numsectors; j++)
|
|
{
|
|
if (cz <= sprite[i].z && sprite[i].z <= fz && inside(dax,day, j) == 1)
|
|
{
|
|
if (fixmaponsave_sprites || (unsigned)sprite[i].sectnum >= numsectors+0u)
|
|
{
|
|
if (printfirsttime == 0)
|
|
{
|
|
initprintf("--------------------\n");
|
|
printfirsttime = 1;
|
|
}
|
|
initprintf("Changed sectnum of sprite #%d from %d to %d\n",
|
|
i, TrackerCast(sprite[i].sectnum), j);
|
|
|
|
changespritesect(i, j);
|
|
numfixedsprites++;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return numfixedsprites;
|
|
}
|
|
|
|
static int32_t movewalls(int32_t start, int32_t offs)
|
|
{
|
|
int32_t i;
|
|
|
|
if (offs < 0) //Delete
|
|
{
|
|
for (i=start; i<numwalls+offs; i++)
|
|
Bmemcpy(&wall[i], &wall[i-offs], sizeof(walltype));
|
|
}
|
|
else if (offs > 0) //Insert
|
|
{
|
|
for (i=numwalls+offs-1; i>=start+offs; i--)
|
|
Bmemcpy(&wall[i], &wall[i-offs], sizeof(walltype));
|
|
|
|
if (ovh.bak_wallsdrawn > 0)
|
|
{
|
|
if (ovh.suckwall >= start)
|
|
ovh.suckwall += offs;
|
|
if (ovh.splitstartwall >= start)
|
|
ovh.splitstartwall += offs;
|
|
}
|
|
}
|
|
|
|
numwalls += offs;
|
|
for (i=0; i<numwalls; i++)
|
|
{
|
|
if (wall[i].nextwall >= start) wall[i].nextwall += offs;
|
|
if (wall[i].point2 >= start) wall[i].point2 += offs;
|
|
}
|
|
#ifdef YAX_ENABLE
|
|
yax_tweakwalls(start, offs);
|
|
#endif
|
|
|
|
return(0);
|
|
}
|
|
|
|
int32_t wallength(int16_t i)
|
|
{
|
|
int64_t dax = POINT2(i).x - wall[i].x;
|
|
int64_t day = POINT2(i).y - wall[i].y;
|
|
#if 1 //def POLYMOST
|
|
int64_t hypsq = dax*dax + day*day;
|
|
if (hypsq > (int64_t)INT32_MAX)
|
|
return (int32_t)sqrt((double)hypsq);
|
|
else
|
|
return ksqrt((uint32_t)hypsq);
|
|
#else
|
|
return ksqrt(dax*dax + day*day);
|
|
#endif
|
|
}
|
|
|
|
void fixrepeats(int16_t i)
|
|
{
|
|
int32_t dist = wallength(i);
|
|
int32_t day = wall[i].yrepeat;
|
|
|
|
wall[i].xrepeat = clamp(mulscale10(dist,day), 1, 255);
|
|
}
|
|
|
|
uint32_t getlenbyrep(int32_t len, int32_t repeat)
|
|
{
|
|
if (repeat <= 0)
|
|
return ((uint32_t)len)<<12;
|
|
|
|
return divscale12(len, repeat);
|
|
}
|
|
|
|
void fixxrepeat(int16_t wallnum, uint32_t lenrepquot) // lenrepquot: divscale12(wallength,xrepeat)
|
|
{
|
|
if (lenrepquot != 0)
|
|
{
|
|
uint32_t res = (((wallength(wallnum)<<12)+(1<<11))/lenrepquot);
|
|
wall[wallnum].xrepeat = clamp(res, 1, 255);
|
|
}
|
|
}
|
|
|
|
|
|
int32_t overridepm16y = -1;
|
|
|
|
void clearmidstatbar16(void)
|
|
{
|
|
int32_t y = overridepm16y<0 ? STATUS2DSIZ : overridepm16y;
|
|
|
|
begindrawing();
|
|
CLEARLINES2D(ydim-y+25, STATUS2DSIZ+2-(25<<1), 0);
|
|
enddrawing();
|
|
}
|
|
|
|
static void clearministatbar16(void)
|
|
{
|
|
int32_t i, col = whitecol - 21;
|
|
|
|
begindrawing();
|
|
|
|
for (i=ydim-STATUS2DSIZ2; i<ydim; i++)
|
|
{
|
|
// drawline256(0, i<<12, xdim<<12, i<<12, col);
|
|
CLEARLINES2D(i, 1, (col<<24)|(col<<16)|(col<<8)|col);
|
|
|
|
col--;
|
|
if (col <= 0) break;
|
|
}
|
|
|
|
CLEARLINES2D(i, ydim-i, 0);
|
|
|
|
if (xdim >= 800)
|
|
{
|
|
Bsnprintf(tempbuf, sizeof(tempbuf), "%s %s", AppProperName, CallExtGetVer());
|
|
printext16(xdim2d-(Bstrlen(tempbuf)<<3)-3, ydim2d-STATUS2DSIZ2+10, editorcolors[4],-1, tempbuf, 0);
|
|
printext16(xdim2d-(Bstrlen(tempbuf)<<3)-2, ydim2d-STATUS2DSIZ2+9, editorcolors[12],-1, tempbuf, 0);
|
|
}
|
|
|
|
enddrawing();
|
|
}
|
|
|
|
// <startwall> has to be the starting wall of a loop!
|
|
//
|
|
// Assuming that <startwall> indicates a CW (outer) loop, the return value can
|
|
// be seen as a boolean of whether (x,y) is inside it.
|
|
//
|
|
// XXX: this function suffers from asymmetry issues in degenerate cases,
|
|
// similar to how inside() did before r3898.
|
|
int32_t loopinside(int32_t x, int32_t y, int16_t startwall)
|
|
{
|
|
int32_t cnt = clockdir(startwall);
|
|
int32_t i = startwall;
|
|
|
|
do
|
|
{
|
|
int32_t x1 = wall[i].x;
|
|
int32_t x2 = POINT2(i).x;
|
|
|
|
if (x <= x1 || x <= x2)
|
|
{
|
|
int32_t y1 = wall[i].y;
|
|
int32_t y2 = POINT2(i).y;
|
|
|
|
if (y1 > y2)
|
|
{
|
|
swaplong(&x1, &x2);
|
|
swaplong(&y1, &y2);
|
|
}
|
|
|
|
if (y1 <= y && y < y2)
|
|
if ((uint64_t)x1*(y-y2) + (uint64_t)x2*(y1-y) <= (uint64_t)x*(y1-y2))
|
|
cnt ^= 1;
|
|
}
|
|
|
|
i = wall[i].point2;
|
|
}
|
|
while (i != startwall);
|
|
|
|
return cnt;
|
|
}
|
|
|
|
#if 0
|
|
static int32_t numloopsofsector(int16_t sectnum)
|
|
{
|
|
int32_t i, numloops, startwall, endwall;
|
|
|
|
numloops = 0;
|
|
startwall = sector[sectnum].wallptr;
|
|
endwall = startwall + sector[sectnum].wallnum;
|
|
for (i=startwall; i<endwall; i++)
|
|
if (wall[i].point2 < i) numloops++;
|
|
return(numloops);
|
|
}
|
|
#endif
|
|
|
|
int32_t getnumber_internal1(char ch, int32_t *danumptr, int32_t maxnumber, char sign)
|
|
{
|
|
int32_t danum = *danumptr;
|
|
|
|
if (ch >= '0' && ch <= '9')
|
|
{
|
|
int64_t nbig;
|
|
if (danum >= 0)
|
|
{
|
|
nbig = ((int64_t)danum*10)+(ch-'0');
|
|
if (nbig <= (int64_t)maxnumber) danum = nbig;
|
|
}
|
|
else if (sign) // this extra check isn't hurting anything
|
|
{
|
|
nbig = ((int64_t)danum*10)-(ch-'0');
|
|
if (nbig >= (int64_t)(-maxnumber)) danum = nbig;
|
|
}
|
|
}
|
|
else if (ch == 8 || ch == 127) // backspace
|
|
{
|
|
danum /= 10;
|
|
}
|
|
else if (ch == 13)
|
|
{
|
|
return 1;
|
|
}
|
|
else if (ch == '-' && sign) // negate
|
|
{
|
|
danum = -danum;
|
|
}
|
|
|
|
*danumptr = danum;
|
|
return 0;
|
|
}
|
|
|
|
int32_t getnumber_autocomplete(const char *namestart, char ch, int32_t *danum, int32_t flags)
|
|
{
|
|
if (flags!=1 && flags!=2)
|
|
return 0;
|
|
|
|
if (flags==2 && *danum<0)
|
|
return 0;
|
|
|
|
if (isalpha(ch))
|
|
{
|
|
char b[2];
|
|
const char *gotstr;
|
|
int32_t i, diddel;
|
|
|
|
b[0] = ch;
|
|
b[1] = 0;
|
|
gotstr = getstring_simple(namestart, b, (flags==1)?sizeof(names[0])-1:TAGLAB_MAX-1, flags);
|
|
if (!gotstr || !gotstr[0])
|
|
return 0;
|
|
|
|
if (flags==1)
|
|
{
|
|
for (i=0; i<MAXTILES; i++)
|
|
if (!Bstrcasecmp(names[i], gotstr))
|
|
{
|
|
*danum = i;
|
|
return 1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
i = taglab_gettag(gotstr);
|
|
//initprintf("taglab: s=%s, i=%d\n",gotstr,i);
|
|
if (i > 0)
|
|
{
|
|
*danum = i;
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
// insert new tag
|
|
if (*danum > 0 && *danum<32768)
|
|
{
|
|
diddel = taglab_add(gotstr, *danum);
|
|
message("Added label \"%s\" for tag %d%s%s", gotstr, *danum,
|
|
diddel?", deleting old ":"",
|
|
(!diddel)?"":(diddel==1?"label":"tag"));
|
|
return 1;
|
|
}
|
|
else if (*danum==0)
|
|
{
|
|
i = taglab_getnextfreetag(NULL);
|
|
if (i >= 1)
|
|
{
|
|
*danum = i;
|
|
diddel = taglab_add(gotstr, *danum);
|
|
message("%sadded label \"%s\" for tag %d%s%s",
|
|
diddel?"Auto-":"Automatically ", gotstr, *danum,
|
|
diddel?", deleting old ":"",
|
|
(!diddel)?"":(diddel==1?"label":"tag"));
|
|
return 1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// sign is now used for more than one flag (also _getnumber256):
|
|
// 1: sign
|
|
// 2: autocomplete names
|
|
// 4: autocomplete taglabels
|
|
// 8: return -1 if cancelled
|
|
int32_t _getnumber16(const char *namestart, int32_t num, int32_t maxnumber, char sign, const char *(func)(int32_t))
|
|
{
|
|
char buffer[80], ournamestart[80-17], ch;
|
|
int32_t n, danum, oldnum;
|
|
uint8_t flags = (sign&(2|4|8))>>1;
|
|
sign &= 1;
|
|
|
|
danum = num;
|
|
oldnum = danum;
|
|
|
|
// need to have 4+11+2==17 chars room at the end
|
|
// ("^011", max. string length of an int32, "_ ")
|
|
Bstrncpyz(ournamestart, namestart, sizeof(ournamestart));
|
|
|
|
bflushchars();
|
|
while (keystatus[0x1] == 0)
|
|
{
|
|
if (handleevents())
|
|
quitevent = 0;
|
|
|
|
idle();
|
|
ch = bgetchar();
|
|
|
|
Bsprintf(buffer, "%s^011%d", ournamestart, danum);
|
|
n = Bstrlen(buffer); // maximum is 62+4+11 == 77
|
|
if (totalclock & 32)
|
|
Bstrcat(buffer,"_ ");
|
|
// max strlen now 79
|
|
_printmessage16("%s", buffer);
|
|
|
|
if (func != NULL)
|
|
{
|
|
Bsnprintf(buffer, sizeof(buffer), "%s", func(danum));
|
|
// printext16(200L-24, ydim-STATUS2DSIZ+20L, editorcolors[9], editorcolors[0], buffer, 0);
|
|
printext16(n<<3, ydim-STATUS2DSIZ+128, editorcolors[11], -1, buffer,0);
|
|
}
|
|
|
|
showframe(1);
|
|
|
|
n = 0;
|
|
if (getnumber_internal1(ch, &danum, maxnumber, sign) ||
|
|
(n=getnumber_autocomplete(ournamestart, ch, &danum, flags&(1+2))))
|
|
{
|
|
if (flags==1 || n==0)
|
|
printmessage16("%s", buffer);
|
|
if (danum != oldnum)
|
|
asksave = 1;
|
|
oldnum = danum;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (keystatus[0x1] && (flags&4))
|
|
oldnum = -1;
|
|
|
|
clearkeys();
|
|
|
|
return oldnum;
|
|
}
|
|
|
|
static void getnumber_clearline(void)
|
|
{
|
|
char cbuf[128];
|
|
int32_t i;
|
|
for (i=0; i<min(xdim>>3, (signed)sizeof(cbuf)-1); i++)
|
|
cbuf[i] = ' ';
|
|
cbuf[i] = 0;
|
|
printext256(0, 0, whitecol, 0, cbuf, 0);
|
|
}
|
|
|
|
// sign: |16: don't draw scene
|
|
int32_t _getnumber256(const char *namestart, int32_t num, int32_t maxnumber, char sign, const char *(func)(int32_t))
|
|
{
|
|
char buffer[80], ournamestart[80-13], ch;
|
|
int32_t danum, oldnum;
|
|
uint8_t flags = (sign&(2|4|8|16))>>1;
|
|
sign &= 1;
|
|
|
|
danum = num;
|
|
oldnum = danum;
|
|
|
|
// need to have 11+2==13 chars room at the end
|
|
// (max. string length of an int32, "_ ")
|
|
Bstrncpyz(ournamestart, namestart, sizeof(ournamestart));
|
|
|
|
bflushchars();
|
|
while (keystatus[0x1] == 0)
|
|
{
|
|
if (handleevents())
|
|
quitevent = 0;
|
|
|
|
if ((flags&8)==0)
|
|
M32_DrawRoomsAndMasks();
|
|
|
|
ch = bgetchar();
|
|
if (keystatus[0x1])
|
|
break;
|
|
clearkeys();
|
|
|
|
mouseb = 0;
|
|
searchx = osearchx;
|
|
searchy = osearchy;
|
|
|
|
inputchecked = 1;
|
|
|
|
if ((flags&8)==0)
|
|
CallExtCheckKeys();
|
|
|
|
getnumber_clearline();
|
|
|
|
Bsprintf(buffer,"%s%d",ournamestart,danum);
|
|
// max strlen now 66+11==77
|
|
if (totalclock & 32)
|
|
Bstrcat(buffer,"_ ");
|
|
// max strlen now 79
|
|
printmessage256(0, 0, buffer);
|
|
if (func != NULL)
|
|
{
|
|
Bsnprintf(buffer, sizeof(buffer), "%s", func(danum));
|
|
printmessage256(0, 9, buffer);
|
|
}
|
|
|
|
showframe(1);
|
|
|
|
if (getnumber_internal1(ch, &danum, maxnumber, sign) ||
|
|
getnumber_autocomplete(ournamestart, ch, &danum, flags&(1+2)))
|
|
{
|
|
if (danum != oldnum)
|
|
asksave = 1;
|
|
oldnum = danum;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (keystatus[0x1] && (flags&4))
|
|
oldnum = -1;
|
|
|
|
clearkeys();
|
|
|
|
lockclock = totalclock; //Reset timing
|
|
|
|
return oldnum;
|
|
}
|
|
|
|
// querystr: e.g. "Name: ", must be !=NULL
|
|
// defaultstr: can be NULL
|
|
// NO overflow checks are done when copying them!
|
|
// maxlen: maximum length of entry string, if ==1, enter single char
|
|
// completion: 0=none, 1=names[][], 2=taglabels
|
|
const char *getstring_simple(const char *querystr, const char *defaultstr, int32_t maxlen, int32_t completion)
|
|
{
|
|
static char buf[128];
|
|
int32_t ei=0, qrylen=0, maxidx, havecompl=0;
|
|
char ch;
|
|
|
|
bflushchars();
|
|
clearkeys();
|
|
|
|
Bmemset(buf, 0, sizeof(buf));
|
|
|
|
qrylen = Bstrlen(querystr);
|
|
Bmemcpy(buf, querystr, qrylen);
|
|
|
|
if (maxlen==0)
|
|
maxlen = 64;
|
|
|
|
maxidx = min((signed)sizeof(buf), xdim>>3);
|
|
|
|
ei = qrylen;
|
|
|
|
if (defaultstr)
|
|
{
|
|
int32_t deflen = Bstrlen(defaultstr);
|
|
Bmemcpy(&buf[ei], defaultstr, deflen);
|
|
ei += deflen;
|
|
}
|
|
|
|
buf[ei] = '_';
|
|
buf[ei+1] = 0;
|
|
|
|
while (1)
|
|
{
|
|
if (in3dmode())
|
|
getnumber_clearline();
|
|
|
|
if (in3dmode())
|
|
printext256(0, 0, whitecol, 0, buf, 0);
|
|
else
|
|
_printmessage16("%s", buf);
|
|
|
|
showframe(1);
|
|
|
|
if (handleevents())
|
|
quitevent = 0;
|
|
|
|
idle();
|
|
ch = bgetchar();
|
|
|
|
if (ch==13)
|
|
{
|
|
if (maxlen != 1)
|
|
buf[ei] = 0;
|
|
break;
|
|
}
|
|
else if (keystatus[1])
|
|
{
|
|
clearkeys();
|
|
return completion ? NULL : defaultstr;
|
|
}
|
|
|
|
if (maxlen!=1)
|
|
{
|
|
// blink...
|
|
if (totalclock&32)
|
|
{
|
|
buf[ei] = '_';
|
|
buf[ei+1] = 0;
|
|
}
|
|
else
|
|
{
|
|
buf[ei] = 0;
|
|
}
|
|
|
|
if (ei>qrylen && (ch==8 || ch==127))
|
|
{
|
|
buf[ei--] = 0;
|
|
buf[ei] = '_';
|
|
havecompl = 0;
|
|
}
|
|
else if (ei<maxidx-2 && ei-qrylen<maxlen && isprint(ch))
|
|
{
|
|
if (completion==2 && ch==' ')
|
|
ch = '_';
|
|
buf[ei++] = ch;
|
|
buf[ei] = '_';
|
|
buf[ei+1] = 0;
|
|
}
|
|
|
|
if (completion && ((ei>qrylen && ch==9) || havecompl)) // tab: maybe do auto-completion
|
|
{
|
|
char cmpbuf[128];
|
|
char completions[3][16];
|
|
const char *cmpstr;
|
|
int32_t len=ei-qrylen, i, j, k=len, first=1, numcompl=0;
|
|
|
|
Bmemcpy(cmpbuf, &buf[qrylen], len);
|
|
cmpbuf[len] = 0;
|
|
|
|
for (i=(completion!=1); i<((completion==1)?MAXTILES:32768); i++)
|
|
{
|
|
cmpstr = (completion==1) ? names[i] : taglab_getlabel(i);
|
|
if (!cmpstr)
|
|
continue;
|
|
|
|
if (Bstrncasecmp(cmpbuf, cmpstr, len) || Bstrlen(cmpstr)==(unsigned)len) // compare the prefix
|
|
continue;
|
|
|
|
if (ch==9)
|
|
{
|
|
if (first)
|
|
{
|
|
Bstrncpy(cmpbuf+len, cmpstr+len, sizeof(cmpbuf)-len);
|
|
cmpbuf[sizeof(cmpbuf)-1] = 0;
|
|
first = 0;
|
|
}
|
|
else
|
|
{
|
|
for (k=len; cmpstr[k] && cmpbuf[k] && Btolower(cmpstr[k])==Btolower(cmpbuf[k]); k++)
|
|
/* nop */;
|
|
cmpbuf[k] = 0;
|
|
}
|
|
}
|
|
|
|
if (numcompl<3)
|
|
{
|
|
Bstrncpyz(completions[numcompl], cmpstr+len, sizeof(completions[0]));
|
|
|
|
for (k=0; completions[numcompl][k]; k++)
|
|
completions[numcompl][k] = Btolower(completions[numcompl][k]);
|
|
numcompl++;
|
|
}
|
|
|
|
for (k=len; cmpbuf[k]; k++)
|
|
cmpbuf[k] = Btolower(cmpbuf[k]);
|
|
}
|
|
|
|
ei = qrylen;
|
|
for (i=0; i<k && i<maxlen && ei<maxidx-2; i++)
|
|
buf[ei++] = cmpbuf[i];
|
|
|
|
if (k==len && numcompl>0) // no chars autocompleted/completion request
|
|
{
|
|
buf[ei] = '{';
|
|
buf[ei+1] = 0;
|
|
|
|
i = ei+1;
|
|
for (k=0; k<numcompl; k++)
|
|
{
|
|
j = 0;
|
|
while (i<maxidx-1 && completions[k][j])
|
|
buf[i++] = completions[k][j++];
|
|
if (i<maxidx-1)
|
|
buf[i++] = (k==numcompl-1) ? '}' : ',';
|
|
}
|
|
buf[i] = 0;
|
|
|
|
havecompl = 1;
|
|
}
|
|
else
|
|
{
|
|
buf[ei] = '_';
|
|
buf[ei+1] = 0;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (isalnum(ch) || ch==' ')
|
|
buf[ei] = Btoupper(ch);
|
|
}
|
|
}
|
|
|
|
clearkeys();
|
|
|
|
return buf+qrylen;
|
|
}
|
|
|
|
static int32_t getfilenames(const char *path, const char *kind)
|
|
{
|
|
const int32_t addflags = (!pathsearchmode && grponlymode ? CACHE1D_OPT_NOSTACK : 0);
|
|
|
|
fnlist_getnames(&fnlist, path, kind, addflags|CACHE1D_FIND_DRIVE, addflags);
|
|
|
|
finddirshigh = fnlist.finddirs;
|
|
findfileshigh = fnlist.findfiles;
|
|
currentlist = (findfileshigh != NULL);
|
|
|
|
return(0);
|
|
}
|
|
|
|
static void tweak_sboardfilename(void)
|
|
{
|
|
if (pathsearchmode)
|
|
Bcanonicalisefilename(selectedboardfilename, 1); // clips off the last token and compresses relative path
|
|
else
|
|
Bcorrectfilename(selectedboardfilename, 1);
|
|
}
|
|
|
|
////////// FILE SELECTION MENU //////////
|
|
|
|
static char g_oldpath[BMAX_PATH];
|
|
|
|
static void menuselect_try_findlast(void)
|
|
{
|
|
// PK 20080103: start with last selected map
|
|
const char *boardbasename = getbasefn(boardfilename);
|
|
|
|
for (; findfileshigh; findfileshigh=findfileshigh->next)
|
|
if (!Bstrcmp(findfileshigh->name, boardbasename))
|
|
break;
|
|
|
|
if (!findfileshigh)
|
|
findfileshigh = fnlist.findfiles;
|
|
}
|
|
|
|
// vvv PK ------------------------------------
|
|
// copied off menuselect
|
|
|
|
static int32_t menuselect_auto(int32_t direction) // 20080104: jump to next (direction!=0) or prev (direction==0) file
|
|
{
|
|
Bstrcpy(selectedboardfilename, g_oldpath);
|
|
tweak_sboardfilename();
|
|
|
|
getfilenames(selectedboardfilename, "*.map");
|
|
if (fnlist.numfiles==0)
|
|
return -2;
|
|
|
|
menuselect_try_findlast();
|
|
|
|
if (direction)
|
|
{
|
|
if (findfileshigh->next)
|
|
findfileshigh=findfileshigh->next;
|
|
else
|
|
return -1;
|
|
}
|
|
else
|
|
{
|
|
if (findfileshigh->prev)
|
|
findfileshigh=findfileshigh->prev;
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
Bstrcat(selectedboardfilename, findfileshigh->name);
|
|
|
|
return 0;
|
|
}
|
|
// ^^^ PK ------------------------------------
|
|
|
|
static int32_t menuselect(void)
|
|
{
|
|
int32_t listsize;
|
|
int32_t i;
|
|
char ch, buffer[96];
|
|
const int32_t bakpathsearchmode = pathsearchmode;
|
|
|
|
Bstrcpy(selectedboardfilename, g_oldpath);
|
|
tweak_sboardfilename();
|
|
|
|
getfilenames(selectedboardfilename, "*.map");
|
|
|
|
menuselect_try_findlast();
|
|
|
|
_printmessage16("Select map file with arrow keys and enter.");
|
|
|
|
ydim16 = ydim-STATUS2DSIZ2;
|
|
listsize = (ydim16-32)/9;
|
|
|
|
do
|
|
{
|
|
begindrawing(); //{{{
|
|
|
|
CLEARLINES2D(0, ydim16, 0);
|
|
|
|
if (pathsearchmode)
|
|
Bstrcpy(buffer,"Local filesystem mode. Ctrl-F: game filesystem");
|
|
else
|
|
Bsnprintf(buffer, sizeof(buffer),
|
|
"Game filesystem %smode. Ctrl-F: local filesystem, Ctrl-G: %s",
|
|
grponlymode?"GRP-only ":"", grponlymode?"all files":"GRP contents only");
|
|
|
|
printext16(halfxdim16-(8*Bstrlen(buffer)/2), 4, editorcolors[12],editorcolors[0],buffer,0);
|
|
|
|
Bsnprintf(buffer, sizeof(buffer), "(%d dirs, %d files) %s",
|
|
fnlist.numdirs, fnlist.numfiles, selectedboardfilename);
|
|
|
|
printext16(8,ydim16-8-1,editorcolors[8],editorcolors[0],buffer,0);
|
|
|
|
if (finddirshigh)
|
|
{
|
|
const CACHE1D_FIND_REC *dir = finddirshigh;
|
|
|
|
for (i=(listsize/2)-1; i>=0; i--)
|
|
{
|
|
if (!dir->prev) break;
|
|
else dir=dir->prev;
|
|
}
|
|
for (i=0; ((i<listsize) && dir); i++, dir=dir->next)
|
|
{
|
|
int32_t c = (dir->type == CACHE1D_FIND_DIR ? 2 : 3); //PK
|
|
Bmemset(buffer,0,sizeof(buffer));
|
|
Bstrncpy(buffer,dir->name,25);
|
|
if (Bstrlen(buffer) == 25)
|
|
buffer[21] = buffer[22] = buffer[23] = '.', buffer[24] = 0;
|
|
if (dir == finddirshigh)
|
|
{
|
|
if (currentlist == 0) printext16(8,16+9*i,editorcolors[c|8],editorcolors[0],"->",0);
|
|
printext16(32,16+9*i,editorcolors[c|8],editorcolors[0],buffer,0);
|
|
}
|
|
else
|
|
{
|
|
printext16(32,16+9*i,editorcolors[c],editorcolors[0],buffer,0);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (findfileshigh)
|
|
{
|
|
const CACHE1D_FIND_REC *dir = findfileshigh;
|
|
|
|
for (i=(listsize/2)-1; i>=0; i--)
|
|
{
|
|
if (!dir->prev) break;
|
|
else dir=dir->prev;
|
|
}
|
|
for (i=0; (i<listsize && dir); i++, dir=dir->next)
|
|
{
|
|
if (dir == findfileshigh)
|
|
{
|
|
if (currentlist == 1) printext16(240,16+9*i,editorcolors[7|8],editorcolors[0],"->",0);
|
|
printext16(240+24,16+9*i,editorcolors[7|8],editorcolors[0],dir->name,0);
|
|
}
|
|
else
|
|
{
|
|
printext16(240+24,16+9*i,editorcolors[7],editorcolors[0],dir->name,0);
|
|
}
|
|
}
|
|
}
|
|
|
|
enddrawing(); //}}}
|
|
showframe(1);
|
|
|
|
keystatus[0xcb] = 0;
|
|
keystatus[0xcd] = 0;
|
|
keystatus[0xc8] = 0;
|
|
keystatus[0xd0] = 0;
|
|
keystatus[0x1c] = 0; //enter
|
|
keystatus[0x0e] = 0; //backspace
|
|
keystatus[0xf] = 0; //tab
|
|
keystatus[1] = 0; //esc
|
|
ch = 0; //Interesting fakery of ch = getch()
|
|
while (ch == 0)
|
|
{
|
|
if (handleevents())
|
|
if (quitevent)
|
|
{
|
|
keystatus[1] = 1;
|
|
quitevent = 0;
|
|
}
|
|
|
|
idle();
|
|
|
|
ch = bgetchar();
|
|
|
|
{
|
|
// JBF 20040208: seek to first name matching pressed character
|
|
CACHE1D_FIND_REC *seeker = currentlist ? fnlist.findfiles : fnlist.finddirs;
|
|
if (keystatus[0xc7]||keystatus[0xcf]) // home/end
|
|
{
|
|
while (keystatus[0xcf]?seeker->next:seeker->prev)
|
|
seeker = keystatus[0xcf]?seeker->next:seeker->prev;
|
|
if (seeker)
|
|
{
|
|
if (currentlist) findfileshigh = seeker;
|
|
else finddirshigh = seeker;
|
|
}
|
|
ch = keystatus[0xcf]?80:72;
|
|
keystatus[0xc7] = keystatus[0xcf] = 0;
|
|
}
|
|
else if (keystatus[0xc9]|keystatus[0xd1]) // page up/down
|
|
{
|
|
seeker = currentlist?findfileshigh:finddirshigh;
|
|
i = (ydim2d-STATUS2DSIZ2-48)>>5/*3*/; //PK
|
|
|
|
while (i>0)
|
|
{
|
|
if (keystatus[0xd1]?seeker->next:seeker->prev)
|
|
seeker = keystatus[0xd1]?seeker->next:seeker->prev;
|
|
i--;
|
|
}
|
|
if (seeker)
|
|
{
|
|
if (currentlist) findfileshigh = seeker;
|
|
else finddirshigh = seeker;
|
|
}
|
|
ch = keystatus[0xd1]?80:72;
|
|
keystatus[0xc9] = keystatus[0xd1] = 0;
|
|
}
|
|
else
|
|
{
|
|
char ch2;
|
|
if (ch > 0 && ((ch >= 'a' && ch <= 'z') || (ch >= 'A' && ch <= 'Z') ||
|
|
(ch >= '0' && ch <= '9') || (ch=='_')))
|
|
{
|
|
#ifdef _WIN32
|
|
if (ch >= 'a') ch -= ('a'-'A');
|
|
#endif
|
|
while (seeker)
|
|
{
|
|
ch2 = seeker->name[0];
|
|
#ifdef _WIN32
|
|
if (ch2 >= 'a' && ch2 <= 'z') ch2 -= ('a'-'A');
|
|
#endif
|
|
if (ch2 == ch) break;
|
|
seeker = seeker->next;
|
|
}
|
|
if (seeker)
|
|
{
|
|
if (currentlist) findfileshigh = seeker;
|
|
else finddirshigh = seeker;
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (keystatus[0xcb]) ch = 9; // left arr
|
|
if (keystatus[0xcd]) ch = 9; // right arr
|
|
if (keystatus[0xc8]) ch = 72; // up arr
|
|
if (keystatus[0xd0]) ch = 80; // down arr
|
|
}
|
|
|
|
if (ch==6) // Ctrl-F
|
|
{
|
|
currentlist = 0;
|
|
pathsearchmode = 1-pathsearchmode;
|
|
if (pathsearchmode)
|
|
{
|
|
Bstrcpy(selectedboardfilename, "");
|
|
Bcanonicalisefilename(selectedboardfilename, 0);
|
|
}
|
|
else Bstrcpy(selectedboardfilename, "/");
|
|
|
|
getfilenames(selectedboardfilename, "*.map");
|
|
Bstrcpy(g_oldpath, selectedboardfilename);
|
|
}
|
|
else if (ch==7) // Ctrl-G
|
|
{
|
|
if (!pathsearchmode)
|
|
{
|
|
grponlymode = 1-grponlymode;
|
|
getfilenames(selectedboardfilename, "*.map");
|
|
Bstrcpy(g_oldpath, selectedboardfilename);
|
|
}
|
|
}
|
|
else if (ch == 9)
|
|
{
|
|
if ((currentlist == 0 && fnlist.findfiles) || (currentlist == 1 && fnlist.finddirs))
|
|
currentlist = 1-currentlist;
|
|
}
|
|
else if (keystatus[0xc8] /*(ch == 75) || (ch == 72)*/)
|
|
{
|
|
if (currentlist == 0)
|
|
{
|
|
if (finddirshigh && finddirshigh->prev)
|
|
finddirshigh = finddirshigh->prev;
|
|
}
|
|
else
|
|
{
|
|
if (findfileshigh && findfileshigh->prev)
|
|
findfileshigh = findfileshigh->prev;
|
|
}
|
|
}
|
|
else if (keystatus[0xd0] /*(ch == 77) || (ch == 80)*/)
|
|
{
|
|
if (currentlist == 0)
|
|
{
|
|
if (finddirshigh && finddirshigh->next)
|
|
finddirshigh = finddirshigh->next;
|
|
}
|
|
else
|
|
{
|
|
if (findfileshigh && findfileshigh->next)
|
|
findfileshigh = findfileshigh->next;
|
|
}
|
|
}
|
|
else if (keystatus[0x0e]) // backspace
|
|
{
|
|
Bstrcat(selectedboardfilename, "../");
|
|
tweak_sboardfilename();
|
|
Bstrcpy(g_oldpath, selectedboardfilename);
|
|
getfilenames(selectedboardfilename, "*.map");
|
|
keystatus[0x0e] = 0;
|
|
}
|
|
else if (ch == 13 && currentlist == 0)
|
|
{
|
|
if (finddirshigh->type == CACHE1D_FIND_DRIVE)
|
|
Bstrcpy(selectedboardfilename, finddirshigh->name);
|
|
else
|
|
Bstrcat(selectedboardfilename, finddirshigh->name);
|
|
|
|
Bstrcat(selectedboardfilename, "/");
|
|
tweak_sboardfilename();
|
|
|
|
Bstrcpy(g_oldpath, selectedboardfilename);
|
|
//printf("Changing directories to: %s\n", selectedboardfilename);
|
|
|
|
if (finddirshigh == fnlist.finddirs)
|
|
{
|
|
getfilenames(selectedboardfilename, "*.map");
|
|
currentlist = 0;
|
|
}
|
|
else getfilenames(selectedboardfilename, "*.map");;
|
|
|
|
|
|
ch = 0;
|
|
|
|
begindrawing();
|
|
CLEARLINES2D(0, ydim16, 0);
|
|
enddrawing();
|
|
showframe(1);
|
|
}
|
|
|
|
if (ch == 13 && !findfileshigh) ch = 0;
|
|
}
|
|
while (ch != 13 && ch != 27);
|
|
|
|
if (ch == 13)
|
|
{
|
|
Bstrcat(selectedboardfilename, findfileshigh->name);
|
|
//printf("Selected file: %s\n", selectedboardfilename);
|
|
|
|
return 0;
|
|
}
|
|
|
|
pathsearchmode = bakpathsearchmode;
|
|
|
|
return -1;
|
|
}
|
|
|
|
static inline int32_t imod(int32_t a, int32_t b)
|
|
{
|
|
if (a >= 0)
|
|
return a%b;
|
|
|
|
return ((a+1)%b)+b-1;
|
|
}
|
|
|
|
int32_t fillsector(int16_t sectnum, int32_t fillcolor)
|
|
{
|
|
int32_t x1, x2, y1, y2, sy, y, daminy;
|
|
int32_t lborder, rborder, uborder, dborder, miny, maxy, dax;
|
|
int16_t z, zz, startwall, endwall, fillcnt;
|
|
|
|
char col;
|
|
|
|
if (fillcolor < 0)
|
|
col = 159-klabs(sintable[((totalclock<<3)&2047)]>>11);
|
|
else
|
|
col = fillcolor;
|
|
|
|
lborder = 0; rborder = xdim;
|
|
y = OSD_GetRowsCur();
|
|
uborder = (y>=0)?(y+1)*8:0; dborder = ydim16-STATUS2DSIZ2;
|
|
|
|
if (sectnum == -1)
|
|
return 0;
|
|
|
|
miny = dborder-1;
|
|
maxy = uborder;
|
|
|
|
startwall = sector[sectnum].wallptr;
|
|
endwall = startwall + sector[sectnum].wallnum - 1;
|
|
for (z=startwall; z<=endwall; z++)
|
|
{
|
|
screencoords(&x1,&y1, wall[z].x-pos.x,wall[z].y-pos.y, zoom);
|
|
if (m32_sideview)
|
|
y1 += getscreenvdisp(getflorzofslope(sectnum,wall[z].x,wall[z].y)-pos.z, zoom);
|
|
|
|
x1 += halfxdim16;
|
|
y1 += midydim16;
|
|
|
|
if (m32_sideview)
|
|
{
|
|
tempxyar[z][0] = x1;
|
|
tempxyar[z][1] = y1;
|
|
}
|
|
|
|
miny = min(miny, y1);
|
|
maxy = max(maxy, y1);
|
|
}
|
|
|
|
if (miny < uborder) miny = uborder;
|
|
if (maxy >= dborder) maxy = dborder-1;
|
|
|
|
daminy = miny+2 - imod(miny+2,3);
|
|
if (sector[sectnum].floorz > minhlsectorfloorz)
|
|
daminy++;
|
|
|
|
//+((totalclock>>2)&3)
|
|
for (sy=daminy; sy<=maxy; sy+=3) // JBF 20040116: numframes%3 -> (totalclock>>2)&3
|
|
{
|
|
y = pos.y + ((sy-midydim16)<<14)/zoom;
|
|
|
|
fillist[0] = lborder; fillcnt = 1;
|
|
for (z=startwall; z<=endwall; z++)
|
|
{
|
|
if (m32_sideview)
|
|
{
|
|
x1 = tempxyar[z][0];
|
|
y1 = tempxyar[z][1];
|
|
x2 = tempxyar[wall[z].point2][0];
|
|
y2 = tempxyar[wall[z].point2][1];
|
|
|
|
if (y1 > y2)
|
|
{
|
|
swaplong(&x1, &x2);
|
|
swaplong(&y1, &y2);
|
|
}
|
|
|
|
if (y1 <= sy && sy < y2)
|
|
{
|
|
if (fillcnt == ARRAY_SIZE(fillist))
|
|
break;
|
|
|
|
x1 += scale(sy-y1, x2-x1, y2-y1);
|
|
fillist[fillcnt++] = x1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
x1 = wall[z].x; x2 = POINT2(z).x;
|
|
y1 = wall[z].y; y2 = POINT2(z).y;
|
|
|
|
if (y1 > y2)
|
|
{
|
|
swaplong(&x1, &x2);
|
|
swaplong(&y1, &y2);
|
|
}
|
|
|
|
if (y1 <= y && y < y2)
|
|
//if (x1*(y-y2) + x2*(y1-y) <= 0)
|
|
{
|
|
dax = x1 + scale(y-y1, x2-x1, y2-y1);
|
|
dax = halfxdim16 + (((dax-pos.x)*zoom)>>14);
|
|
if (dax >= lborder)
|
|
{
|
|
if (fillcnt == ARRAY_SIZE(fillist))
|
|
break;
|
|
|
|
fillist[fillcnt++] = dax;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (fillcnt > 0)
|
|
{
|
|
for (z=1; z<fillcnt; z++)
|
|
for (zz=0; zz<z; zz++)
|
|
if (fillist[z] < fillist[zz])
|
|
swaplong(&fillist[z], &fillist[zz]);
|
|
|
|
for (z=(fillcnt&1); z<fillcnt-1; z+=2)
|
|
{
|
|
if (fillist[z] > rborder)
|
|
break;
|
|
if (fillist[z+1] > rborder)
|
|
fillist[z+1] = rborder;
|
|
|
|
drawline16(fillist[z]+1,sy, fillist[z+1]-1,sy, col); //editorcolors[fillcolor]
|
|
}
|
|
}
|
|
}
|
|
|
|
return(0);
|
|
}
|
|
|
|
static int16_t whitelinescan(int16_t sucksect, int16_t dalinehighlight)
|
|
{
|
|
int32_t i, j, k;
|
|
int16_t tnewnumwalls;
|
|
|
|
if (numsectors >= MAXSECTORS)
|
|
return MAXWALLS+1;
|
|
|
|
Bmemcpy(§or[numsectors], §or[sucksect], sizeof(sectortype));
|
|
sector[numsectors].wallptr = numwalls;
|
|
sector[numsectors].wallnum = 0;
|
|
|
|
i = dalinehighlight;
|
|
tnewnumwalls = numwalls;
|
|
do
|
|
{
|
|
if (tnewnumwalls >= MAXWALLS)
|
|
return MAXWALLS+1;
|
|
|
|
j = lastwall(i);
|
|
if (wall[j].nextwall >= 0)
|
|
{
|
|
j = wall[j].point2;
|
|
for (k=0; k<numwalls; k++)
|
|
{
|
|
if (POINT2(k).x == wall[j].x && POINT2(k).y == wall[j].y)
|
|
if (wall[k].nextwall == -1)
|
|
{
|
|
j = k;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
Bmemcpy(&wall[tnewnumwalls], &wall[i], sizeof(walltype));
|
|
|
|
wall[tnewnumwalls].nextwall = j;
|
|
wall[tnewnumwalls].nextsector = sectorofwall(j);
|
|
|
|
tnewnumwalls++;
|
|
sector[numsectors].wallnum++;
|
|
|
|
i = j;
|
|
}
|
|
while (i != dalinehighlight);
|
|
|
|
for (i=numwalls; i<tnewnumwalls-1; i++)
|
|
wall[i].point2 = i+1;
|
|
wall[tnewnumwalls-1].point2 = numwalls;
|
|
|
|
return (clockdir(numwalls) == CLOCKDIR_CCW) ? -1 : tnewnumwalls;
|
|
}
|
|
|
|
int32_t loadnames(const char *namesfile, int8_t root)
|
|
{
|
|
char buffer[1024], *p, *name, *number, *endptr;
|
|
static int32_t syms=0;
|
|
int32_t num, line=0, a, comment=0;
|
|
int8_t quotes=0, anglebrackets=0;
|
|
BFILE *fp;
|
|
|
|
Bstrncpyz(buffer, namesfile, sizeof(buffer));
|
|
|
|
fp = fopenfrompath(buffer,"r");
|
|
if (!fp)
|
|
{
|
|
p = buffer;
|
|
while (*p)
|
|
{
|
|
*p = Btolower(*p);
|
|
p++;
|
|
}
|
|
|
|
if ((fp = fopenfrompath(buffer,"r")) == NULL)
|
|
{
|
|
initprintf("Failed to open %s\n", buffer);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
if (root)
|
|
//clearbufbyte(names, sizeof(names), 0);
|
|
Bmemset(names,0,sizeof(names));
|
|
|
|
initprintf("Loading %s\n", buffer);
|
|
|
|
while (Bfgets(buffer, 1024, fp))
|
|
{
|
|
a = Bstrlen(buffer);
|
|
if (a >= 1)
|
|
{
|
|
if (a > 1)
|
|
if (buffer[a-2] == '\r') buffer[a-2] = 0;
|
|
if (buffer[a-1] == '\n') buffer[a-1] = 0;
|
|
}
|
|
|
|
p = buffer;
|
|
line++;
|
|
while (*p == 32) p++; // 32 == 0x20 == space
|
|
if (*p == 0) continue; // blank line
|
|
|
|
if (*p == '#') // make '#' optional for compatibility
|
|
p++;
|
|
|
|
if (*p == '/')
|
|
{
|
|
if (*(p+1) == '/') continue; // comment
|
|
if (*(p+1) == '*') {comment++; continue;} /* comment */
|
|
}
|
|
else if (*p == '*' && p[1] == '/')
|
|
{
|
|
comment--;
|
|
continue;
|
|
}
|
|
else if (comment)
|
|
continue;
|
|
else if (!comment)
|
|
{
|
|
while (*p == 32) p++;
|
|
if (*p == 0) continue; // null directive
|
|
|
|
if (!Bstrncmp(p, "define ", 7))
|
|
{
|
|
// #define_...
|
|
p += 7;
|
|
while (*p == 32) p++;
|
|
if (*p == 0)
|
|
{
|
|
initprintf("Error: Malformed #define at line %d\n", line-1);
|
|
continue;
|
|
}
|
|
|
|
name = p;
|
|
while (*p != 32 && *p != 0) p++;
|
|
if (*p == 32)
|
|
{
|
|
*(p++) = 0;
|
|
while (*p == 32) p++;
|
|
if (*p == 0) // #define_NAME with no number
|
|
{
|
|
initprintf("Error: No number given for name \"%s\" (line %d)\n", name, line-1);
|
|
continue;
|
|
}
|
|
|
|
number = p;
|
|
while (*p != 0 && *p != 32) p++;
|
|
*p = 0;
|
|
|
|
// add to list
|
|
num = Bstrtol(number, &endptr, 10);
|
|
if (*endptr != 0)
|
|
{
|
|
p = endptr;
|
|
goto badline;
|
|
}
|
|
//printf("Grokked \"%s\" -> \"%d\"\n", name, num);
|
|
if (num < 0 || num >= MAXTILES)
|
|
{
|
|
initprintf("Error: Constant %d for name \"%s\" out of range (line %d)\n", num, name, line-1);
|
|
continue;
|
|
}
|
|
|
|
if (Bstrlen(name) > 24)
|
|
initprintf("Warning: Name \"%s\" longer than 24 characters (line %d). Truncating.\n", name, line-1);
|
|
|
|
Bstrncpyz(names[num], name, 25);
|
|
|
|
syms++;
|
|
|
|
continue;
|
|
}
|
|
else // #define_NAME with no number
|
|
{
|
|
initprintf("Error: No number given for name \"%s\" (line %d)\n", name, line-1);
|
|
continue;
|
|
}
|
|
}
|
|
else if (!Bstrncmp(p, "include ", 8))
|
|
{
|
|
// #include_...
|
|
p += 8;
|
|
while (*p == 32) p++;
|
|
if (*p == 0)
|
|
{
|
|
initprintf("Error: Malformed #include at line %d\n", line-1);
|
|
continue;
|
|
}
|
|
|
|
if (*p == '\"')
|
|
{
|
|
quotes = 1;
|
|
p++;
|
|
}
|
|
else if (*p == '<')
|
|
{
|
|
anglebrackets = 1;
|
|
p++;
|
|
}
|
|
|
|
name = p;
|
|
if (quotes == 1)
|
|
{
|
|
while (*p != '\"' && *p != 0) p++;
|
|
quotes = 0;
|
|
if (*p == 0)
|
|
{
|
|
initprintf("Error: Missing \'\"\' in #include at line %d\n", line-1);
|
|
continue;
|
|
}
|
|
*p = 0;
|
|
}
|
|
else if (anglebrackets == 1)
|
|
{
|
|
while (*p != '>' && *p != 0) p++;
|
|
anglebrackets = 0;
|
|
if (*p == 0)
|
|
{
|
|
initprintf("Error: Missing \'>\' in #include at line %d\n", line-1);
|
|
continue;
|
|
}
|
|
*p = 0;
|
|
}
|
|
else
|
|
{
|
|
while (*p != 32 && *p != 0) p++;
|
|
*p = 0;
|
|
}
|
|
|
|
loadnames(name, 0);
|
|
|
|
continue;
|
|
}
|
|
}
|
|
badline:
|
|
initprintf("Error: Invalid statement found at character %d on line %d\n", (int32_t)(p-buffer), line-1);
|
|
}
|
|
if (root)
|
|
initprintf("Loaded %d names.\n", syms);
|
|
|
|
Bfclose(fp);
|
|
return 0;
|
|
}
|
|
|
|
void printcoords16(int32_t posxe, int32_t posye, int16_t ange)
|
|
{
|
|
char snotbuf[128];
|
|
int32_t i, m;
|
|
int32_t v8 = (numsectors > MAXSECTORSV7 || numwalls > MAXWALLSV7 ||
|
|
Numsprites > MAXSPRITESV7 || numyaxbunches > 0);
|
|
#if M32_UNDO
|
|
Bsprintf(snotbuf,"x:%d y:%d ang:%d r%d",posxe,posye,ange,map_revision-1);
|
|
#else
|
|
Bsprintf(snotbuf,"x:%d y:%d ang:%d",posxe,posye,ange);
|
|
#endif
|
|
i = 0;
|
|
while ((snotbuf[i] != 0) && (i < 33))
|
|
i++;
|
|
while (i < 33)
|
|
{
|
|
snotbuf[i] = 32;
|
|
i++;
|
|
}
|
|
snotbuf[33] = 0;
|
|
|
|
clearministatbar16();
|
|
|
|
printext16(8, ydim-STATUS2DSIZ+128, whitecol, -1, snotbuf,0);
|
|
|
|
if (highlightcnt<=0 && highlightsectorcnt<=0)
|
|
{
|
|
m = Bsprintf(snotbuf,"%d/%d %s. %d",
|
|
numsectors, v8?MAXSECTORSV8:MAXSECTORSV7,
|
|
numyaxbunches>0 ? "SEC":"sec", numwalls);
|
|
if (numyaxbunches > 0)
|
|
{
|
|
if (xdim >= 800)
|
|
Bsprintf(&snotbuf[m], "/%d wal. %d/16k spr. %d/256 bn.",
|
|
MAXWALLSV8, Numsprites, numyaxbunches);
|
|
else
|
|
Bsprintf(&snotbuf[m], " wal. %d spr. %d/256 bn.",
|
|
Numsprites, numyaxbunches);
|
|
}
|
|
else
|
|
{
|
|
if (xdim >= 800)
|
|
Bsprintf(&snotbuf[m], "/%d wal. %d/%d spr.",
|
|
v8?MAXWALLSV8:MAXWALLSV7, Numsprites,
|
|
v8?MAXSPRITESV8:MAXSPRITESV7);
|
|
else
|
|
Bsprintf(&snotbuf[m], "/%dk wal. %d/%dk spr.",
|
|
(v8?MAXWALLSV8:MAXWALLSV7)/1000, Numsprites,
|
|
(v8?MAXSPRITESV8:MAXSPRITESV7)/1000);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (highlightcnt>0)
|
|
{
|
|
m = 0;
|
|
for (i=0; i<highlightcnt; i++)
|
|
m += !!(highlight[i]&16384);
|
|
Bsprintf(snotbuf, "%d sprites, %d walls selected", m, highlightcnt-m);
|
|
}
|
|
else if (highlightsectorcnt>0)
|
|
{
|
|
int32_t ii=Bsprintf(snotbuf, "%d sectors with a total of %d walls selected",
|
|
highlightsectorcnt, numhlsecwalls);
|
|
if (m32_rotateang)
|
|
Bsprintf(&snotbuf[ii], " (ang=%d)", m32_rotateang);
|
|
}
|
|
else
|
|
{
|
|
snotbuf[0] = 0;
|
|
}
|
|
|
|
v8 = 1; // yellow color
|
|
}
|
|
|
|
m = xdim/8 - 264/8;
|
|
m = clamp(m, 1, (signed)sizeof(snotbuf)-1);
|
|
|
|
i = 0;
|
|
while (snotbuf[i] && i < m)
|
|
i++;
|
|
while (i < m)
|
|
{
|
|
snotbuf[i] = 32;
|
|
i++;
|
|
}
|
|
snotbuf[m] = 0;
|
|
|
|
printext16(264, ydim-STATUS2DSIZ+128, v8?editorcolors[10]:whitecol, -1, snotbuf,0);
|
|
}
|
|
|
|
#define DOPRINT(Yofs, fmt, ...) do { \
|
|
Bsprintf(snotbuf, fmt, ## __VA_ARGS__); \
|
|
printext16(8+col*200, ydim/*-(row*96)*/-STATUS2DSIZ+Yofs, color, -1, snotbuf, 0); \
|
|
} while (0)
|
|
|
|
void showsectordata(int16_t sectnum, int16_t small)
|
|
{
|
|
char snotbuf[80];
|
|
int32_t col=0; //,row = 0;
|
|
int32_t color = small ? whitecol : editorcolors[11];
|
|
|
|
const sectortype *const sec = §or[sectnum];
|
|
|
|
if (small)
|
|
{
|
|
_printmessage16("^10Sector %d %s ^O(F7 to edit)", sectnum, CallExtGetSectorCaption(sectnum));
|
|
return;
|
|
}
|
|
|
|
DOPRINT(32, "^10Sector %d", sectnum);
|
|
DOPRINT(48, "Firstwall: %d", TrackerCast(sec->wallptr));
|
|
DOPRINT(56, "Numberofwalls: %d", TrackerCast(sec->wallnum));
|
|
DOPRINT(64, "Firstsprite: %d", headspritesect[sectnum]);
|
|
DOPRINT(72, "Tags: %d, %d", TrackerCast(sec->hitag), TrackerCast(sec->lotag));
|
|
DOPRINT(80, " (0x%x), (0x%x)", TrackerCast(sec->hitag), TrackerCast(sec->lotag));
|
|
DOPRINT(88, "Extra: %d", TrackerCast(sec->extra));
|
|
DOPRINT(96, "Visibility: %d", TrackerCast(sec->visibility));
|
|
DOPRINT(104, "Pixel height: %d", (sec->floorz-sec->ceilingz)>>8);
|
|
|
|
col++;
|
|
|
|
DOPRINT(32, "^10CEILING:^O");
|
|
DOPRINT(48, "Flags (hex): %x", TrackerCast(sec->ceilingstat));
|
|
{
|
|
int32_t xp=sec->ceilingxpanning, yp=sec->ceilingypanning;
|
|
#ifdef YAX_ENABLE__COMPAT
|
|
if (yax_getbunch(sectnum, YAX_CEILING) >= 0)
|
|
xp = yp = 0;
|
|
#endif
|
|
DOPRINT(56, "(X,Y)pan: %d, %d", xp, yp);
|
|
}
|
|
DOPRINT(64, "Shade byte: %d", TrackerCast(sec->ceilingshade));
|
|
DOPRINT(72, "Z-coordinate: %d", TrackerCast(sec->ceilingz));
|
|
DOPRINT(80, "Tile number: %d", TrackerCast(sec->ceilingpicnum));
|
|
DOPRINT(88, "Ceiling heinum: %d", TrackerCast(sec->ceilingheinum));
|
|
DOPRINT(96, "Palookup number: %d", TrackerCast(sec->ceilingpal));
|
|
#ifdef YAX_ENABLE
|
|
DOPRINT(104, "Bunch number: %d", yax_getbunch(sectnum, YAX_CEILING));
|
|
#endif
|
|
|
|
col++;
|
|
|
|
DOPRINT(32, "^10FLOOR:^O");
|
|
DOPRINT(48, "Flags (hex): %x", TrackerCast(sec->floorstat));
|
|
{
|
|
int32_t xp=sec->floorxpanning, yp=sec->floorypanning;
|
|
#ifdef YAX_ENABLE__COMPAT
|
|
if (yax_getbunch(sectnum, YAX_FLOOR) >= 0)
|
|
xp = yp = 0;
|
|
#endif
|
|
DOPRINT(56, "(X,Y)pan: %d, %d", xp, yp);
|
|
}
|
|
DOPRINT(64, "Shade byte: %d", TrackerCast(sec->floorshade));
|
|
DOPRINT(72, "Z-coordinate: %d", TrackerCast(sec->floorz));
|
|
DOPRINT(80, "Tile number: %d", TrackerCast(sec->floorpicnum));
|
|
DOPRINT(88, "Floor heinum: %d", TrackerCast(sec->floorheinum));
|
|
DOPRINT(96, "Palookup number: %d", TrackerCast(sec->floorpal));
|
|
#ifdef YAX_ENABLE
|
|
DOPRINT(104, "Bunch number: %d", yax_getbunch(sectnum, YAX_FLOOR));
|
|
#endif
|
|
}
|
|
|
|
void showwalldata(int16_t wallnum, int16_t small)
|
|
{
|
|
walltype const * const wal = &wall[wallnum];
|
|
int32_t sec;
|
|
char snotbuf[80];
|
|
int32_t col=0; //, row = 0;
|
|
int32_t color = small ? whitecol : editorcolors[11];
|
|
|
|
if (small)
|
|
{
|
|
_printmessage16("^10Wall %d %s ^O(F8 to edit)", wallnum,
|
|
CallExtGetWallCaption(wallnum));
|
|
return;
|
|
}
|
|
|
|
DOPRINT(32, "^10Wall %d", wallnum);
|
|
DOPRINT(48, "X-coordinate: %d", TrackerCast(wal->x));
|
|
DOPRINT(56, "Y-coordinate: %d", TrackerCast(wal->y));
|
|
DOPRINT(64, "Point2: %d", TrackerCast(wal->point2));
|
|
DOPRINT(72, "Sector: ^010%d", sectorofwall(wallnum));
|
|
|
|
DOPRINT(88, "Tags: %d, %d", TrackerCast(wal->hitag), TrackerCast(wal->lotag));
|
|
DOPRINT(96, " (0x%x), (0x%x)", TrackerCast(wal->hitag), TrackerCast(wal->lotag));
|
|
|
|
col++;
|
|
|
|
DOPRINT(32, "^10%s^O", (wal->picnum>=0 && wal->picnum<MAXTILES) ? names[wal->picnum] : "!INVALID!");
|
|
DOPRINT(48, "Flags (hex): %x", TrackerCast(wal->cstat));
|
|
DOPRINT(56, "Shade: %d", TrackerCast(wal->shade));
|
|
DOPRINT(64, "Pal: %d", TrackerCast(wal->pal));
|
|
DOPRINT(72, "(X,Y)repeat: %d, %d", TrackerCast(wal->xrepeat), TrackerCast(wal->yrepeat));
|
|
DOPRINT(80, "(X,Y)pan: %d, %d", TrackerCast(wal->xpanning), TrackerCast(wal->ypanning));
|
|
DOPRINT(88, "Tile number: %d", TrackerCast(wal->picnum));
|
|
DOPRINT(96, "OverTile number: %d", TrackerCast(wal->overpicnum));
|
|
|
|
col++;
|
|
|
|
DOPRINT(48-(small?16:0), "nextsector: %d", TrackerCast(wal->nextsector));
|
|
DOPRINT(56-(small?16:0), "nextwall: %d", TrackerCast(wal->nextwall));
|
|
|
|
DOPRINT(72-(small?16:0), "Extra: %d", TrackerCast(wal->extra));
|
|
|
|
// TX 20050102 I'm not sure what unit dist<<4 is supposed to be, but dist itself is correct in terms of game coordinates as one would expect
|
|
DOPRINT(96-(small?16:0), "Wall length: %d", wallength(wallnum));
|
|
|
|
sec = sectorofwall(wallnum);
|
|
DOPRINT(104-(small?16:0), "Pixel height: %d", (sector[sec].floorz-sector[sec].ceilingz)>>8);
|
|
}
|
|
|
|
void showspritedata(int16_t spritenum, int16_t small)
|
|
{
|
|
spritetype *spr;
|
|
char snotbuf[80];
|
|
int32_t col=0; //, row = 0;
|
|
int32_t color = small ? whitecol : editorcolors[11];
|
|
|
|
spr = &sprite[spritenum];
|
|
|
|
if (small)
|
|
{
|
|
_printmessage16("^10Sprite %d %s ^O(F8 to edit)",spritenum, CallExtGetSpriteCaption(spritenum));
|
|
return;
|
|
}
|
|
|
|
DOPRINT(32, "^10Sprite %d", spritenum);
|
|
DOPRINT(48, "X-coordinate: %d", TrackerCast(spr->x));
|
|
DOPRINT(56, "Y-coordinate: %d", TrackerCast(spr->y));
|
|
DOPRINT(64, "Z-coordinate: %d", TrackerCast(spr->z));
|
|
|
|
DOPRINT(72, "Sectnum: ^010%d", TrackerCast(spr->sectnum));
|
|
DOPRINT(80, "Statnum: %d", TrackerCast(spr->statnum));
|
|
|
|
DOPRINT(96, "Tags: %d, %d", TrackerCast(spr->hitag), TrackerCast(spr->lotag));
|
|
DOPRINT(104, " (0x%x), (0x%x)", TrackerCast(spr->hitag), TrackerCast(spr->lotag));
|
|
|
|
col++;
|
|
|
|
DOPRINT(32, "^10,0 ^O"); // 24 blanks
|
|
DOPRINT(32, "^10%s^O", (spr->picnum>=0 && spr->picnum<MAXTILES) ? names[spr->picnum] : "!INVALID!");
|
|
DOPRINT(48, "Flags (hex): %x", TrackerCast(spr->cstat));
|
|
DOPRINT(56, "Shade: %d", TrackerCast(spr->shade));
|
|
DOPRINT(64, "Pal: %d", TrackerCast(spr->pal));
|
|
DOPRINT(72, "Blend: %d", TrackerCast(spr->blend));
|
|
DOPRINT(80, "(X,Y)repeat: %d, %d", TrackerCast(spr->xrepeat), TrackerCast(spr->yrepeat));
|
|
DOPRINT(88, "(X,Y)offset: %d, %d", TrackerCast(spr->xoffset), TrackerCast(spr->yoffset));
|
|
DOPRINT(96, "Tile number: %d", TrackerCast(spr->picnum));
|
|
|
|
col++;
|
|
|
|
DOPRINT(48, "Angle (2048 degrees): %d", TrackerCast(spr->ang));
|
|
DOPRINT(56, "X-Velocity: %d", TrackerCast(spr->xvel));
|
|
DOPRINT(64, "Y-Velocity: %d", TrackerCast(spr->yvel));
|
|
DOPRINT(72, "Z-Velocity: %d", TrackerCast(spr->zvel));
|
|
DOPRINT(80, "Owner: %d", TrackerCast(spr->owner));
|
|
DOPRINT(88, "Clipdist: %d", TrackerCast(spr->clipdist));
|
|
DOPRINT(96, "Extra: %d", TrackerCast(spr->extra));
|
|
}
|
|
|
|
#undef DOPRINT
|
|
|
|
// gets called once per totalclock increment since last call
|
|
static void keytimerstuff(void)
|
|
{
|
|
if (!g_doHardcodedMovement)
|
|
return;
|
|
|
|
if (DOWN_BK(STRAFE) == 0)
|
|
{
|
|
if (DOWN_BK(TURNLEFT)) angvel = max(angvel-pk_turnaccel, -128);
|
|
if (DOWN_BK(TURNRIGHT)) angvel = min(angvel+pk_turnaccel, 127);
|
|
}
|
|
else
|
|
{
|
|
if (DOWN_BK(TURNLEFT)) svel = min(svel+16, 255); // svel and vel aren't even chars...
|
|
if (DOWN_BK(TURNRIGHT)) svel = max(svel-16, -256);
|
|
}
|
|
if (DOWN_BK(MOVEFORWARD)) vel = min(vel+16, 255);
|
|
if (DOWN_BK(MOVEBACKWARD)) vel = max(vel-16, -256);
|
|
/* if (DOWN_BK(STRAFELEFT)) svel = min(svel+8, 127);
|
|
if (DOWN_BK(STRAFERIGHT)) svel = max(svel-8, -128); */
|
|
|
|
if (angvel < 0) angvel = min(angvel+pk_turndecel, 0);
|
|
if (angvel > 0) angvel = max(angvel-pk_turndecel, 0);
|
|
if (svel < 0) svel = min(svel+6, 0);
|
|
if (svel > 0) svel = max(svel-6, 0);
|
|
if (vel < 0) vel = min(vel+6, 0);
|
|
if (vel > 0) vel = max(vel-6, 0);
|
|
/* if(mlook) pos.z -= (horiz-101)*(vel/40); */
|
|
}
|
|
|
|
#if 0
|
|
int32_t snfillprintf(char *outbuf, size_t bufsiz, int32_t fill, const char *fmt, ...)
|
|
{
|
|
char tmpstr[256];
|
|
int32_t nwritten, ofs;
|
|
va_list va;
|
|
|
|
va_start(va, fmt);
|
|
nwritten = Bvsnprintf(tmpstr, bufsiz, fmt, va);
|
|
va_end(va);
|
|
|
|
ofs = min(nwritten, (signed)bufsiz-1);
|
|
Bmemset(outbuf, fill, bufsiz-ofs);
|
|
|
|
return ofs;
|
|
}
|
|
#endif
|
|
|
|
void _printmessage16(const char *fmt, ...)
|
|
{
|
|
int32_t i, ybase;
|
|
char snotbuf[156];
|
|
char tmpstr[160];
|
|
va_list va;
|
|
|
|
va_start(va, fmt);
|
|
Bvsnprintf(tmpstr, sizeof(tmpstr), fmt, va);
|
|
va_end(va);
|
|
|
|
i = 0;
|
|
while (tmpstr[i] && i < 146)
|
|
{
|
|
snotbuf[i] = tmpstr[i];
|
|
i++;
|
|
}
|
|
snotbuf[i] = 0;
|
|
if (lastpm16time == totalclock)
|
|
Bstrcpy(lastpm16buf, snotbuf);
|
|
|
|
clearministatbar16();
|
|
|
|
ybase = ydim-STATUS2DSIZ+128-8;
|
|
|
|
printext16(8, ybase+8, whitecol, -1, snotbuf, 0);
|
|
}
|
|
|
|
void printmessage256(int32_t x, int32_t y, const char *name)
|
|
{
|
|
char snotbuf[80];
|
|
int32_t i;
|
|
|
|
i = 0;
|
|
while (name[i] && i < 62)
|
|
{
|
|
snotbuf[i] = name[i];
|
|
i++;
|
|
}
|
|
while (i < 62)
|
|
{
|
|
snotbuf[i] = 32;
|
|
i++;
|
|
}
|
|
snotbuf[62] = 0;
|
|
printext256(x+2,y+2,0,-1,snotbuf,0);
|
|
printext256(x,y,whitecol,-1,snotbuf,0);
|
|
}
|
|
|
|
//Find closest point (*dax, *day) on wall (dawall) to (x, y)
|
|
static void getclosestpointonwall(int32_t x, int32_t y, int32_t dawall, int32_t *nx, int32_t *ny,
|
|
int32_t maybe_screen_coord_p)
|
|
{
|
|
int64_t i, j, wx,wy, wx2,wy2, dx, dy;
|
|
|
|
if (m32_sideview && maybe_screen_coord_p)
|
|
{
|
|
wx = m32_wallscreenxy[dawall][0];
|
|
wy = m32_wallscreenxy[dawall][1];
|
|
wx2 = m32_wallscreenxy[wall[dawall].point2][0];
|
|
wy2 = m32_wallscreenxy[wall[dawall].point2][1];
|
|
}
|
|
else
|
|
{
|
|
wx = wall[dawall].x;
|
|
wy = wall[dawall].y;
|
|
wx2 = POINT2(dawall).x;
|
|
wy2 = POINT2(dawall).y;
|
|
}
|
|
|
|
dx = wx2 - wx;
|
|
dy = wy2 - wy;
|
|
i = dx*(x-wx) + dy*(y-wy);
|
|
if (i <= 0) { *nx = wx; *ny = wy; return; }
|
|
j = dx*dx + dy*dy;
|
|
if (i >= j) { *nx = wx2; *ny = wy2; return; }
|
|
i=((i<<15)/j)<<15;
|
|
*nx = wx + ((dx*i)>>30);
|
|
*ny = wy + ((dy*i)>>30);
|
|
}
|
|
|
|
static void initcrc(void)
|
|
{
|
|
int32_t i, j, k, a;
|
|
|
|
for (j=0; j<256; j++) //Calculate CRC table
|
|
{
|
|
k = (j<<8); a = 0;
|
|
for (i=7; i>=0; i--)
|
|
{
|
|
if (((k^a)&0x8000) > 0)
|
|
a = ((a<<1)&65535) ^ 0x1021; //0x1021 = genpoly
|
|
else
|
|
a = ((a<<1)&65535);
|
|
k = ((k<<1)&65535);
|
|
}
|
|
crctable[j] = (a&65535);
|
|
}
|
|
}
|
|
|
|
static int32_t GetWallBaseZ(int32_t wallnum)
|
|
{
|
|
int32_t z=0;
|
|
|
|
const int32_t sectnum = sectorofwall(wallnum);
|
|
const int32_t nextsec = wall[wallnum].nextsector;
|
|
|
|
if (nextsec == -1) //1-sided wall
|
|
{
|
|
if (wall[wallnum].cstat&4) // floor-aligned
|
|
z = sector[sectnum].floorz;
|
|
else
|
|
z = sector[sectnum].ceilingz;
|
|
}
|
|
else //2-sided wall
|
|
{
|
|
if (wall[wallnum].cstat&4)
|
|
z = sector[sectnum].ceilingz;
|
|
else
|
|
{
|
|
if (sector[nextsec].ceilingz > sector[sectnum].ceilingz)
|
|
z = sector[nextsec].ceilingz; //top step
|
|
if (sector[nextsec].floorz < sector[sectnum].floorz)
|
|
z = sector[nextsec].floorz; //bottom step
|
|
}
|
|
}
|
|
|
|
return z;
|
|
}
|
|
|
|
|
|
////////// AUTOMATIC WALL ALIGNMENT //////////
|
|
|
|
static void AlignWalls_(int32_t tilenum, int32_t z0, int32_t z1, int32_t doxpanning,
|
|
int32_t w0_pan, int32_t w0_rep, int32_t w1_pan, int32_t w1_rep)
|
|
{
|
|
int32_t n;
|
|
|
|
if (tilesiz[tilenum].x==0 || tilesiz[tilenum].y==0)
|
|
return;
|
|
|
|
//do the x alignment
|
|
if (doxpanning)
|
|
wall[w1_pan].xpanning = (uint8_t)((wall[w0_pan].xpanning + (wall[w0_rep].xrepeat<<3))%tilesiz[tilenum].x);
|
|
|
|
for (n=picsiz[tilenum]>>4; (1<<n)<tilesiz[tilenum].y; n++);
|
|
|
|
wall[w1_rep].yrepeat = wall[w0_rep].yrepeat;
|
|
wall[w1_pan].ypanning = (uint8_t)(wall[w0_pan].ypanning + (((z1-z0)*wall[w0_rep].yrepeat)>>(n+3)));
|
|
}
|
|
|
|
static void AlignWalls(int32_t w0, int32_t z0, int32_t w1, int32_t z1, int32_t doxpanning)
|
|
{
|
|
AlignWalls_(wall[w0].picnum, z0, z1, doxpanning, w0, w0, w1, w1);
|
|
}
|
|
|
|
void AlignWallPoint2(int32_t w0)
|
|
{
|
|
int32_t w1 = wall[w0].point2;
|
|
AlignWalls(w0,GetWallBaseZ(w0), w1,GetWallBaseZ(w1), 1);
|
|
}
|
|
|
|
#define ALIGN_WALLS_CSTAT_MASK (4+8+256)
|
|
|
|
static int32_t AlignGetWall(int32_t botswap, int32_t w)
|
|
{
|
|
return botswap && (wall[w].cstat&2) && wall[w].nextwall >= 0 ? wall[w].nextwall : w;
|
|
}
|
|
|
|
// flags:
|
|
// 1: more than once
|
|
// 2: (unused)
|
|
// 4: carry pixel width from first wall over to the rest
|
|
// 8: align TROR nextwalls
|
|
// 16: iterate lastwall()s (point2 in reverse)
|
|
// 32: use special logic for 'bottom-swapped' walls
|
|
int32_t AutoAlignWalls(int32_t w0, uint32_t flags, int32_t nrecurs)
|
|
{
|
|
static int32_t numaligned, wall0, cstat0;
|
|
static uint32_t lenrepquot;
|
|
|
|
const int32_t totheleft = flags&16;
|
|
const int32_t botswap = flags&32;
|
|
|
|
int32_t z0 = GetWallBaseZ(w0);
|
|
int32_t w1 = totheleft ? lastwall(w0) : wall[w0].point2;
|
|
|
|
int32_t w0b = AlignGetWall(botswap, w0);
|
|
const int32_t tilenum = wall[w0b].picnum;
|
|
|
|
if (nrecurs == 0)
|
|
{
|
|
//clear visited bits
|
|
Bmemset(visited, 0, sizeof(visited));
|
|
visited[w0>>3] |= (1<<(w0&7));
|
|
numaligned = 0;
|
|
lenrepquot = getlenbyrep(wallength(w0), wall[w0].xrepeat);
|
|
wall0 = w0;
|
|
cstat0 = wall[w0b].cstat & ALIGN_WALLS_CSTAT_MASK; // top/bottom orientation; x/y-flip
|
|
}
|
|
|
|
//loop through walls at this vertex in point2 order
|
|
while (1)
|
|
{
|
|
int32_t w1b = AlignGetWall(botswap, w1);
|
|
|
|
//break if this wall would connect us in a loop
|
|
if (visited[w1>>3]&(1<<(w1&7)))
|
|
break;
|
|
|
|
visited[w1>>3] |= (1<<(w1&7));
|
|
|
|
#ifdef YAX_ENABLE
|
|
if (flags&8)
|
|
{
|
|
int32_t cf, ynw;
|
|
|
|
for (cf=0; cf<2; cf++)
|
|
{
|
|
ynw = yax_getnextwall(w0, cf);
|
|
|
|
if (ynw >= 0 && wall[ynw].picnum==tilenum && (visited[ynw>>3]&(1<<(ynw&7)))==0)
|
|
{
|
|
wall[ynw].xrepeat = wall[w0].xrepeat;
|
|
wall[ynw].xpanning = wall[w0].xpanning;
|
|
AlignWalls(w0,z0, ynw,GetWallBaseZ(ynw), 0); // initial vertical alignment
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
//break if reached back of left wall
|
|
if (wall[w1].nextwall == w0)
|
|
break;
|
|
|
|
if (wall[w1b].picnum == tilenum)
|
|
{
|
|
int32_t visible = 1;
|
|
const int32_t nextsec = wall[w1].nextsector;
|
|
|
|
if (nextsec >= 0)
|
|
{
|
|
int32_t cz,fz, czn,fzn;
|
|
const int32_t sectnum = NEXTWALL(w1).nextsector;
|
|
|
|
//ignore two sided walls that have no visible face
|
|
// TODO: can be more precise (i.e. taking into account the wall face)
|
|
// ... needs to be factored out from some engine code maybe...
|
|
// as is the whole "z base" determination.
|
|
getzsofslope(sectnum, wall[w1].x,wall[w1].y, &cz, &fz);
|
|
getzsofslope(nextsec, wall[w1].x,wall[w1].y, &czn, &fzn);
|
|
|
|
if (czn <= cz && fzn >= fz)
|
|
visible = 0;
|
|
}
|
|
|
|
if (visible)
|
|
{
|
|
const int32_t z1 = GetWallBaseZ(w1);
|
|
|
|
if ((flags&4) && w1!=wall0)
|
|
fixxrepeat(w1, lenrepquot);
|
|
AlignWalls_(tilenum,z0, z1, 1, w0b, w0, w1b, w1);
|
|
wall[w1b].cstat &= ~ALIGN_WALLS_CSTAT_MASK;
|
|
wall[w1b].cstat |= cstat0;
|
|
numaligned++;
|
|
|
|
if ((flags&1)==0)
|
|
return 1;
|
|
|
|
//if wall was 1-sided, no need to recurse
|
|
if (wall[w1].nextwall < 0)
|
|
{
|
|
w0 = w1;
|
|
w0b = AlignGetWall(botswap, w0);
|
|
z0 = GetWallBaseZ(w0);
|
|
w1 = totheleft ? lastwall(w0) : wall[w0].point2;
|
|
|
|
continue;
|
|
}
|
|
|
|
AutoAlignWalls(w1, flags, nrecurs+1);
|
|
}
|
|
}
|
|
|
|
if (wall[w1].nextwall < 0)
|
|
break;
|
|
w1 = totheleft ? lastwall(wall[w1].nextwall) : NEXTWALL(w1).point2;
|
|
}
|
|
|
|
return numaligned;
|
|
}
|
|
|
|
#define PLAYTEST_MAPNAME "autosave_playtest.map"
|
|
|
|
void test_map(int32_t mode)
|
|
{
|
|
if (!mode)
|
|
updatesector(pos.x, pos.y, &cursectnum);
|
|
else
|
|
updatesector(startpos.x, startpos.y, &startsectnum);
|
|
|
|
#ifdef _WIN32
|
|
if (fullscreen)
|
|
{
|
|
printmessage16("Must be in windowed mode to test map!");
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
if ((!mode && cursectnum >= 0) || (mode && startsectnum >= 0))
|
|
{
|
|
char *param = " -map " PLAYTEST_MAPNAME " -noinstancechecking";
|
|
char *fullparam;
|
|
char current_cwd[BMAX_PATH];
|
|
int32_t slen = 0;
|
|
BFILE *fp;
|
|
|
|
if ((program_origcwd[0] == '\0') || !getcwd(current_cwd, BMAX_PATH))
|
|
current_cwd[0] = '\0';
|
|
else // Before we check if file exists, for the case there's no absolute path.
|
|
Bchdir(program_origcwd);
|
|
|
|
fp = fopen(game_executable, "rb"); // File exists?
|
|
if (fp != NULL)
|
|
fclose(fp);
|
|
else
|
|
{
|
|
#ifdef _WIN32
|
|
fullparam = (char *)Bstrrchr(mapster32_fullpath, '\\');
|
|
#else
|
|
fullparam = (char *)Bstrrchr(mapster32_fullpath, '/');
|
|
#endif
|
|
if (fullparam)
|
|
{
|
|
slen = fullparam-mapster32_fullpath+1;
|
|
Bstrncpy(game_executable, mapster32_fullpath, slen);
|
|
// game_executable is now expected to not be NULL-terminated!
|
|
Bstrncpy(game_executable+slen, DefaultGameExec, sizeof(game_executable));
|
|
}
|
|
else
|
|
Bstrncpy(game_executable, DefaultGameLocalExec, sizeof(game_executable));
|
|
}
|
|
|
|
if (current_cwd[0] != '\0') // Temporarily changing back,
|
|
Bchdir(current_cwd); // after checking if file exists.
|
|
|
|
if (testplay_addparam)
|
|
slen = Bstrlen(testplay_addparam);
|
|
|
|
// Considering the NULL character, quatation marks
|
|
// and a possible extra space not in testplay_addparam,
|
|
// the length should be Bstrlen(game_executable)+Bstrlen(param)+(slen+1)+2+1.
|
|
|
|
fullparam = (char *)Xmalloc(Bstrlen(game_executable)+Bstrlen(param)+slen+4);
|
|
Bsprintf(fullparam,"\"%s\"",game_executable);
|
|
|
|
if (testplay_addparam)
|
|
{
|
|
Bstrcat(fullparam, " ");
|
|
Bstrcat(fullparam, testplay_addparam);
|
|
}
|
|
Bstrcat(fullparam, param);
|
|
|
|
CallExtPreSaveMap();
|
|
if (mode)
|
|
saveboard(PLAYTEST_MAPNAME, &startpos, startang, startsectnum);
|
|
else
|
|
saveboard(PLAYTEST_MAPNAME, &pos, ang, cursectnum);
|
|
|
|
message("Board saved to " PLAYTEST_MAPNAME ". Starting the game...");
|
|
OSD_Printf("...as `%s'\n", fullparam);
|
|
|
|
showframe(1);
|
|
uninitmouse();
|
|
#ifdef _WIN32
|
|
{
|
|
STARTUPINFO si;
|
|
PROCESS_INFORMATION pi;
|
|
|
|
ZeroMemory(&si,sizeof(si));
|
|
ZeroMemory(&pi,sizeof(pi));
|
|
si.cb = sizeof(si);
|
|
|
|
if (!CreateProcess(NULL,fullparam,NULL,NULL,0,0,NULL,NULL,&si,&pi))
|
|
message("Error launching the game!");
|
|
else WaitForSingleObject(pi.hProcess,INFINITE);
|
|
}
|
|
#else
|
|
if (current_cwd[0] != '\0')
|
|
{
|
|
Bchdir(program_origcwd);
|
|
if (system(fullparam))
|
|
message("Error launching the game!");
|
|
Bchdir(current_cwd);
|
|
}
|
|
else system(fullparam);
|
|
#endif
|
|
printmessage16("Game process exited");
|
|
initmouse();
|
|
clearkeys();
|
|
|
|
Bfree(fullparam);
|
|
}
|
|
else
|
|
printmessage16("Position must be in valid player space to test map!");
|
|
}
|
|
|
|
void app_crashhandler(void)
|
|
{
|
|
if (levelname[0])
|
|
{
|
|
append_ext_UNSAFE(levelname, "_crash.map");
|
|
SaveBoard(levelname, M32_SB_NOEXT);
|
|
}
|
|
}
|
|
|
|
// These will be more useful in the future...
|
|
static const char *CallExtGetVer(void)
|
|
{
|
|
return ExtGetVer();
|
|
}
|
|
static int32_t CallExtInit(void)
|
|
{
|
|
return ExtInit();
|
|
}
|
|
static int32_t CallExtPreInit(int32_t argc,const char **argv)
|
|
{
|
|
return ExtPreInit(argc, argv);
|
|
}
|
|
static int32_t CallExtPostStartupWindow(void)
|
|
{
|
|
return ExtPostStartupWindow();
|
|
}
|
|
static void CallExtUnInit(void)
|
|
{
|
|
ExtUnInit();
|
|
}
|
|
static void CallExtPreCheckKeys(void)
|
|
{
|
|
ExtPreCheckKeys();
|
|
}
|
|
static void CallExtAnalyzeSprites(int32_t ourx, int32_t oury, int32_t oura, int32_t smoothr)
|
|
{
|
|
ExtAnalyzeSprites(ourx, oury, oura, smoothr);
|
|
VM_OnEvent(EVENT_ANALYZESPRITES, -1);
|
|
}
|
|
static void CallExtCheckKeys(void)
|
|
{
|
|
ExtCheckKeys();
|
|
}
|
|
static void CallExtPreLoadMap(void)
|
|
{
|
|
VM_OnEvent(EVENT_PRELOADMAP, -1);
|
|
ExtPreLoadMap();
|
|
}
|
|
static void CallExtSetupMapFilename(const char *mapname)
|
|
{
|
|
Bstrncpy(levelname, mapname, sizeof(levelname));
|
|
|
|
Bsnprintf(tempbuf, sizeof(tempbuf), "%s - %s", AppProperName, mapname);
|
|
wm_setapptitle(tempbuf);
|
|
|
|
ExtSetupMapFilename(mapname);
|
|
}
|
|
static void CallExtLoadMap(const char *mapname)
|
|
{
|
|
CallExtSetupMapFilename(mapname);
|
|
ExtLoadMap(mapname);
|
|
VM_OnEvent(EVENT_LOADMAP, -1);
|
|
}
|
|
static int32_t CallExtPreSaveMap(void)
|
|
{
|
|
VM_OnEvent(EVENT_PRESAVEMAP, -1);
|
|
return ExtPreSaveMap();
|
|
}
|
|
static void CallExtSaveMap(const char *mapname)
|
|
{
|
|
ExtSaveMap(mapname);
|
|
saveboard("backup.map", &pos, ang, cursectnum);
|
|
VM_OnEvent(EVENT_SAVEMAP, -1);
|
|
}
|
|
static const char *CallExtGetSectorCaption(int16_t sectnum)
|
|
{
|
|
return ExtGetSectorCaption(sectnum);
|
|
}
|
|
static const char *CallExtGetWallCaption(int16_t wallnum)
|
|
{
|
|
return ExtGetWallCaption(wallnum);
|
|
}
|
|
static const char *CallExtGetSpriteCaption(int16_t spritenum)
|
|
{
|
|
return ExtGetSpriteCaption(spritenum);
|
|
}
|
|
static void CallExtShowSectorData(int16_t sectnum)
|
|
{
|
|
ExtShowSectorData(sectnum);
|
|
}
|
|
static void CallExtShowWallData(int16_t wallnum)
|
|
{
|
|
ExtShowWallData(wallnum);
|
|
}
|
|
static void CallExtShowSpriteData(int16_t spritenum)
|
|
{
|
|
ExtShowSpriteData(spritenum);
|
|
}
|
|
static void CallExtEditSectorData(int16_t sectnum)
|
|
{
|
|
ExtEditSectorData(sectnum);
|
|
}
|
|
static void CallExtEditWallData(int16_t wallnum)
|
|
{
|
|
ExtEditWallData(wallnum);
|
|
}
|
|
static void CallExtEditSpriteData(int16_t spritenum)
|
|
{
|
|
ExtEditSpriteData(spritenum);
|
|
}
|
|
#if 0
|
|
static const char *CallExtGetSectorType(int32_t lotag)
|
|
{
|
|
return ExtGetSectorType(lotag);
|
|
}
|
|
#endif
|