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

Two-way DOM update and mutation observers #40

Open
Tracked by #28282
dvoytenko opened this issue Nov 1, 2019 · 1 comment
Open
Tracked by #28282

Two-way DOM update and mutation observers #40

dvoytenko opened this issue Nov 1, 2019 · 1 comment
Labels
TBD To be discussed/design decisions

Comments

@dvoytenko
Copy link
Contributor

dvoytenko commented Nov 1, 2019

Context: #29 (comment)

The goal is to find a clear way to separate DOM mutations from external script and components.

Usually we can separate user DOM updates from component updates using light and shadow trees. However, this is not always possible, such as the case of amp-selector component.

Let's consider the amp-selector component. For the amp-selector, all mutations are in the main DOM tree and all observable by the main script/css/etc. E.g. if the CE user wants to decorated a selected option within the amp-selector, the following CSS can be used:

.option[selected] {
  border: 1px solid red;
}

Currently this is virtually the only option for us to signify that a selection option (as a DOM element) is currently selected. This is definitely not great. It'd be much better from API point of view if we could set a custom state, e.g. .option:selected {...}. But custom states spec is still ways and ways away.

As a result, a subtree-based MutationObserver would see these mutations too. In other words the following are both valid code paths that mutate DOM:

  1. In-component: user clicks on an option and the React component sets it as selected. Our Slot delegation updates DOM:
// A. React's onClick handler -> state
function AmpSelection.Option() {
  ...
  (<x onClick={() => setSelectedOptionState(props.option)} ...>)
  ...
}

// B. Slot's props.selected is set in React:
function AmpSelection.Option() {
  ...
  (<Slot selected={isSelectedOptionState(props.option)}>)
  ...
}

// C. Slot's side effect sets DOM attribute:

function Slot() {
  const domRef = useRef();
  useEffect(() => {
    const slot = domRef.current;
    ...
    // Update in the main DOM:
    const assignedOption = slot.assignedElements()[0];
    if (props.selected) {
      assignedOption.setAttribute('selected', '');
    } else {
      assignedOption.removeAttribute('selected');
    }
  });
  ...
}
  1. Out-of-component mutation: a user script in the main document manually writes DOM:
button1.onclick = () => {
  option2.setAttribute('selected', '');
};

Update an attribute this way by the user script will trigger mutation observer and trigger React component re-rendering with the new value prop.

We need the mutation observer to synchronize DOM -> React component in the case /2/. But we don't really need mutation observer for /1/ since we ourselves ensure that DOM/React are in full sync. In general case, incorrectly working /1/ can cause cycles. So far the cycles in such mutations have been easy to work around or ignore. But in general case this is still a dangerous situation. It'd be nice to have a more "automatic" solution for this.

@dvoytenko dvoytenko added the TBD To be discussed/design decisions label Nov 1, 2019
@dvoytenko
Copy link
Contributor Author

dvoytenko commented Dec 16, 2019

A possibly much simpler approach - expected mutation matching:

  1. Some internal mutations are ignored and thus can be done safely by components. These include i-amphtml attributes, --i-amphtml-x CSS vars in style, children with i-amphtml-x class or special attribute, i-amphtml-x attribute values (slot=i-amphtml-slide-1), etc. The expectation is that majority of mutations that need to be done in the light subtree can be done this way.
  2. For other light tree mutations we will require a MutationRecord to be created and pushed for the target element into the "pending set". When the MutationObserver is notified with a mutation, it can match the MutationRecord against the "pending set", and ignore the record if a matching one exists.

For instance, the component's implementation modifies the light tree:

useEffect(() => {
  const el = ref.current;

  // This record will be ignore because the attribute's value
  // starts with "i-amphtml-" prefix.
  el.setAttribute('slot', 'i-amphtml-slot1');

  // The argument is in the `MutationRecord` format.
  // The `applyMutation` will actually apply the mutation, e.g. it will
  // call `el.setAttribute('selected', '')`, but it will also record this
  // mutation record in the "pending set".
  applyMutation({target: el, type: 'attributes', attributeName: 'selected', value: ''});
});

The main idea here is that the light tree mutation are very rare and thus can take a bit more verbosity and indirection to execute to avoid confusion w.r.t. where a mutation comes from.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
TBD To be discussed/design decisions
Projects
None yet
Development

No branches or pull requests

1 participant