Made the EventHandler class tree a bit more branchy. Now disallowing creation of Static* via EventHandler.Create.

This commit is contained in:
ZZYZX 2017-01-22 08:56:57 +02:00
parent 5e53b73d60
commit fb1d55101e
7 changed files with 128 additions and 70 deletions

View file

@ -2,24 +2,29 @@
#include "virtual.h" #include "virtual.h"
#include "r_utility.h" #include "r_utility.h"
DEventHandler* E_FirstEventHandler = nullptr; DStaticEventHandler* E_FirstEventHandler = nullptr;
void E_RegisterHandler(DEventHandler* handler) bool E_RegisterHandler(DStaticEventHandler* handler)
{ {
if (handler == nullptr || handler->ObjectFlags & OF_EuthanizeMe) if (handler == nullptr || handler->ObjectFlags & OF_EuthanizeMe)
return; return false;
if (E_CheckHandler(handler))
return false;
// link into normal list // link into normal list
handler->prev = nullptr; handler->prev = nullptr;
handler->next = E_FirstEventHandler; handler->next = E_FirstEventHandler;
if (handler->next) if (handler->next)
handler->next->prev = handler; handler->next->prev = handler;
E_FirstEventHandler = handler; E_FirstEventHandler = handler;
return true;
} }
void E_UnregisterHandler(DEventHandler* handler) bool E_UnregisterHandler(DStaticEventHandler* handler)
{ {
if (handler == nullptr || handler->ObjectFlags & OF_EuthanizeMe) if (handler == nullptr || handler->ObjectFlags & OF_EuthanizeMe)
return; return false;
if (!E_CheckHandler(handler))
return false;
// link out of normal list // link out of normal list
if (handler->prev) if (handler->prev)
handler->prev->next = handler->next; handler->prev->next = handler->next;
@ -27,60 +32,78 @@ void E_UnregisterHandler(DEventHandler* handler)
handler->next->prev = handler->prev; handler->next->prev = handler->prev;
if (handler == E_FirstEventHandler) if (handler == E_FirstEventHandler)
E_FirstEventHandler = handler->next; E_FirstEventHandler = handler->next;
return true;
}
bool E_CheckHandler(DStaticEventHandler* handler)
{
for (DStaticEventHandler* lhandler = E_FirstEventHandler; lhandler; lhandler = lhandler->next)
if (handler == lhandler) return true;
return false;
} }
void E_MapLoaded() void E_MapLoaded()
{ {
for (DEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next) for (DStaticEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next)
handler->MapLoaded(); handler->MapLoaded();
} }
void E_MapUnloading() void E_MapUnloading()
{ {
for (DEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next) for (DStaticEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next)
handler->MapUnloading(); handler->MapUnloading();
} }
void E_RenderFrame() void E_RenderFrame()
{ {
for (DEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next) for (DStaticEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next)
handler->RenderFrame(); handler->RenderFrame();
} }
void E_RenderCamera() void E_RenderCamera()
{ {
for (DEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next) for (DStaticEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next)
handler->RenderCamera(); handler->RenderCamera();
} }
void E_RenderBeforeThing(AActor* thing) void E_RenderBeforeThing(AActor* thing)
{ {
for (DEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next) for (DStaticEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next)
handler->RenderBeforeThing(thing); handler->RenderBeforeThing(thing);
} }
void E_RenderAfterThing(AActor* thing) void E_RenderAfterThing(AActor* thing)
{ {
for (DEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next) for (DStaticEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next)
handler->RenderAfterThing(thing); handler->RenderAfterThing(thing);
} }
// declarations // declarations
IMPLEMENT_CLASS(DStaticEventHandler, false, false);
IMPLEMENT_CLASS(DStaticRenderEventHandler, false, false);
IMPLEMENT_CLASS(DEventHandler, false, false); IMPLEMENT_CLASS(DEventHandler, false, false);
IMPLEMENT_CLASS(DRenderEventHandler, false, false); IMPLEMENT_CLASS(DRenderEventHandler, false, false);
DEFINE_FIELD_X(RenderEventHandler, DRenderEventHandler, ViewPos); DEFINE_FIELD_X(StaticRenderEventHandler, DStaticRenderEventHandler, ViewPos);
DEFINE_FIELD_X(RenderEventHandler, DRenderEventHandler, ViewAngle); DEFINE_FIELD_X(StaticRenderEventHandler, DStaticRenderEventHandler, ViewAngle);
DEFINE_FIELD_X(RenderEventHandler, DRenderEventHandler, ViewPitch); DEFINE_FIELD_X(StaticRenderEventHandler, DStaticRenderEventHandler, ViewPitch);
DEFINE_FIELD_X(RenderEventHandler, DRenderEventHandler, ViewRoll); DEFINE_FIELD_X(StaticRenderEventHandler, DStaticRenderEventHandler, ViewRoll);
DEFINE_FIELD_X(RenderEventHandler, DRenderEventHandler, FracTic); DEFINE_FIELD_X(StaticRenderEventHandler, DStaticRenderEventHandler, FracTic);
DEFINE_FIELD_X(RenderEventHandler, DRenderEventHandler, Camera); DEFINE_FIELD_X(StaticRenderEventHandler, DStaticRenderEventHandler, Camera);
DEFINE_FIELD_X(RenderEventHandler, DRenderEventHandler, CurrentThing); DEFINE_FIELD_X(StaticRenderEventHandler, DStaticRenderEventHandler, CurrentThing);
DEFINE_ACTION_FUNCTION(DEventHandler, Create) DEFINE_ACTION_FUNCTION(DEventHandler, Create)
{ {
PARAM_PROLOGUE; PARAM_PROLOGUE;
PARAM_CLASS(t, DEventHandler); PARAM_CLASS(t, DStaticEventHandler);
// check if type inherits dynamic handlers
if (!t->IsDescendantOf(RUNTIME_CLASS(DEventHandler)) &&
!t->IsDescendantOf(RUNTIME_CLASS(DRenderEventHandler)))
{
// disallow static types creation with Create()
ACTION_RETURN_OBJECT(nullptr);
}
// generate a new object of this type. // generate a new object of this type.
ACTION_RETURN_OBJECT(t->CreateNew()); ACTION_RETURN_OBJECT(t->CreateNew());
} }
@ -88,9 +111,16 @@ DEFINE_ACTION_FUNCTION(DEventHandler, Create)
DEFINE_ACTION_FUNCTION(DEventHandler, CreateOnce) DEFINE_ACTION_FUNCTION(DEventHandler, CreateOnce)
{ {
PARAM_PROLOGUE; PARAM_PROLOGUE;
PARAM_CLASS(t, DEventHandler); PARAM_CLASS(t, DStaticEventHandler);
// check if type inherits dynamic handlers
if (!t->IsDescendantOf(RUNTIME_CLASS(DEventHandler)) &&
!t->IsDescendantOf(RUNTIME_CLASS(DRenderEventHandler)))
{
// disallow static types creation with Create()
ACTION_RETURN_OBJECT(nullptr);
}
// check if there are already registered handlers of this type. // check if there are already registered handlers of this type.
for (DEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next) for (DStaticEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next)
if (handler->GetClass() == t) // check precise class if (handler->GetClass() == t) // check precise class
ACTION_RETURN_OBJECT(nullptr); ACTION_RETURN_OBJECT(nullptr);
// generate a new object of this type. // generate a new object of this type.
@ -100,8 +130,8 @@ DEFINE_ACTION_FUNCTION(DEventHandler, CreateOnce)
DEFINE_ACTION_FUNCTION(DEventHandler, Find) DEFINE_ACTION_FUNCTION(DEventHandler, Find)
{ {
PARAM_PROLOGUE; PARAM_PROLOGUE;
PARAM_CLASS(t, DEventHandler); PARAM_CLASS(t, DStaticEventHandler);
for (DEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next) for (DStaticEventHandler* handler = E_FirstEventHandler; handler; handler = handler->next)
if (handler->GetClass() == t) // check precise class if (handler->GetClass() == t) // check precise class
ACTION_RETURN_OBJECT(handler); ACTION_RETURN_OBJECT(handler);
ACTION_RETURN_OBJECT(nullptr); ACTION_RETURN_OBJECT(nullptr);
@ -110,17 +140,17 @@ DEFINE_ACTION_FUNCTION(DEventHandler, Find)
DEFINE_ACTION_FUNCTION(DEventHandler, Register) DEFINE_ACTION_FUNCTION(DEventHandler, Register)
{ {
PARAM_PROLOGUE; PARAM_PROLOGUE;
PARAM_OBJECT(handler, DEventHandler); PARAM_OBJECT(handler, DStaticEventHandler);
E_RegisterHandler(handler); if (handler->IsStatic()) ACTION_RETURN_BOOL(false);
return 0; ACTION_RETURN_BOOL(E_RegisterHandler(handler));
} }
DEFINE_ACTION_FUNCTION(DEventHandler, Unregister) DEFINE_ACTION_FUNCTION(DEventHandler, Unregister)
{ {
PARAM_PROLOGUE; PARAM_PROLOGUE;
PARAM_OBJECT(handler, DEventHandler); PARAM_OBJECT(handler, DStaticEventHandler);
E_UnregisterHandler(handler); if (handler->IsStatic()) ACTION_RETURN_BOOL(false);
return 0; ACTION_RETURN_BOOL(E_UnregisterHandler(handler));
} }
#define DEFINE_EVENT_HANDLER(cls, funcname, args) DEFINE_ACTION_FUNCTION(cls, funcname) \ #define DEFINE_EVENT_HANDLER(cls, funcname, args) DEFINE_ACTION_FUNCTION(cls, funcname) \
@ -139,21 +169,21 @@ void cls::funcname(args) \
} \ } \
} }
DEFINE_EVENT_HANDLER(DEventHandler, MapLoaded,) DEFINE_EVENT_HANDLER(DStaticEventHandler, MapLoaded,)
DEFINE_EVENT_HANDLER(DEventHandler, MapUnloading,) DEFINE_EVENT_HANDLER(DStaticEventHandler, MapUnloading,)
DEFINE_EVENT_HANDLER(DEventHandler, RenderFrame,) DEFINE_EVENT_HANDLER(DStaticEventHandler, RenderFrame,)
DEFINE_EVENT_HANDLER(DEventHandler, RenderCamera,) DEFINE_EVENT_HANDLER(DStaticEventHandler, RenderCamera,)
DEFINE_EVENT_HANDLER(DEventHandler, RenderBeforeThing, AActor*) DEFINE_EVENT_HANDLER(DStaticEventHandler, RenderBeforeThing, AActor*)
DEFINE_EVENT_HANDLER(DEventHandler, RenderAfterThing, AActor*) DEFINE_EVENT_HANDLER(DStaticEventHandler, RenderAfterThing, AActor*)
// //
void DEventHandler::OnDestroy() void DStaticEventHandler::OnDestroy()
{ {
E_UnregisterHandler(this); E_UnregisterHandler(this);
DObject::OnDestroy(); DObject::OnDestroy();
} }
void DRenderEventHandler::Setup() void DStaticRenderEventHandler::Setup()
{ {
ViewPos = ::ViewPos; ViewPos = ::ViewPos;
ViewAngle = ::ViewAngle; ViewAngle = ::ViewAngle;
@ -163,26 +193,26 @@ void DRenderEventHandler::Setup()
Camera = ::camera; Camera = ::camera;
} }
void DRenderEventHandler::RenderFrame() void DStaticRenderEventHandler::RenderFrame()
{ {
Setup(); Setup();
DEventHandler::RenderFrame(); DStaticEventHandler::RenderFrame();
} }
void DRenderEventHandler::RenderCamera() void DStaticRenderEventHandler::RenderCamera()
{ {
Setup(); Setup();
DEventHandler::RenderCamera(); DStaticEventHandler::RenderCamera();
} }
void DRenderEventHandler::RenderBeforeThing(AActor* thing) void DStaticRenderEventHandler::RenderBeforeThing(AActor* thing)
{ {
CurrentThing = thing; CurrentThing = thing;
DEventHandler::RenderBeforeThing(thing); DStaticEventHandler::RenderBeforeThing(thing);
} }
void DRenderEventHandler::RenderAfterThing(AActor* thing) void DStaticRenderEventHandler::RenderAfterThing(AActor* thing)
{ {
CurrentThing = thing; CurrentThing = thing;
DEventHandler::RenderAfterThing(thing); DStaticEventHandler::RenderAfterThing(thing);
} }

