_dotnet events

This commit is contained in:
2024-04-03 20:42:37 +03:00
parent 91a6450c94
commit 75d83a2ef3
4 changed files with 98 additions and 9 deletions

View File

@@ -872,6 +872,13 @@ void VisualScriptExecutor::ProcessGroupFunction(Box* boxBase, Node* node, Value&
// Find event binding callback
auto eventBinder = ScriptingEvents::EventsTable.TryGet(Pair<ScriptingTypeHandle, StringView>(eventType, eventName));
if (!eventBinder)
{
const StringAnsiView eventNameAnsi(node->Values[1]);
auto managedEvent = eventType.GetClass()->GetEvent(eventNameAnsi.GetText());
//eventBinder = (void(**)(ScriptingObject*, void*, bool))managedEvent;
}
if (!eventBinder)
{
LOG(Error, "Cannot bind to missing event {0} from type {1}.", eventName, eventTypeName);

View File

@@ -45,6 +45,16 @@ namespace FlaxEngine.Interop
internal uint fieldAttributes;
}
[StructLayout(LayoutKind.Sequential)]
internal struct NativeEventDefinitions
{
internal IntPtr name;
internal ManagedHandle eventHandle;
internal ManagedHandle addMethodHandle;
internal ManagedHandle removeMethodHandle;
internal uint attributes;
}
[StructLayout(LayoutKind.Sequential)]
internal struct NativePropertyDefinitions
{
@@ -368,6 +378,46 @@ namespace FlaxEngine.Interop
*classFieldsCount = fields.Length;
}
[UnmanagedCallersOnly]
internal static void GetClassEvents(ManagedHandle typeHandle, NativeEventDefinitions** classEvents, int* classEventCount)
{
Type type = Unsafe.As<TypeHolder>(typeHandle.Target);
var events = type.GetEvents(BindingFlags.Static | BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic);
var arr = (NativeEventDefinitions*)NativeAlloc(events.Length, Unsafe.SizeOf<NativeEventDefinitions>());
for (int i = 0; i < events.Length; i++)
{
IntPtr ptr = IntPtr.Add(new IntPtr(arr), Unsafe.SizeOf<NativeEventDefinitions>() * i);
ManagedHandle eventHandle = ManagedHandle.Alloc(events[i]);
var raiseMethod = events[i].GetRaiseMethod(true);
var addMethod = events[i].GetAddMethod(true);
var removeMethod = events[i].GetRemoveMethod(true);
var classEvent = new NativeEventDefinitions
{
name = NativeAllocStringAnsi(events[i].Name),
eventHandle = eventHandle,
attributes = (uint)(addMethod?.Attributes ?? removeMethod?.Attributes ?? MethodAttributes.PrivateScope),
};
/*if (raiseMethod != null)
{
classEvent.raiseMethodHandle = GetMethodGCHandle(raiseMethod);
}*/
if (addMethod != null)
{
classEvent.addMethodHandle = GetMethodGCHandle(addMethod);
}
if (removeMethod != null)
{
classEvent.removeMethodHandle = GetMethodGCHandle(removeMethod);
}
Unsafe.Write(ptr.ToPointer(), classEvent);
}
*classEvents = arr;
*classEventCount = events.Length;
}
[UnmanagedCallersOnly]
internal static void GetClassProperties(ManagedHandle typeHandle, NativePropertyDefinitions** classProperties, int* classPropertiesCount)
{

View File

@@ -34,7 +34,7 @@ public:
#if USE_MONO
explicit MEvent(MonoEvent* monoEvent, const char* name, MClass* parentClass);
#elif USE_NETCORE
MEvent(MClass* parentClass, void* handle, const char* name);
MEvent(MClass* parentClass, void* handle, void* addMethodHandle, void* removeMethodHandle, MMethodAttributes attributes, const char* name);
#endif
public:

View File

@@ -241,6 +241,15 @@ struct NativeFieldDefinitions
MFieldAttributes fieldAttributes;
};
struct NativeEventDefinitions
{
const char* name;
void* eventHandle;
void* addMethodHandle;
void* removeMethodHandle;
MMethodAttributes attributes;
};
struct NativePropertyDefinitions
{
const char* name;
@@ -1045,7 +1054,23 @@ const Array<MEvent*>& MClass::GetEvents() const
if (_hasCachedEvents)
return _events;
// TODO: implement MEvent in .NET
ScopeLock lock(BinaryModule::Locker);
if (_hasCachedEvents)
return _events;
NativeEventDefinitions* foundEvents;
int numEvents;
static void* GetClassEventsPtr = GetStaticMethodPointer(TEXT("GetClassEvents"));
CallStaticMethod<void, void*, NativeEventDefinitions**, int*>(GetClassEventsPtr, _handle, &foundEvents, &numEvents);
_events.SetCapacity(numEvents, false);
for (int i = 0; i < numEvents; i++)
{
const NativeEventDefinitions& definition = foundEvents[i];
MEvent* evnt = New<MEvent>(const_cast<MClass*>(this), definition.eventHandle, definition.addMethodHandle, definition.removeMethodHandle, definition.attributes, definition.name);
_events.Add(evnt);
MCore::GC::FreeMemory((void*)definition.name);
}
MCore::GC::FreeMemory(foundEvents);
_hasCachedEvents = true;
return _events;
@@ -1154,26 +1179,33 @@ void MDomain::Dispatch() const
{
}
MEvent::MEvent(MClass* parentClass, void* handle, const char* name)
MEvent::MEvent(MClass* parentClass, void* handle, void* addMethodHandle, void* removeMethodHandle, MMethodAttributes attributes, const char* name)
: _handle(handle)
, _addMethod(nullptr)
, _removeMethod(nullptr)
, _parentClass(parentClass)
, _name(name)
, _hasCachedAttributes(false)
, _hasAddMonoMethod(true)
, _hasRemoveMonoMethod(true)
{
//_raiseMethod = New<MMethod>(parentClass, StringAnsi("raise_" + _name), handle, 1, attributes);
_hasAddMonoMethod = addMethodHandle != nullptr;
if (_hasAddMonoMethod)
_addMethod = New<MMethod>(parentClass, StringAnsi("add_" + _name), addMethodHandle, 1, attributes);
else
_addMethod = nullptr;
_hasRemoveMonoMethod = removeMethodHandle != nullptr;
if (_hasRemoveMonoMethod)
_removeMethod = New<MMethod>(parentClass, StringAnsi("remove_" + _name), removeMethodHandle, 1, attributes);
else
_removeMethod = nullptr;
}
MMethod* MEvent::GetAddMethod() const
{
return nullptr; // TODO: implement MEvent in .NET
return _addMethod;
}
MMethod* MEvent::GetRemoveMethod() const
{
return nullptr; // TODO: implement MEvent in .NET
return _removeMethod;
}
bool MEvent::HasAttribute(MClass* monoClass) const