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

Support inline JavaScript events (v2) #1290

Open
wants to merge 22 commits into
base: develop
Choose a base branch
from

Conversation

shawncrawley
Copy link
Contributor

@shawncrawley shawncrawley commented Apr 1, 2025

Description

This is an alternative solution to PR #1289, so only one of them need be merged. I created this separately to better compare approaches. This approach modifies the VDOM schema to include a new jsExecutables property alongside eventHandlers.

Checklist

Please update this checklist as you complete each item:

  • Tests have been developed for bug fixes or new functionality.
  • The changelog has been updated, if necessary.
  • Documentation has been updated, if necessary.
  • GitHub Issues closed by this PR have been linked.

By submitting this pull request I agree that all contributions comply with this project's open source license(s).

@shawncrawley shawncrawley marked this pull request as ready for review April 1, 2025 16:22
@shawncrawley
Copy link
Contributor Author

This is ready to be looked at - again, aside from the changelog.

I can get this to work "out of the box" with string_to_reactpy if I simply loosen the "on" event listener regex to not expect camelCase, which is usually how raw HTML will be written. The question is if we should coerce all properties starting with "on" to camelCase on the Python/Vdom side (e.g. "onclick" to "onClick"). Right now I'm not doing that, which is how I've written the string_to_reactpy test to expect the Vdom result.

I ultimately assumed that coercing them to camelCase on the Vdom side is unnecessary overhead. Let me know if you agree.

@Archmonger
Copy link
Contributor

onclick should already be automatically mutated into onClick by string_to_reactpy at this LOC.

@shawncrawley
Copy link
Contributor Author

All comments have been addressed, changelog is updated... This PR should be ready to go.

Copy link
Contributor

@Archmonger Archmonger left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can be merged after handling a few more nitpicks

Comment on lines +194 to +195
([name, inlineJavaScript]) =>
createInlineJavascript(name, inlineJavaScript),
Copy link
Contributor

@Archmonger Archmonger Apr 4, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Minor nitpick about capitalization consistency. Either choose Javascript or JavaScript. This same sentiment applies to every use of javascript within this PR.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good catch - I didn't notice I had been inconsistent myself. I'll go with JavaScript.

Comment on lines -40 to +52
def html_props_to_reactjs(vdom: VdomDict) -> None:
"""Convert HTML prop names to their ReactJS equivalents."""
if "attributes" in vdom:
items = cast(VdomAttributes, vdom["attributes"].items())
vdom["attributes"] = cast(
VdomAttributes,
{REACT_PROP_SUBSTITUTIONS.get(k, k): v for k, v in items},
)
def _attributes_to_reactjs(attributes: VdomAttributes):
"""Convert HTML attribute names to their ReactJS equivalents.
This method is private because it is called prior to instantiating a
Vdom class from a parsed html string, so it does not need to be called
as part of this class's instantiation (see comments in __init__ above).
"""
attrs = cast(VdomAttributes, attributes.items())
attrs = cast(
VdomAttributes,
{REACT_PROP_SUBSTITUTIONS.get(k, k): v for k, v in attrs},
)
return attrs
Copy link
Contributor

@Archmonger Archmonger Apr 4, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Revert this function and create a separate transform function within RequiredTransforms that performs the InlineJavascript transforms.

The cadence set within RequiredTransforms is that it never needs any external calls, it just "does everything" automatically during initialization.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure what to do here. I know about the RequiredTransforms cadence due to the comment in its __init__ function, but to handle the "onclick" vs "onClick", "attributes" vs "inlineJavaScript", and ^on\w+ vs ^on[A-Z]\w+ dilemma (which we decided to stick with camelCase regex), I have to convert the inline html attributes to camelCase prior to instantiating the Vdom object, which is also prior to calling the RequiredTransforms - since these transforms all operate on Vdom object. Note the call order below:

image

I injected the camelCase transform to occur on the parsed HTML etree object (again, prior to it being used to instantiate the Vdom object and applying the standard RequiredTransforms. And if conversion to camelCase happens prior to Vdom instantiation, then there's no need to execute it again after. That is why I deleted the old function and just created a new "private" one, where it's only private so that it won't be invoked within the official cadence.

Does this makes sense? So again, I'm a bit confused. Do you want me to just restore the old function in addition to keeping the new one while maintaining the call order I've implemented? In which case it will perform a camelCase conversion twice - once before Vdom and once after.

@Archmonger Archmonger mentioned this pull request Apr 4, 2025
4 tasks
Comment on lines +231 to +245
const wrappedExecutable = function (...args: any[]) {
function handleExecution(...args: any[]) {
const evalResult = eval(inlineJavaScript);
if (typeof evalResult == "function") {
return evalResult(...args);
}
}
if (args.length > 0 && args[0] instanceof Event) {
return handleExecution.call(args[0].currentTarget, ...args);
} else {
return handleExecution(...args);
}
};
wrappedExecutable.isHandler = false;
return [name, wrappedExecutable];
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since there's a couple of logical paths here, some comments would be appreciated on this function

Comment on lines +898 to +903
class JavaScript(str):
"""Simple subclass that flags a user's string in ReactPy VDOM attributes as executable JavaScript."""

pass


Copy link
Contributor

@Archmonger Archmonger Apr 4, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should rename this to InlineJavaScript for consistency. Would also help users understand that JavaScript is not an "executor function", and as such the following scenario is not valid syntax:

@component
def my_example():
    def on_click(event):
        result = JavaScript("this.innerText = 'hello'")
    ...

@Archmonger Archmonger changed the title Support raw JavaScript events - v2, w/VDOM schema change approach Support inline JavaScript event (v2) Apr 4, 2025
@Archmonger Archmonger changed the title Support inline JavaScript event (v2) Support inline JavaScript events (v2) Apr 4, 2025
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

Successfully merging this pull request may close these issues.

2 participants