View file

@ -3,12 +3,14 @@
#include "dobject.h" #include "dobject.h"
class DEventHandler; class DStaticEventHandler;
// register // register
void E_RegisterHandler(DEventHandler* handler); bool E_RegisterHandler(DStaticEventHandler* handler);
// unregister // unregister
void E_UnregisterHandler(DEventHandler* handler); bool E_UnregisterHandler(DStaticEventHandler* handler);
// find
bool E_CheckHandler(DStaticEventHandler* handler);
// called right after the map has loaded (approximately same time as OPEN ACS scripts) // called right after the map has loaded (approximately same time as OPEN ACS scripts)
void E_MapLoaded(); void E_MapLoaded();
@ -23,18 +25,22 @@ void E_RenderBeforeThing(AActor* thing);
// called after adding each actor to the render list // called after adding each actor to the render list
void E_RenderAfterThing(AActor* thing); void E_RenderAfterThing(AActor* thing);
class DEventHandler : public DObject // make it a part of normal GC process class DStaticEventHandler : public DObject // make it a part of normal GC process
{ {
DECLARE_CLASS(DEventHandler, DObject) DECLARE_CLASS(DStaticEventHandler, DObject)
public: public:
DEventHandler* prev; DStaticEventHandler* prev;
DEventHandler* next; DStaticEventHandler* next;
DEventHandler* unregPrev; DStaticEventHandler* unregPrev;
DEventHandler* unregNext; DStaticEventHandler* unregNext;
virtual bool IsStatic() { return true; }
// destroy handler. this unlinks EventHandler from the list automatically. // destroy handler. this unlinks EventHandler from the list automatically.
void OnDestroy() override; void OnDestroy() override;
// this checks if we are /actually/ static, using DObject dynamic typing system.
static bool IsActuallyStatic(PClass* type);
// called right after the map has loaded (approximately same time as OPEN ACS scripts) // called right after the map has loaded (approximately same time as OPEN ACS scripts)
virtual void MapLoaded(); virtual void MapLoaded();
// called when the map is about to unload (approximately same time as UNLOADING ACS scripts) // called when the map is about to unload (approximately same time as UNLOADING ACS scripts)
@ -48,11 +54,17 @@ public:
// called after adding each actor to the render list // called after adding each actor to the render list
virtual void RenderAfterThing(AActor* thing); virtual void RenderAfterThing(AActor* thing);
}; };
extern DEventHandler* E_FirstEventHandler; class DEventHandler : public DStaticEventHandler
class DRenderEventHandler : public DEventHandler
{ {
DECLARE_CLASS(DRenderEventHandler, DEventHandler) DECLARE_CLASS(DEventHandler, DStaticEventHandler) // TODO: make sure this does not horribly break anything
public:
bool IsStatic() override { return false; }
};
extern DStaticEventHandler* E_FirstEventHandler;
class DStaticRenderEventHandler : public DStaticEventHandler
{
DECLARE_CLASS(DStaticRenderEventHandler, DStaticEventHandler)
public: public:
// these are for all render events // these are for all render events
DVector3 ViewPos; DVector3 ViewPos;
@ -91,5 +103,11 @@ public:
private: private:
void Setup(); void Setup();
}; };
class DRenderEventHandler : public DStaticRenderEventHandler
{
DECLARE_CLASS(DRenderEventHandler, DStaticRenderEventHandler) // TODO: make sure this does not horribly break anythings
public:
bool IsStatic() override { return false; }
};
#endif #endif

