319 lines
10 KiB
C++
319 lines
10 KiB
C++
/*
|
|
* Portions of this file are copyright Rebirth contributors and licensed as
|
|
* described in COPYING.txt.
|
|
* Portions of this file are copyright Parallax Software and licensed
|
|
* according to the Parallax license.
|
|
* See COPYING.txt for license details.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <type_traits>
|
|
#include <physfs.h>
|
|
#include "maths.h"
|
|
#include "fwd-object.h"
|
|
#include "fwd-segment.h"
|
|
#include "cpp-valptridx.h"
|
|
|
|
namespace dcx {
|
|
using actdoornum_t = uint8_t;
|
|
constexpr unsigned MAX_WALLS = 255; // Maximum number of walls
|
|
constexpr std::integral_constant<std::size_t, 90> MAX_DOORS{}; // Maximum number of open doors
|
|
struct active_door;
|
|
}
|
|
DXX_VALPTRIDX_DECLARE_SUBTYPE(dcx::, active_door, actdoornum_t, MAX_DOORS);
|
|
|
|
#include "fwd-valptridx.h"
|
|
|
|
#ifdef dsx
|
|
namespace dsx {
|
|
#if defined(DXX_BUILD_DESCENT_I)
|
|
constexpr std::integral_constant<std::size_t, 30> MAX_WALL_ANIMS{}; // Maximum different types of doors
|
|
#elif defined(DXX_BUILD_DESCENT_II)
|
|
constexpr std::integral_constant<std::size_t, 60> MAX_WALL_ANIMS{}; // Maximum different types of doors
|
|
#endif
|
|
|
|
enum wall_type_t : uint8_t
|
|
{
|
|
WALL_NORMAL = 0, // Normal wall
|
|
WALL_BLASTABLE = 1, // Removable (by shooting) wall
|
|
WALL_DOOR = 2, // Door
|
|
WALL_ILLUSION = 3, // Wall that appears to be there, but you can fly thru
|
|
WALL_OPEN = 4, // Just an open side. (Trigger)
|
|
WALL_CLOSED = 5, // Wall. Used for transparent walls.
|
|
#if defined(DXX_BUILD_DESCENT_II)
|
|
WALL_OVERLAY = 6, // Goes over an actual solid side. For triggers
|
|
WALL_CLOAKED = 7, // Can see it, and see through it
|
|
#endif
|
|
};
|
|
}
|
|
|
|
namespace dcx {
|
|
typedef unsigned wall_flag_t;
|
|
constexpr std::integral_constant<wall_flag_t, 1> WALL_BLASTED{}; // Blasted out wall.
|
|
constexpr std::integral_constant<wall_flag_t, 2> WALL_DOOR_OPENED{}; // Open door.
|
|
constexpr std::integral_constant<wall_flag_t, 8> WALL_DOOR_LOCKED{}; // Door is locked.
|
|
constexpr std::integral_constant<wall_flag_t, 16> WALL_DOOR_AUTO{}; // Door automatically closes after time.
|
|
constexpr std::integral_constant<wall_flag_t, 32> WALL_ILLUSION_OFF{}; // Illusionary wall is shut off.
|
|
}
|
|
#if defined(DXX_BUILD_DESCENT_II)
|
|
namespace dsx {
|
|
constexpr std::integral_constant<wall_flag_t, 64> WALL_WALL_SWITCH{}; // This wall is openable by a wall switch.
|
|
constexpr std::integral_constant<wall_flag_t, 128> WALL_BUDDY_PROOF{}; // Buddy assumes he cannot get through this wall.
|
|
}
|
|
#endif
|
|
|
|
namespace dcx {
|
|
typedef unsigned wall_state_t;
|
|
constexpr std::integral_constant<wall_state_t, 0> WALL_DOOR_CLOSED{}; // Door is closed
|
|
constexpr std::integral_constant<wall_state_t, 1> WALL_DOOR_OPENING{}; // Door is opening.
|
|
constexpr std::integral_constant<wall_state_t, 2> WALL_DOOR_WAITING{}; // Waiting to close
|
|
constexpr std::integral_constant<wall_state_t, 3> WALL_DOOR_CLOSING{}; // Door is closing
|
|
}
|
|
#if defined(DXX_BUILD_DESCENT_II)
|
|
namespace dsx {
|
|
constexpr std::integral_constant<wall_state_t, 4> WALL_DOOR_OPEN{}; // Door is open, and staying open
|
|
constexpr std::integral_constant<wall_state_t, 5> WALL_DOOR_CLOAKING{}; // Wall is going from closed -> open
|
|
constexpr std::integral_constant<wall_state_t, 6> WALL_DOOR_DECLOAKING{}; // Wall is going from open -> closed
|
|
}
|
|
#endif
|
|
|
|
namespace dcx {
|
|
enum wall_key_t : uint8_t
|
|
{
|
|
KEY_NONE = 1,
|
|
KEY_BLUE = 2,
|
|
KEY_RED = 4,
|
|
KEY_GOLD = 8,
|
|
};
|
|
|
|
constexpr std::integral_constant<fix, 100 * F1_0> WALL_HPS{}; // Normal wall's hp
|
|
constexpr std::integral_constant<fix, 5 * F1_0> WALL_DOOR_INTERVAL{}; // How many seconds a door is open
|
|
|
|
constexpr fix DOOR_OPEN_TIME = i2f(2); // How long takes to open
|
|
constexpr fix DOOR_WAIT_TIME = i2f(5); // How long before auto door closes
|
|
}
|
|
|
|
#ifdef dsx
|
|
namespace dsx {
|
|
#if defined(DXX_BUILD_DESCENT_I)
|
|
constexpr std::integral_constant<std::size_t, 20> MAX_CLIP_FRAMES{};
|
|
#elif defined(DXX_BUILD_DESCENT_II)
|
|
constexpr std::integral_constant<std::size_t, 50> MAX_CLIP_FRAMES{};
|
|
#endif
|
|
}
|
|
|
|
namespace dcx {
|
|
template <unsigned value>
|
|
struct WALL_IS_DOORWAY_FLAG
|
|
{
|
|
constexpr operator unsigned() const { return value; }
|
|
template <unsigned F2>
|
|
constexpr WALL_IS_DOORWAY_FLAG<value | F2> operator|(WALL_IS_DOORWAY_FLAG<F2>) const
|
|
{
|
|
return {};
|
|
}
|
|
void *operator &() const = delete;
|
|
};
|
|
|
|
template <unsigned F>
|
|
struct WALL_IS_DOORWAY_sresult_t
|
|
{
|
|
};
|
|
|
|
template <unsigned F>
|
|
static inline constexpr WALL_IS_DOORWAY_sresult_t<F> WALL_IS_DOORWAY_sresult(WALL_IS_DOORWAY_FLAG<F>)
|
|
{
|
|
return {};
|
|
}
|
|
|
|
struct WALL_IS_DOORWAY_result_t;
|
|
|
|
// WALL_IS_DOORWAY flags.
|
|
constexpr WALL_IS_DOORWAY_FLAG<1> WID_FLY_FLAG{};
|
|
constexpr WALL_IS_DOORWAY_FLAG<2> WID_RENDER_FLAG{};
|
|
constexpr WALL_IS_DOORWAY_FLAG<4> WID_RENDPAST_FLAG{};
|
|
constexpr WALL_IS_DOORWAY_FLAG<8> WID_EXTERNAL_FLAG{};
|
|
}
|
|
#if defined(DXX_BUILD_DESCENT_II)
|
|
namespace dsx {
|
|
constexpr WALL_IS_DOORWAY_FLAG<16> WID_CLOAKED_FLAG{};
|
|
}
|
|
#endif
|
|
|
|
namespace dcx {
|
|
// WALL_IS_DOORWAY return values F/R/RP
|
|
constexpr auto WID_WALL = WALL_IS_DOORWAY_sresult(WID_RENDER_FLAG); // 0/1/0 wall
|
|
constexpr auto WID_TRANSPARENT_WALL = WALL_IS_DOORWAY_sresult(WID_RENDER_FLAG | WID_RENDPAST_FLAG); // 0/1/1 transparent wall
|
|
constexpr auto WID_ILLUSORY_WALL = WALL_IS_DOORWAY_sresult(WID_FLY_FLAG | WID_RENDER_FLAG); // 1/1/0 illusory wall
|
|
constexpr auto WID_TRANSILLUSORY_WALL = WALL_IS_DOORWAY_sresult(WID_FLY_FLAG | WID_RENDER_FLAG | WID_RENDPAST_FLAG); // 1/1/1 transparent illusory wall
|
|
constexpr auto WID_NO_WALL = WALL_IS_DOORWAY_sresult(WID_FLY_FLAG | WID_RENDPAST_FLAG); // 1/0/1 no wall, can fly through
|
|
constexpr auto WID_EXTERNAL = WALL_IS_DOORWAY_sresult(WID_EXTERNAL_FLAG); // 0/0/0/1 don't see it, dont fly through it
|
|
}
|
|
#if defined(DXX_BUILD_DESCENT_II)
|
|
namespace dsx {
|
|
constexpr auto WID_CLOAKED_WALL = WALL_IS_DOORWAY_sresult(WID_RENDER_FLAG | WID_RENDPAST_FLAG | WID_CLOAKED_FLAG);
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
namespace dcx {
|
|
|
|
struct stuckobj;
|
|
struct v16_wall;
|
|
struct v19_wall;
|
|
|
|
typedef unsigned wall_clip_flag_t;
|
|
constexpr std::integral_constant<wall_clip_flag_t, 1> WCF_EXPLODES{}; //door explodes when opening
|
|
constexpr std::integral_constant<wall_clip_flag_t, 2> WCF_BLASTABLE{}; //this is a blastable wall
|
|
constexpr std::integral_constant<wall_clip_flag_t, 4> WCF_TMAP1{}; //this uses primary tmap, not tmap2
|
|
constexpr std::integral_constant<wall_clip_flag_t, 8> WCF_HIDDEN{}; //this uses primary tmap, not tmap2
|
|
}
|
|
|
|
#ifdef dsx
|
|
namespace dsx {
|
|
struct wall;
|
|
struct wclip;
|
|
constexpr std::integral_constant<std::size_t, 20> MAX_CLIP_FRAMES_D1{};
|
|
}
|
|
|
|
#if defined(DXX_BUILD_DESCENT_II)
|
|
namespace dsx {
|
|
struct cloaking_wall;
|
|
constexpr std::integral_constant<std::size_t, 10> MAX_CLOAKING_WALLS{};
|
|
using clwallnum_t = uint8_t;
|
|
}
|
|
DXX_VALPTRIDX_DECLARE_SUBTYPE(dsx::, cloaking_wall, clwallnum_t, dsx::MAX_CLOAKING_WALLS);
|
|
namespace dsx {
|
|
DXX_VALPTRIDX_DEFINE_SUBTYPE_TYPEDEFS(cloaking_wall, clwall);
|
|
}
|
|
#endif
|
|
|
|
DXX_VALPTRIDX_DECLARE_SUBTYPE(dsx::, wall, wallnum_t, dcx::MAX_WALLS);
|
|
namespace dsx {
|
|
DXX_VALPTRIDX_DEFINE_SUBTYPE_TYPEDEFS(wall, wall);
|
|
extern array<wclip, MAX_WALL_ANIMS> WallAnims;
|
|
constexpr valptridx<wall>::magic_constant<0xffff> wall_none{};
|
|
}
|
|
|
|
namespace dcx {
|
|
DXX_VALPTRIDX_DEFINE_SUBTYPE_TYPEDEFS(active_door, actdoor);
|
|
#define Num_walls Walls.get_count()
|
|
extern unsigned Num_wall_anims;
|
|
}
|
|
#endif
|
|
|
|
#if DXX_USE_EDITOR
|
|
#ifdef dsx
|
|
namespace dsx {
|
|
void wall_init();
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
#if defined(DXX_BUILD_DESCENT_I) || defined(DXX_BUILD_DESCENT_II)
|
|
// Automatically checks if a there is a doorway (i.e. can fly through)
|
|
#ifdef dsx
|
|
namespace dsx {
|
|
WALL_IS_DOORWAY_result_t wall_is_doorway (const side &side);
|
|
|
|
// Deteriorate appearance of wall. (Changes bitmap (paste-ons))
|
|
}
|
|
#endif
|
|
void wall_damage(vmsegptridx_t seg, int side, fix damage);
|
|
|
|
// Destroys a blastable wall. (So it is an opening afterwards)
|
|
void wall_destroy(vmsegptridx_t seg, int side);
|
|
|
|
void wall_illusion_on(vmsegptridx_t seg, int side);
|
|
void wall_illusion_off(vmsegptridx_t seg, int side);
|
|
|
|
#ifdef dsx
|
|
namespace dsx {
|
|
|
|
// Opens a door
|
|
void wall_open_door(vmsegptridx_t seg, int side);
|
|
|
|
}
|
|
#endif
|
|
#if defined(DXX_BUILD_DESCENT_I)
|
|
#elif defined(DXX_BUILD_DESCENT_II)
|
|
// Closes a door
|
|
void wall_close_door(vmsegptridx_t seg, int side);
|
|
#endif
|
|
|
|
//return codes for wall_hit_process()
|
|
enum class wall_hit_process_t : unsigned
|
|
{
|
|
WHP_NOT_SPECIAL = 0, //wasn't a quote-wall-unquote
|
|
WHP_NO_KEY = 1, //hit door, but didn't have key
|
|
WHP_BLASTABLE = 2, //hit blastable wall
|
|
WHP_DOOR = 3, //a door (which will now be opening)
|
|
};
|
|
|
|
// Determines what happens when a wall is shot
|
|
//obj is the object that hit...either a weapon or the player himself
|
|
#ifdef dsx
|
|
class player_flags;
|
|
namespace dsx {
|
|
wall_hit_process_t wall_hit_process(player_flags, vmsegptridx_t seg, int side, fix damage, int playernum, vmobjptr_t obj);
|
|
|
|
// Opens/destroys specified door.
|
|
}
|
|
#endif
|
|
void wall_toggle(vmsegptridx_t segnum, unsigned side);
|
|
|
|
// Tidy up Walls array for load/save purposes.
|
|
void reset_walls();
|
|
|
|
// Called once per frame..
|
|
#ifdef dsx
|
|
namespace dsx {
|
|
void wall_frame_process();
|
|
|
|
// An object got stuck in a door (like a flare).
|
|
// Add global entry.
|
|
}
|
|
#endif
|
|
void add_stuck_object(vmobjptridx_t objp, vmsegptr_t segnum, int sidenum);
|
|
void remove_obsolete_stuck_objects();
|
|
|
|
//set the tmap_num or tmap_num2 field for a wall/door
|
|
void wall_set_tmap_num(vmsegptridx_t seg,int side,vmsegptridx_t csegp,int cside,int anim_num,int frame_num);
|
|
|
|
// Remove any flares from a wall
|
|
#ifdef dsx
|
|
namespace dsx {
|
|
void kill_stuck_objects(wallnum_t wallnum);
|
|
|
|
}
|
|
#endif
|
|
#if defined(DXX_BUILD_DESCENT_II)
|
|
//start wall open <-> closed transitions
|
|
void start_wall_cloak(vmsegptridx_t seg, int side);
|
|
void start_wall_decloak(vmsegptridx_t seg, int side);
|
|
|
|
void cloaking_wall_read(cloaking_wall &cw, PHYSFS_File *fp);
|
|
void cloaking_wall_write(const cloaking_wall &cw, PHYSFS_File *fp);
|
|
#endif
|
|
void blast_nearby_glass(const object &objp, fix damage);
|
|
#endif
|
|
|
|
void wclip_read(PHYSFS_File *, wclip &wc);
|
|
#if 0
|
|
void wclip_write(PHYSFS_File *, const wclip &);
|
|
#endif
|
|
|
|
void v16_wall_read(PHYSFS_File *fp, v16_wall &w);
|
|
void v19_wall_read(PHYSFS_File *fp, v19_wall &w);
|
|
void wall_read(PHYSFS_File *fp, wall &w);
|
|
|
|
void active_door_read(PHYSFS_File *fp, active_door &ad);
|
|
void active_door_write(PHYSFS_File *fp, const active_door &ad);
|
|
|
|
void wall_write(PHYSFS_File *fp, const wall &w, short version);
|
|
void wall_close_door_ref(active_door &);
|
|
void init_stuck_objects();
|
|
void clear_stuck_objects();
|
|
#endif
|