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: position storage #1529

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

feat: position storage #1529

wants to merge 8 commits into from

Conversation

nperez0111
Copy link
Contributor

This is still in progress, the idea here is to make a class that can be used for keeping track of positions in the editor across collaborative or single user transactions.

This would be useful not only for the suggestions plugin, but for anything that would need to hold onto a position across transactions.

I still am working through setting up the tests to prove that it is able to keep track of those positions, but I had it preliminarily working for the suggestion plugin at one point, but then decided to solve this in a more generic way.

Copy link

vercel bot commented Mar 14, 2025

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

Name Status Preview Updated (UTC)
blocknote ✅ Ready (Inspect) Visit Preview Apr 1, 2025 8:05am
blocknote-website ✅ Ready (Inspect) Visit Preview Apr 1, 2025 8:05am

Comment on lines 371 to 381
/**
* Internal properties that are not part of the public API and may change in the future.
*
* @internal
*/
public readonly '~internal': {
/**
* Stores positions of elements in the editor.
*/
positionStorage: PositionStorage<BSchema, ISchema, SSchema>;
};
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Let's have a discussion on this.

In general, here is what I'm thinking:

  • The editor instance is the main thing passed around and is essentially a god object
  • configuration is primarily done through the editor instance, making it the most convenient place to store and retrieve references to the current config

I think what we need to be aiming for here instead is a de-coupled approach like prosekit, where the editor doesn't "have" these sorts of values, it can execute commands to move between states, but the values of these are stored elsewhere (like context).

Given that we don't have a good solution for state management at the moment, I propose here to just sweep things under the rug for a little while, and put them under an ~internal property. Using ~ makes it both cumbersome (read intentional) and shows the value last in TS autocomplete idea from standard-schema.

For configuration, I think that if the schema was defined separately, it would be much easier to configure things like tables and codeBlocks. For example:

import { defaultSchema, tableBlock } from '@blocknote/core/schema';|
import { BlockNoteEditor } from '@blocknote/core';

const editor = BlockNoteEditor.create({
  schema: defaultSchema.replace('table', tableBlock.configure({ option: true })
})

// editor.schema.table.config.option === true

Obviously, there are a lot of different ways to do it, but the point is that the editor doesn't really have to know about this.

What about all of the prosemirror plugins? It could maybe be something like:

import { BlockNoteEditor, BlockNoteExtension, getDefaultExtensions } from '@blocknote/core';

const editor = BlockNoteEditor.create({
  extensions: getDefaultExtensions.remove(['tables']).add({ myExt: BlockNoteExtension.create({ name: 'myExt', plugin: new Plugin() }) })
})

// editor.extensions.myExt.plugin

Again, different approaches here too

Comment on lines 51 to 66
constructor(
editor: BlockNoteEditor<BSchema, ISchema, SSchema>,
{ shouldMount = true }: { shouldMount?: boolean } = {}
) {
this.editor = editor;
this.onTransactionHandler = this.onTransactionHandler.bind(this);

if (!shouldMount) {
return;
}

if (!this.editor._tiptapEditor) {
throw new Error("Editor not mounted");
}
this.editor._tiptapEditor.on("transaction", this.onTransactionHandler);
}
Copy link
Contributor Author

Choose a reason for hiding this comment

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

An alternative to this, would be to have a prosemirror plugin that gets these transactions between state updates. Perhaps that is preferable?

Copy link
Collaborator

Choose a reason for hiding this comment

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

this is fine imo, unless there's an advantage to doing this in a plugin, but I don't think so.

Another alternative would be to emit our own event in BlockNoteEditor.dispatch - but there's a change that misses things when people use _tiptapEditor.dispatch

@nperez0111 nperez0111 force-pushed the feature/suggestions-plugin branch from eead3ec to c6fd9ac Compare March 31, 2025 14:19
@nperez0111 nperez0111 changed the title feat: position storage WIP feat: position storage Mar 31, 2025
@nperez0111 nperez0111 marked this pull request as ready for review March 31, 2025 15:24
Copy link
Collaborator

@YousefED YousefED left a comment

Choose a reason for hiding this comment

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

really like this approach to abstract it away and make it reusable! even better that it's unit tested etc.

I have one main question about the architecture and some minor comments, see inline

Comment on lines 51 to 66
constructor(
editor: BlockNoteEditor<BSchema, ISchema, SSchema>,
{ shouldMount = true }: { shouldMount?: boolean } = {}
) {
this.editor = editor;
this.onTransactionHandler = this.onTransactionHandler.bind(this);

if (!shouldMount) {
return;
}

if (!this.editor._tiptapEditor) {
throw new Error("Editor not mounted");
}
this.editor._tiptapEditor.on("transaction", this.onTransactionHandler);
}
Copy link
Collaborator

Choose a reason for hiding this comment

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

this is fine imo, unless there's an advantage to doing this in a plugin, but I don't think so.

Another alternative would be to emit our own event in BlockNoteEditor.dispatch - but there's a change that misses things when people use _tiptapEditor.dispatch

this.hadRemoteTransaction = true;
} else {
this.mapping.appendMapping(transaction.mapping);
this.mappingLength += transaction.mapping.maps.length;
Copy link
Collaborator

Choose a reason for hiding this comment

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

can't get this from this.mapping?

Copy link
Collaborator

Choose a reason for hiding this comment

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

ah; after reading further; I suppose the answer is that you can use a single PositionStorage object to track multiple positions. That makes sense and is pretty nifty. On the other hand, having a single PositionStorage object per tracked position would make it a little bit more efficient I suppose

ystate.binding.mapping
);