View file

@ -85,6 +85,7 @@
#include "p_spec.h" #include "p_spec.h"
#include "serializer.h" #include "serializer.h"
#include "virtual.h" #include "virtual.h"
#include "events.h"
#include "gi.h" #include "gi.h"
@ -655,6 +656,7 @@ void G_ChangeLevel(const char *levelname, int position, int flags, int nextSkill
// [RH] Give scripts a chance to do something // [RH] Give scripts a chance to do something
unloading = true; unloading = true;
FBehavior::StaticStartTypedScripts (SCRIPT_Unloading, NULL, false, 0, true); FBehavior::StaticStartTypedScripts (SCRIPT_Unloading, NULL, false, 0, true);
E_MapUnloading();
unloading = false; unloading = false;
STAT_ChangeLevel(nextlevel); STAT_ChangeLevel(nextlevel);

View file

@ -650,7 +650,7 @@ void GLSprite::Process(AActor* thing, sector_t * sector, int thruportal)
if (thing == nullptr) if (thing == nullptr)
return; return;
DRenderEventHandler::AutoThing autoRenderThingEvent(thing); DStaticRenderEventHandler::AutoThing autoRenderThingEvent(thing);
// [ZZ] allow CustomSprite-style direct picnum specification // [ZZ] allow CustomSprite-style direct picnum specification
bool isPicnumOverride = thing->picnum.isValid(); bool isPicnumOverride = thing->picnum.isValid();

View file

@ -39,6 +39,7 @@
#include "d_event.h" #include "d_event.h"
#include "g_level.h" #include "g_level.h"
#include "gstrings.h" #include "gstrings.h"
#include "events.h"
#include "i_system.h" #include "i_system.h"
#include "m_argv.h" #include "m_argv.h"
@ -1442,6 +1443,8 @@ void P_SpawnSpecials (void)
break; break;
} }
} }
// [ZZ] Loading event hook
E_MapLoaded();
// [RH] Start running any open scripts on this map // [RH] Start running any open scripts on this map
FBehavior::StaticStartTypedScripts (SCRIPT_Open, NULL, false); FBehavior::StaticStartTypedScripts (SCRIPT_Open, NULL, false);
} }

