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

fix(overlay): returning only the first element for slot assignedElements #4930

Open
wants to merge 3 commits into
base: main
Choose a base branch
from

Conversation

zolar
Copy link

@zolar zolar commented Nov 11, 2024

Only return first element form assignedElements instead of an array of elements.

Description

This PR reverses a helper method that was used in the previous version of the OverLayTrigger. In that version, the slotchange handler uses assignedNodes instead of assignedElements. I think using assignedElements renders the slot so fast, that our dialog component has not completed loading, leading to forcing a reload, in which it does goes threw the same process.

private extractSlotContentFromEvent(event: Event): HTMLElement | undefined {

Related issue(s)

[Bug]: OverlayTrigger can get stuck in a render loop causing page to crash

Motivation and context

Resolves an issue related to endless render loop.

How has this been tested?

  • Did it pass in Desktop?
  • Did it pass in Mobile?
  • Did it pass in iPad?

Screenshots (if appropriate)

N/A

Types of changes

  • Bug fix (non-breaking change which fixes an issue)
  • New feature (non-breaking change which adds functionality)
  • Breaking change (fix or feature that would cause existing functionality to change)
  • Chore (minor updates related to the tooling or maintenance of the repository, does not impact compiled assets)

Checklist

  • I have signed the Adobe Open Source CLA.
  • My code follows the code style of this project.
  • If my change required a change to the documentation, I have updated the documentation in this pull request.
  • I have read the CONTRIBUTING document.
  • I have added tests to cover my changes.
  • All new and existing tests passed.
  • I have reviewed at the Accessibility Practices for this feature, see: Aria Practices

Best practices

This repository uses conventional commit syntax for each commit message; note that the GitHub UI does not use this by default so be cautious when accepting suggested changes. Avoid the "Update branch" button on the pull request and opt instead for rebasing your branch against main.

@zolar zolar requested a review from a team as a code owner November 11, 2024 18:20
@Rajdeepc
Copy link
Contributor

We would like to see you use case here considering this is very specific. Can you add a video or a repro to show how does this solve your issue?

@zolar
Copy link
Author

zolar commented Nov 26, 2024

We would like to see you use case here considering this is very specific. Can you add a video or a repro to show how does this solve your issue?

@Rajdeepc Yes, I can add a video or we can do another debug session so I can walk threw it with you or you team?

@zolar
Copy link
Author

zolar commented Nov 27, 2024

@Rajdeepc This is a video of the component not rendering. The component container stays blank and you can see the console output going forever. It will eventually crash the browser.

Sequence.low.02.mp4

@Rajdeepc
Copy link
Contributor

@Rajdeepc This is a video of the component not rendering. The component container stays blank and you can see the console output going forever. It will eventually crash the browser.

Sequence.low.02.mp4

This seems to be a condition where the entire component is re-rendering multiple times due to state or property changes. Can you do a repro of your use case by any means?

Copy link

changeset-bot bot commented Nov 27, 2024

⚠️ No Changeset found

Latest commit: 38d3f0e

Merging this PR will not cause a version bump for any packages. If these changes should not result in a new version, you're good to go. If these changes should result in a version bump, you need to add a changeset.

Click here to learn what changesets are, and how to add one.

Click here if you're a maintainer who wants to add a changeset to this PR

@zolar
Copy link
Author

zolar commented Nov 27, 2024

@Rajdeepc This is a video of the component not rendering. The component container stays blank and you can see the console output going forever. It will eventually crash the browser.
Sequence.low.02.mp4

This seems to be a condition where the entire component is re-rendering multiple times due to state or property changes. Can you do a repro of your use case by any means?

@Rajdeepc
This issue is only repoing in Express app. We have an example / development application in which we do not see this issue. I think that the issue has to do with Express being so resource intensive. I think this is a race condition in which the Dialog is not completely finished rendering when the slothandler attaches to the slot content, which forces it to rerender. I think that using he nodes.find delays in just enough that the Dialog completes rendering. The issue can also be fixed using the current slot.assignedElements and a zero wait promise new Promise(resolve => setTimeout(resolve, 0));.

Express does use Service Workers but its still very resource heavy and I think its effecting the rendering of our component. We also use a lot of OverlayTriggers that might effect render changes also.

@zolar
Copy link
Author

zolar commented Dec 10, 2024

@Rajdeepc This is a video of the component not rendering. The component container stays blank and you can see the console output going forever. It will eventually crash the browser.
Sequence.low.02.mp4

This seems to be a condition where the entire component is re-rendering multiple times due to state or property changes. Can you do a repro of your use case by any means?

@Rajdeepc This only happens in Express, it does not repo in our development at or any other environment that I know of. There maybe a late state change that is happening but its not from a map or any other iterator that I could find. I think its just a normal state change or its being delayed by the slag in the process.

@coveralls
Copy link
Collaborator

coveralls commented Dec 11, 2024

Pull Request Test Coverage Report for Build 12362748808

Details

  • 30 of 30 (100.0%) changed or added relevant lines in 1 file are covered.
  • 12 unchanged lines in 1 file lost coverage.
  • Overall coverage decreased (-0.03%) to 98.178%

Files with Coverage Reduction New Missed Lines %
packages/overlay/src/LongpressController.ts 12 91.67%
Totals Coverage Status
Change from base Build 12278981558: -0.03%
Covered Lines: 32999
Relevant Lines: 33435

💛 - Coveralls

@zolar zolar force-pushed the zolar/4689-overlay-trigger-update branch from 5bb9e8b to 38d3f0e Compare December 16, 2024 22:51
Copy link
Contributor

@Rajdeepc Rajdeepc left a comment

Choose a reason for hiding this comment

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

Can you please make sure that the PR is clean from any css related changes on radio and search!

return slot.assignedElements({ flatten: true }) as HTMLElement[];
const nodes = slot.assignedNodes({ flatten: true });
return [
nodes.find((node) => node instanceof HTMLElement),
Copy link
Contributor

Choose a reason for hiding this comment

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

The changed implementation is different from what we used to deliver. This method will return on the instance of the first element. Whereas our prev implementation signified that we return all the elements. As you said there can be race condition happening on the express side and the Dialog is waiting for it to mount on page causing successive re-renders. nodes.find is a memory intensive implementation which works for you since it delays your render and the issue gets solved. But I don't feel this is a right approach to do this change on our side since there can be instances where the expected returned elements should be an array and not the first element.
A possible improvement: Adding a check or delay to ensure the slot is ready

private getAssignedElementsFromSlot(slot: HTMLSlotElement): HTMLElement[] {
    const nodes = slot.assignedNodes({ flatten: true });
    return nodes.filter((node): node is HTMLElement => node instanceof HTMLElement);
}

@Rajdeepc
Copy link
Contributor

Rajdeepc commented Jan 2, 2025

@zolar Also please check the tests which are failing in overlay-trigger-longpress.test.ts line 222 which needs to flatten to

  await waitUntil(() => {
      const localName = (
          this.el as unknown as { targetContent: HTMLElement }
      ).targetContent?.localName;

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.

3 participants