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

[SuperEditor] Fix text duplicated when a placeholder is inserted in a text with attributions (Resolves #2595) #2605

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

Conversation

angelosilvestre
Copy link
Collaborator

[SuperEditor] Fix text duplicated when a placeholder is inserted in a text with attributions (Resolves #2595)

When inserting a placeholder after an attributed region, the editor duplicates the text:

Nagranie.z.ekranu.2025-02-27.o.11.43.02.mov

The node's text itself isn't duplicated, the issue lives in the InlineSpan computation:

} else {
// This section is text. The end of this text is either the
// end of the AttributedText, or the index of the next placeholder.
contentEnd = span.end + 1;
for (final entry in placeholders.entries) {
if (entry.key > start) {
contentEnd = entry.key;
break;
}
}
inlineSpans.add(
TextSpan(
text: substring(start, contentEnd),
style: styleBuilder(span.attributions),
),
);
}

When there is a placeholder, we are assuming that the placeholder belongs in the current text span, which isn't always the case when there are attributions before the placeholder.

This PR fixes the issue by checking if the placeholder isn't part of a subsequent span.

I added tests for the computeInlineSpan extension method, but we could add tests using SuperEditor if needed.

I also found a separate issue when trying to insert a placeholder if the inline markdown plugin converter is used. I filed #2604 for that.

'Welcome to SuperEditor',
AttributedSpans(
attributions: [
const SpanMarker(attribution: boldAttribution, offset: 0, markerType: SpanMarkerType.start),
Copy link
Contributor

Choose a reason for hiding this comment

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

Did you intend for the "bold" attribution to apply to [xWelcom]e or did you mean for it to be x[Welcome]?

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

I intended it to be x[Welcome]. I forgot to adjust the offsets to account for the placeholder.

[_inlineWidgetBuilder],
);

// Ensure the text was not modified.
Copy link
Contributor

Choose a reason for hiding this comment

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

We should find some other way to describe this test and the expectation. It's not clear what "modification" might happen, or that we're guarding against.

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

The test was modified.

attributions: [
const SpanMarker(attribution: boldAttribution, offset: 0, markerType: SpanMarkerType.start),
const SpanMarker(attribution: boldAttribution, offset: 6, markerType: SpanMarkerType.end),
const SpanMarker(attribution: boldAttribution, offset: 11, markerType: SpanMarkerType.start),
Copy link
Contributor

Choose a reason for hiding this comment

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

I think these indices are also incorrect. With the placeholder at index 10, "SuperEditor" should start at index 12.

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Yeah, updated.

import 'package:super_editor/super_editor.dart';

void main() {
group('SuperEditor > computeInlineSpan >', () {
Copy link
Contributor

Choose a reason for hiding this comment

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

These tests should probably go with the other placeholder tests. Do we have an existing test suite for those in Super Editor? Or did I only add them to attributed_text?

Also, in these tests you verify the plain text, but given the nuanced nature of computing the inline span, we should probably verify the entire inline span structure. Is that a reasonable effort?

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Updated.

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Do we have an existing test suite for those in Super Editor? Or did I only add them to attributed_text?

I didn't find any test suite for placeholders in SuperEditor, only in attributed_text.

@@ -63,13 +63,26 @@ extension ComputeTextSpan on AttributedText {
}
} else {
// This section is text. The end of this text is either the
// end of the AttributedText, or the index of the next placeholder.
// end of the current span, or the index of the next placeholder.
Copy link
Contributor

Choose a reason for hiding this comment

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

I'm wondering if this whole if/else should be reworked....

It looks like when I wrote it, I treated the placeholder at the beginning of a span as a special case, but it looks like you've discovered that it's not a special case, right?

So the original if/else essentially thought about spans like this:

[|x|and some text]
// or just
[and some text]

But this PR seems to suggest that we might have:

[|x| and |x| and |x|]

Where "[ ]" represents a singular group of attributions, such as a single bold attribution.

If my understanding is correct, then I think we probably don't need an if/else at all. Instead, we should take the span and move through it from left to right, looking for any number of placeholders and slicing it up. No need to create a special condition for when a placeholder appears at the start.

Such a loop would take:

[|x| and |x| and |x|]

and chop it into:

|x| " and " |x| " and " |x|

I also noticed that we're building a text style with zero attributions for placeholders, but that's probably wrong? Attributions like bold are irrelevant for placeholders, but something like a font size attribution might be important. So we should probably use the same base text style for the placeholders as we do for the text in the overall span.

Copy link
Collaborator Author

Choose a reason for hiding this comment

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

Updated.

// We found a placeholder. Build a widget for it.

if (characterIndex > startOfInlineSpan) {
// There is text before the placeholder.
Copy link
Contributor

Choose a reason for hiding this comment

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

Please add to this comment a description of the action you're taking. You've found a placeholder....and therefore what?

@angelosilvestre angelosilvestre force-pushed the 2595_inserting-placeholder-duplicates-text branch from e5c7dd4 to 6cc4dc7 Compare April 1, 2025 23:22
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.

[BUG] - duplicated text with attribution after inserting the placeholder
2 participants