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

feat: add support for mixed html/js content #128

Merged

Conversation

delca85
Copy link
Member

@delca85 delca85 commented Jun 10, 2020

Working on #10 , I have enabled:

  1. html mixed mode in MarkupEditor, syntax highlighting working even for js script now
  2. script tag are now retrieved in Preview component and evaluated through window.eval. Before markup is passed as Scrollable child in dangerouslySetInnerHtml all the script elements are removed.

Right now, this kind of code:

<label for="checkbox">Check</label>
<input
  id="checkbox"
  type="checkbox"
/>
<button id="btn">
  Accept
</button>

<button id="btn2">
Decline
</button>

<script>
  const btn = document.getElementById('btn');
  btn.addEventListener('click', () => {
    alert('click');
  });
</script>

<script>
  const btn2 = document.getElementById('btn2');
  btn2.addEventListener('click', () => {
    alert('click no');
  });
</script>

can be inserted by the user and the result is an interactive Preview panel.

Copy link
Member

@smeijer smeijer left a comment

Choose a reason for hiding this comment

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

The branch is looking great! It's already very useful. But I do have a few concerns. See the two comments below.

Also, the scripts aren't evaluated on page load. When pressing F5 or when opening a reply from a shared link, I first need to edit the markdown manually, to trigger the evals.

() =>
scripts.length
? scripts.reduce(
(html, script) => html.replace(`<script>${script}</script>`, ''),
Copy link
Member

Choose a reason for hiding this comment

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

This doesn't remove script elements that specify the type attribute. Type attributes for script elements default to text/javascript, but it's still a valid attribute to provide. 🙂

Copy link
Member Author

Choose a reason for hiding this comment

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

Yes, totally right! I am going to fix it!

const container = document.createElement('div');
container.innerHTML = markup;
const scriptsCollections = container.getElementsByTagName('script');
return Array.from(scriptsCollections).map((script) => script.innerHTML);
Copy link
Member

Choose a reason for hiding this comment

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

This also parses script elements of other types than javascript. For example <script type="text/css">. Type attributes for script elements default to text/javascript, so we should only extract script elements without a type attribute, or with a type === "text/javascript"

<script type="text/javascript">
  document.getElementById('btn').addEventListener('click', () => {
    alert('click');
  });
</script>

See also this repl.

Copy link
Member Author

Choose a reason for hiding this comment

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

Got it! I am going to fix it again!

@delca85
Copy link
Member Author

delca85 commented Jun 11, 2020

Regarding the first comment, the one about the missed evaluation on page loading maybe I have some idea to fix it. I should try!

@delca85
Copy link
Member Author

delca85 commented Jun 11, 2020

I should have fixed the issues you highlighted.
Regarding the problem that no script was evaluated on component mount, the problem was that useEffect runs before Scrollable children are rendered so I added a check on htmlRoot.current to make the effect runs again when children are rendered.
Furthermore I replaced usage of eval with new Function which should be a little safer.

A remaining issue is that if the user modifies the script code in MarkupEditor, the script is evaluate one time for each inserted char and this behavior affects the user experience. I am considering the idea of throttling the evaluation call but I would be glad to hear your thoughts.

@delca85
Copy link
Member Author

delca85 commented Jun 12, 2020

I have moved try catch block inside the forEach function, because in the first version if a script failed all the scripts evaluation were be aborted. I added an alert to notify the user that he has inserted a failing script but it has just an idea!

@delca85
Copy link
Member Author

delca85 commented Jun 12, 2020

I have prepared a fix to keep track of the already evaluated script, I could push it too if it's ok to you.

@delca85
Copy link
Member Author

delca85 commented Jun 12, 2020

I am going to push it because I won't be able to do that during this morning, if that fix doesn't sound good to you I will turn it back!

@delca85
Copy link
Member Author

delca85 commented Jun 13, 2020

Hi @smeijer ! I don't wanna to hurry you, but reading again my comments worry me that I haven't been so clear. I already pushed all the code I have for this feature and I would like to receive your thoughts.
(reading my comments made me wonder if you were still waiting for some code)

@smeijer
Copy link
Member

smeijer commented Jun 14, 2020

Hi @delca85.

I'll do my best to check it today. Thanks for the reminder 👍

@smeijer smeijer changed the title Feature/supporting html mixed mode and interactive preview panel feat: add support for mixed html/js content Jun 14, 2020
@smeijer smeijer merged commit 85c240c into testing-library:develop Jun 14, 2020
@smeijer
Copy link
Member

smeijer commented Jun 14, 2020

@delca85, Thanks so much for your help!

I've added you as a 🐸 collaborator on the project. Please make sure that you review the MAINTAINING.md and CONTRIBUTING.md files (specifically the bit about the commit messages and the git hooks) and familiarize yourself with the code of conduct (we're using the contributor covenant).

You might also want to watch the repo to be notified when someone files an issue/PR. Please continue to make PRs as you feel the need (you can make your branches directly on the repo rather than your fork if you want).

Thanks! And welcome to the team 🎉

@delca85
Copy link
Member Author

delca85 commented Jun 14, 2020

Wow! Thank you @smeijer! I am really really happy and excited about it. I'll keep on doing my best to collaborate to this awesome project. 🎉 🥳

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