Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Proper prototype handling #60

Open
Ablu opened this issue Oct 5, 2018 · 5 comments
Open

Proper prototype handling #60

Ablu opened this issue Oct 5, 2018 · 5 comments

Comments

@Ablu
Copy link
Collaborator

Ablu commented Oct 5, 2018

Problem

Currently JavaScript.Net does not fully support prototype related use cases:

While it is possible to register a function which creates a new object like this:

_context.SetParameter("TestClass", new Func<TestClass>(() => new TestClass()));

_context.Run(@"
    let c = new TestClass();
    c.foo()").Should().Be(true);

problems with this approach:

  • instanceOf checks are not working. c instanceof TestClass is false
  • Prototype extensions are not working (TestClass.prototype.newMethod = function() {...})

Things to consider for the solution

  • Objects which are created by C# code should also be able to pass to JavaScript under an existing prototype
  • Fine grained control about which types are constructable by JS is required

Solution proposal

  • add method like SetConstructor(ConstructorInfo constructor) to the context which generates a prototype and registers under the name
  • when wrapping an object:
    • lookup whether a prototype already exists for the type
    • if no prototype is known: generate a prototype (but do not make it available to the context under a name)
    • wrap object by using the newly generated or existing prototype

Side notes

This could also improve performance of wrapping objects since the analysis of type is only required once, repetitive passing could become a lot faster.


/cc @spahnke

@oliverbock do you think the proposal above is sane? Then I would start to implement it.

@oliverbock
Copy link
Contributor

Sounds hard to implement, but useful. What will ConstructorInfo contain? Can you distinguish between the C# class instance (maps to the JavaScript object) and the C# method definitions (maps to the JavaScript object's prototype)?

@Ablu
Copy link
Collaborator Author

Ablu commented Oct 5, 2018

With ConstructorInfo I meant https://docs.microsoft.com/en-us/dotnet/api/system.reflection.constructorinfo?view=netframework-4.7.2. So I get full access to the type via DeclaringType.

@Ablu
Copy link
Collaborator Author

Ablu commented Oct 5, 2018

I will break it into small self-contained tasks:

  1. change the current wrapping logic to be prototype based
  2. reuse prototypes for same types
  3. add actual support for JS called constructors

This hopefully keeps the individual PRs reviewable and small(er)

Ablu pushed a commit to Ablu/Javascript.Net that referenced this issue Jan 22, 2019
Ablu pushed a commit to Ablu/Javascript.Net that referenced this issue Jan 29, 2019
This allows to register a function (delegate in .NET) as constructor
function in JavaScript.
In order to support this a new method `SetConstructor` was introduced
which allows to register a constructor function for a specific type
under a name. This fuction is then available to the executed JavaScript
code and invoking `new` with this function results in the objects being
registered as `instanceof` the registered function.

Additionally, objects which are created in .NET code are also converted
to V8 while preserving the `instanceof` relation if the type is
registered.

Part of JavascriptNet#60
@Ablu
Copy link
Collaborator Author

Ablu commented Jan 29, 2019

It was simpler than anticipated before. The linked PR (#71) should fix/add this. The only thing which might need more work is inheritance on .NET side... However, I guess that could be handled as a separate issue (not sure whether we need it at the moment)?

@Ablu
Copy link
Collaborator Author

Ablu commented Jan 29, 2019

Also we just realized that overloaded constructors would need additional support

Ablu pushed a commit to Ablu/Javascript.Net that referenced this issue Jan 29, 2019
This allows to register a function (delegate in .NET) as constructor
function in JavaScript.
In order to support this a new method `SetConstructor` was introduced
which allows to register a constructor function for a specific type
under a name. This fuction is then available to the executed JavaScript
code and invoking `new` with this function results in the objects being
registered as `instanceof` the registered function.

Additionally, objects which are created in .NET code are also converted
to V8 while preserving the `instanceof` relation if the type is
registered.

Part of JavascriptNet#60
spahnke pushed a commit to spahnke/Javascript.Net that referenced this issue Feb 1, 2019
This allows to register a function (delegate in .NET) as constructor
function in JavaScript.
In order to support this a new method `SetConstructor` was introduced
which allows to register a constructor function for a specific type
under a name. This fuction is then available to the executed JavaScript
code and invoking `new` with this function results in the objects being
registered as `instanceof` the registered function.

Additionally, objects which are created in .NET code are also converted
to V8 while preserving the `instanceof` relation if the type is
registered.

Part of JavascriptNet#60
Ablu pushed a commit to Ablu/Javascript.Net that referenced this issue Feb 19, 2019
This allows to register a function (delegate in .NET) as constructor
function in JavaScript.
In order to support this a new method `SetConstructor` was introduced
which allows to register a constructor function for a specific type
under a name. This fuction is then available to the executed JavaScript
code and invoking `new` with this function results in the objects being
registered as `instanceof` the registered function.

Additionally, objects which are created in .NET code are also converted
to V8 while preserving the `instanceof` relation if the type is
registered.

Part of JavascriptNet#60
Ablu pushed a commit to Ablu/Javascript.Net that referenced this issue Feb 19, 2019
This allows to register a function (delegate in .NET) as constructor
function in JavaScript.
In order to support this a new method `SetConstructor` was introduced
which allows to register a constructor function for a specific type
under a name. This fuction is then available to the executed JavaScript
code and invoking `new` with this function results in the objects being
registered as `instanceof` the registered function.

Additionally, objects which are created in .NET code are also converted
to V8 while preserving the `instanceof` relation if the type is
registered.

Part of JavascriptNet#60
oliverbock pushed a commit that referenced this issue Feb 25, 2019
This allows to register a function (delegate in .NET) as constructor
function in JavaScript.
In order to support this a new method `SetConstructor` was introduced
which allows to register a constructor function for a specific type
under a name. This fuction is then available to the executed JavaScript
code and invoking `new` with this function results in the objects being
registered as `instanceof` the registered function.

Additionally, objects which are created in .NET code are also converted
to V8 while preserving the `instanceof` relation if the type is
registered.

Part of #60
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants