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

[lexical][lexical-react][lexical-table] Fix: Add horizontal scroll for tables #6713

Closed
wants to merge 17 commits into from

Conversation

GermanJablo
Copy link
Contributor

Description

Closes #6480

Having tables overflow is a mistake, as their content can conflict with other elements of the interface:

image

Another incorrect behavior is the one that occurs in the Lexical Playground, where if the table is too large, the entire editor becomes larger. The horizontal scroll bar is not placed over the table, but over the entire content.

image

A super quick way to fix overflow is to add a max-width of 100% to the table.

The problem is, as this user noted, that this creates conflicts with the column width property.

This is what OneNote does and I've always hated it. Sometimes columns resize in a magical or hard-to-understand way. It's better to give the user the flexibility to adjust column widths to their liking and to have overflow if needed.

The correct solution to this is for tables to have an overflow scroll independent of the rest of the editor, similar to what Notion does:

image

The problem is that the <table> tag doesn't support overflow: scroll. A div is needed to wrap it.

This PR introduces a new Scrollable node that automatically wraps tables.

BREAKING CHANGE: TablePlugin now requires the editor to also register the ScrollableNode node. All tables are automatically wrapped in a ScrollableNode.

Test plan

Before

image

After

image

Copy link

vercel bot commented Oct 7, 2024

The latest updates on your projects. Learn more about Vercel for Git ↗︎

Name Status Preview Comments Updated (UTC)
lexical ✅ Ready (Inspect) Visit Preview 💬 Add feedback Nov 6, 2024 2:03am
lexical-playground ✅ Ready (Inspect) Visit Preview 💬 Add feedback Nov 6, 2024 2:03am

@facebook-github-bot facebook-github-bot added the CLA Signed This label is managed by the Facebook bot. Authors need to sign the CLA before a PR can be reviewed. label Oct 7, 2024
Copy link

github-actions bot commented Oct 7, 2024

size-limit report 📦

Path Size
lexical - cjs 29.92 KB (0%)
lexical - esm 29.78 KB (0%)
@lexical/rich-text - cjs 38.57 KB (0%)
@lexical/rich-text - esm 31.63 KB (0%)
@lexical/plain-text - cjs 37.22 KB (0%)
@lexical/plain-text - esm 28.94 KB (0%)
@lexical/react - cjs 40.33 KB (0%)
@lexical/react - esm 33.06 KB (0%)

Copy link
Collaborator

@etrepum etrepum left a comment

Choose a reason for hiding this comment

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

I don't have time for QA right now or to audit the tests but I took a quick look at the code. Seems like a useful improvement! The downside is that it is a bit backwards incompatible due to the additional node and requirements for table use. Would maybe be a bit nicer if it was done in a way that could be backwards compatible. @ivailop7 should have a look

@ivailop7
Copy link
Collaborator

Will give this a read this evening!

@etrepum
Copy link
Collaborator

etrepum commented Oct 16, 2024

@ivailop7 an assessment on this one sooner than later would be good, since it changes the DOM of all tables it will generate conflicts pretty quickly

@etrepum
Copy link
Collaborator

etrepum commented Oct 17, 2024

We discussed this PR in the weekly meeting today and it seems very promising and overall the right direction given the current constraints in the reconciler. However, the backwards compatibility issue makes it very tricky to accept as-is because it would likely cause some problems with existing deployments (particularly those using yjs, rolling out with multiple concurrent versions of lexical, etc.). It would be ideal if there was a way to preserve the possibility to deploy table nodes as-is without the ScrollableNode wrapper and then gracefully upgrade somehow to include the wrapper.

We think it will be possible to make the ScrollableNode opt-in by moving the transforms from static methods to separately exported transform functions that need to be registered explicitly with the editor (this could be abstracted with a register function that does a mergeRegister of all necessary transforms and a React plugin to make it easier to install). This would also allow future usage of ScrollableNode with other types of nodes as the logic would be in a function that can be replaced (e.g. by parameterizing the $isTableNode predicate).

It's acceptable that the tests will expect that this plugin is already registered so that work wouldn't have to be undone, but we'd probably want to add some minimal tests to show that some table functionality still works without the opt-in ScrollableNode.

Overall, if we can get this into an opt-in state to make the upgrade path smooth then we are onboard with merging this quickly!

Maybe something along these lines (with $normalizePoint__EXPERIMENTAL exported from LexicalNormalization like $normalizeSelection is?)

export interface ScrollableNodeConfig {
  scrollableChildNodes: readonly Klass<ElementNode>[];
  $isScrollableChild?: (node: LexicalNode | null) => boolean;
}

export function registerScrollableNodeTransform(
  editor: LexicalEditor,
  config: ScrollableNodeConfig,
): () => void {
  invariant(
    editor.hasNodes([ScrollableNode]),
    'TablePlugin: ScrollableNode not registered on editor',
  );
  const {
    scrollableChildNodes,
    $isScrollableChild = (node) =>
      node !== null &&
      scrollableChildNodes.some((klass) => node instanceof klass),
  } = config;
  return mergeRegister(
    editor.registerNodeTransform(ScrollableNode, (node) => {
      let onlyScrollableChild: LexicalNode | null = null;
      for (const child of node.getChildren()) {
        if (onlyScrollableChild === null && $isScrollableChild(child)) {
          onlyScrollableChild = child;
        } else {
          child.remove();
        }
      }
      if (onlyScrollableChild === null) {
        node.remove();
        const root = $getRoot();
        if (root.isEmpty()) {
          root.append($createParagraphNode());
        }
        return;
      }
      const selection = $getSelection();
      if (!selection) {
        return;
      }
      const nodeKey = node.getKey();
      for (const point of selection.getStartEndPoints() || []) {
        if (point.key === nodeKey) {
          $normalizePoint__EXPERIMENTAL(point);
        }
      }
    }),
    ...scrollableChildNodes.map((klass) =>
      editor.registerNodeTransform(klass, (node) => {
        const parent = node.getParent();
        if (!$isScrollableNode(parent)) {
          const scrollable = $createScrollableNode();
          node.insertBefore(scrollable);
          scrollable.append(node);
        }
      }),
    ),
  );
}

@etrepum
Copy link
Collaborator

etrepum commented Oct 18, 2024

I added a commit here to show what I am proposing #6743 42bf4f8

@GermanJablo
Copy link
Contributor Author

Thanks for the review @etrepum. Everything you say makes sense.

The biggest difficulty will be not in moving the transforms to a plugin, but in maintaining the 2 versions of LexicalTableSelectionHelpers.ts. I'll see if I can find a viable way to achieve this.

@etrepum
Copy link
Collaborator

etrepum commented Oct 18, 2024

Maybe the best way to do that would be to instead override TableNode.selectNext, TableNode.selectPrevious, TableNode.insertBefore, etc. to check the parent and delegate if necessary?

I guess the insertBefore/insertAfter stuff could maybe be fixed in a more robust way by having the ScrollableNode transform move non-scrollable elements before or after itself instead of removing them

@GermanJablo
Copy link
Contributor Author

Hmm, you would also need to extend getNextSibling, getPreviousSibling, getParent, getParentOrThrow.

Also, that wouldn't solve every scenario. For example if a paragraph does getNextSibling and it's a scrollableNode?

@etrepum
Copy link
Collaborator

etrepum commented Oct 18, 2024

I don't think that this code does any better at solving all of those scenarios, and I don't think that you would want the get* methods to lie to you. Updating the select* methods are just a form of preemptive normalization of the selection which is useful because necessary information is lost. You don't know whether the selection point should be left or right biased by the time that selectNext or selectPrevious returns.

@GermanJablo
Copy link
Contributor Author

I cherry picked your commit, and modified LexicalTableSelectionHelpers to modify its behavior depending on whether ScrollableNode is registered or not (the last commit so far).

The next step would be to retrieve the initial tests for cases where it is not registered. But I'm not sure what the best way to do that is.

I've made a PoC to make its registration configurable in the bottom left options menu in the playground. The commit is in another branch: c788896

I see that with richtext/plaintext this + environment variables are used. But I don't know if I'm making it more complicated than necessary. In this case, it is only necessary to run a subset of tests with both configurations.

Copy link
Collaborator

@etrepum etrepum left a comment

Choose a reason for hiding this comment

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

This looks like a good approach to me. I'm not personally that concerned about having another toggle in the playground settings for separate e2e testing of scrollable and non-scrollable tables. I think a small suite of unit tests that build an editor without the scrollable plugin would be fine to demonstrate that things still work without it.

Comment on lines +82 to +85
return $getEditor().hasNode(ScrollableNode)
? (tableNode.getParentOrThrow() as ScrollableNode)
: tableNode;
}
Copy link
Collaborator

Choose a reason for hiding this comment

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

It might be cleaner to put an $isScrollableNode check on the parent instead of using the editor config, that way you could toggle the behavior using only the plugin.

Comment on lines +119 to +123
child.remove();
}
}
if (onlyScrollableChild === null) {
node.remove();
Copy link
Collaborator

Choose a reason for hiding this comment

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

based on the other discussions I think that these children should be moved out of the scrollable node either before or after the scrollable node depending on whether there's a child or not

packages/lexical-table/src/LexicalTableSelectionHelpers.ts Outdated Show resolved Hide resolved
packages/lexical-react/src/LexicalTablePlugin.ts Outdated Show resolved Hide resolved
@GermanJablo
Copy link
Contributor Author

Sorry for the delay on this. I'm getting back to this PR and will try to finish it ASAP.

@etrepum
Copy link
Collaborator

etrepum commented Nov 5, 2024

I did mock up an implementation based on this in 351dc2f (#6759) using a new proposed experimental getDOMSlot API which would allow the TableNode to control where its children are rendered. I didn't write any tests but it seemed to behave in the playground when I was testing it locally. Requires a setting to be enabled https://lexical-playground-irlbe8n5m-fbopensource.vercel.app/?tableHorizontalScroll=true

@GermanJablo
Copy link
Contributor Author

That looks super cool! I'm going to pause this PR until we see if DOMSlot works!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
CLA Signed This label is managed by the Facebook bot. Authors need to sign the CLA before a PR can be reviewed.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Feature: Scroll for table
4 participants