View file

@ -1245,7 +1245,7 @@ void R_AddSprites (sector_t *sec, int lightlevel, int fakeside)
if (thing->validcount == validcount) continue; if (thing->validcount == validcount) continue;
thing->validcount = validcount; thing->validcount = validcount;
DRenderEventHandler::AutoThing autoRenderThingEvent(thing); DStaticRenderEventHandler::AutoThing autoRenderThingEvent(thing);
FIntCVar *cvar = thing->GetClass()->distancecheck; FIntCVar *cvar = thing->GetClass()->distancecheck;
if (cvar != NULL && *cvar >= 0) if (cvar != NULL && *cvar >= 0)

View file

@ -1,12 +1,5 @@
class EventHandler : Object native class StaticEventHandler : Object native
{ {
static native EventHandler Create(class<EventHandler> type);
static native EventHandler CreateOnce(class<EventHandler> type);
static native EventHandler Find(class<EventHandler> type);
static native bool Register(EventHandler handler);
static native bool Unregister(EventHandler handler);
virtual native void MapLoaded(); virtual native void MapLoaded();
virtual native void MapUnloading(); virtual native void MapUnloading();
@ -16,7 +9,7 @@ class EventHandler : Object native
virtual native void RenderAfterThing(); virtual native void RenderAfterThing();
} }
class RenderEventHandler : EventHandler native class StaticRenderEventHandler : StaticEventHandler native
{ {
// for frame and camera // for frame and camera
native readonly Vector3 ViewPos; native readonly Vector3 ViewPos;
@ -29,3 +22,15 @@ class RenderEventHandler : EventHandler native
// for thing // for thing
native readonly Actor CurrentThing; native readonly Actor CurrentThing;
} }
class EventHandler : StaticEventHandler native
{
static native StaticEventHandler Create(class<StaticEventHandler> type);
static native StaticEventHandler CreateOnce(class<StaticEventHandler> type);
static native StaticEventHandler Find(class<StaticEventHandler> type);
static native bool Register(StaticEventHandler handler);
static native bool Unregister(StaticEventHandler handler);
}
class RenderEventHandler : StaticRenderEventHandler native { }