// This can happen if the element is deleted
Copy link
Collaborator

Choose a reason for hiding this comment

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

can this cause things to break in a weird way for the user?

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 probably should clarify, this is not just deleted text, but probably more accurately described as garbage collected text (i.e. text that once existed in the document & is not in memory for either the current ydoc or the client ydoc it is synchronized with). So, very unlikely to be hit, and impossible to get a relative position that points to it.

So, I think this might be a case where throwing an error is better, since it is unlikely to happen, and would simplify the callers (since they wouldn't have to deal with an undefined that is not going to happen in practice).

@@ -723,6 +735,13 @@ export class BlockNoteEditor<
// but we still need the schema
this.pmSchema = getSchema(tiptapOptions.extensions!);
}

this["~internal"] = {
positionStorage: new PositionStorage<BSchema, ISchema, SSchema>(this, {
Copy link
Collaborator

Choose a reason for hiding this comment

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

hmm, instantiating this on the editor means we store the mapping of all transactions since the document starts. I'm starting to think that we don't need a "global" instance of this and the alternative approach (in my previous comment) is a bit cleaner. In that case, we also don't need to worry about the "internal" thingg)

Copy link
Contributor Author

Choose a reason for hiding this comment

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

The tradeoff here is potential for memory leaks & ease-of-use.

What you are proposing would require having each position storage listen for transactions, but, when retrieving the value, there is nothing that would clean up that listener, so it'd just keep listening and leak memory.
Having 1 listener, and one map to keep track of the transactions avoids this by storing just a singleton, meaning that when the position is no longer referenced, it can be garbage collected.

What if we just add a cleanup function? Then storing the value would be a bit more of a hassle, since you would need to unsubscribe somewhere. And in something like the prosemirror plugin it is being used in now, you typically just stop referencing the value instead of needing to destructure it so there isn't a good place for it.
What if, on read of the value, we automatically unsubscribe? It makes it inconvenient to reference the value multiple times.

My first attempt at this API did it like you described, but I found the tradeoff with the listener to be problematic later.

Copy link
Collaborator

Choose a reason for hiding this comment

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

Can't you just unsubscribe / dispose the PositionStorage instance when the slash menu is closed, and we no longer need to track the position?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

You could, but it's a less convenient API. I think I have something else that should work.

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 was able to separate it from the editor instance by using a map of editor instance to the mapping. This will still use 1 mapping for the duration of an editor, but I think this is a fine trade-off for making it just as easy to store & use as a raw position value

return {
triggerCharacter:
suggestionPluginTransactionMeta.triggerCharacter,
deleteTriggerCharacter:
suggestionPluginTransactionMeta.deleteTriggerCharacter !==
false,
queryStartPos: newState.selection.from,
// When reading the queryStartPos, we offset the result by the length of the trigger character, to make it easy on the caller
Copy link
Collaborator

Choose a reason for hiding this comment

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

thanks for this comment :)

Copy link
Collaborator

@matthewlipski matthewlipski left a comment

Choose a reason for hiding this comment

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

Nice! Love how clear the comments are and the unit tests also help a lot with understanding how to use trackPosition. I don't really have any comments on the code, but IIRC this change is supposed to fix a bug with the slash menu with collaboration enabled, so would be great if you could update the description with that.

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