-
-
Notifications
You must be signed in to change notification settings - Fork 520
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
base: main
Are you sure you want to change the base?
feat: position storage #1529
Conversation
The latest updates on your projects. Learn more about Vercel for Git ↗︎
|
c7b80a9
to
2c3c9cf
Compare
2c3c9cf
to
0ae54a2
Compare
/** | ||
* 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>; | ||
}; |
There was a problem hiding this comment.
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
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); | ||
} |
There was a problem hiding this comment.
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?
There was a problem hiding this comment.
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
eead3ec
to
c6fd9ac
Compare
There was a problem hiding this 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
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); | ||
} |
There was a problem hiding this comment.
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; |
There was a problem hiding this comment.
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
?
There was a problem hiding this comment.
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 |
There was a problem hiding this comment.
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?
There was a problem hiding this comment.
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, { |
There was a problem hiding this comment.
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)
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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?
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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
packages/core/src/extensions/SuggestionMenu/SuggestionPlugin.ts
Outdated
Show resolved
Hide resolved
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 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
thanks for this comment :)
There was a problem hiding this 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.
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.