-
Notifications
You must be signed in to change notification settings - Fork 71
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
Handle context menu format bold and italic #556
Handle context menu format bold and italic #556
Conversation
Hi @bernhardoj, thanks for submitting this PR. I'm okay with the changes. However, this PR only implements Format menu on web. I believe the same feature is missing on native iOS. Are there any plans to implement this missing feature also in native iOS? @Skalakid and @BartoszGrajdek Could you please review and test this PR? Thanks in advance. |
I don't see the Format option on iOS native. I think it's only available on the web. ios.2.mp4 |
@bernhardoj You're right, if there's no menu on native iOS then we don't need to add one right now. In that case I think we can proceed with web-only implementation. |
src/MarkdownTextInput.web.tsx
Outdated
default: | ||
markdown = ''; |
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 think this state is unreachable so we could just throw an error here.
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.
Is it better if I add underline as the default value?
switch (formatCommand) {
case 'formatBold':
formatType = 'bold';
break;
case 'formatItalic':
formatType = 'italic';
break;
default:
formatType = 'underline';
break;
}
Hello @bernhardoj, thank you for your PR! Yesterday, we merged support for adding custom parsers to the Live Markdown Input - PR. It's a pretty big change, and it extends the use cases for our library. Because of that, I think we should slightly update your solution in this PR. Now you've implemented adding static syntaxes in the formatSelection={(selectedText: string, style: 'bold' | 'italic' | 'underline') => style === 'bold' ? `*${selectedText}*` : style === 'italic' ? `_${selectedText}_` : selectedText} What do you think about it? |
That sounds good, I'll update it tomorrow! |
@Skalakid I need help with defining the type for react-native-live-markdown/src/MarkdownTextInput.tsx Lines 54 to 57 in 9ad0acd
react-native-live-markdown/src/MarkdownTextInput.web.tsx Lines 30 to 36 in 9ad0acd
|
@bernhardoj Just FYI, I was able to add "Format" button in context menu on native iOS with the following code snippet: // inside UITextView
UIMenuItem *formatItem = [[UIMenuItem alloc] initWithTitle:@"Format" action:@selector(command:textView:)];
UIMenuController.sharedMenuController.menuItems = @[formatItem]; - (void)command:(nonnull UICommand *)command textView:(nonnull UITextView *)textView
{
// Do something
} |
Should I add it here? react-native-live-markdown/apple/RCTUITextView+Markdown.mm Lines 29 to 40 in 5695408
Also, I get this error when trying to run the iOS example app.
|
@bernhardoj I think we can implement native iOS in a separate PR. Once we merge #520 it should be a bit simpler to implement that. I think we'll need to send a RN event from the native side to the JS side when bold or italic button in context menu is pressed so the function can modify the React state of the component in order to modify the current value by adding/removing some characters when toggling bold/italic. |
@bernhardoj Also, I'm curious what's the expected behavior when |
@bernhardoj I think, yes. You can add it to both files. Thanks to that there won't be any TS errors in the app |
c732cd4
to
292f1d8
Compare
@Skalakid updated.
I want to see how WhatsApp handles this case, but WA doesn't handle the format command on the web at all. |
@Skalakid hi, how's the review going? |
Hi, I reviewed the PR, and the code looks good to me, however, together with @tomekzaw, we noticed that this issue is more complex than we thought. The current implementation allows to modify the selected text, however, there are many other cases connected to removing style from the text, that should be handled. Here is a list of them:
I think we should reconsider the current solution and create simple to use API for |
On the other hand, I checked now, and WhatsApp doesn't support removing the styles in the format option. ScreenRecording_12-17-2024.14-13-21_1.MP4So I think we can merge it as it is and enhance the |
src/MarkdownTextInput.web.tsx
Outdated
default: | ||
formatType = 'underline'; | ||
break; | ||
} |
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.
Why underline
is set as default?
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 actually asked about this here. Since we only handle the bold, italic, and underline command, I default it to underline. Should I use the "throw" approach?
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.
Oh, I missed that question. Let's use the "throw" approach, we don't want the underline
to be a default option
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.
Since we don't do the conversion anymore, no throw is needed here.
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.
Sorry for the delay, we were OOO.
@bernhardoj The current code looks good to me, but can you add formatSelection
to the example app, please?
Also, here I left one more suggestion ;)
example/src/App.tsx
Outdated
if (formatCommand === 'formatBold') { | ||
return `*${selectedText}*`; | ||
} | ||
|
||
if (formatCommand === 'formatItalic') { | ||
return `_${selectedText}_`; | ||
} | ||
|
||
return selectedText; |
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 a nit but this is a perfect place to use switch/case
block.
if (formatCommand === 'formatBold') { | |
return `*${selectedText}*`; | |
} | |
if (formatCommand === 'formatItalic') { | |
return `_${selectedText}_`; | |
} | |
return selectedText; | |
switch (formatCommand) { | |
case 'formatBold': | |
return `*${selectedText}*`; | |
case 'formatItalic': | |
return `_${selectedText}_`; | |
default: | |
return selectedText; | |
} |
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.
Done
src/MarkdownTextInput.web.tsx
Outdated
const handleFormatSelection = useCallback( | ||
(target: MarkdownTextInputElement, parsedText: string, cursorPosition: number, formatCommand: string): ParseTextResult => { | ||
if (!contentSelection.current || contentSelection.current.end - contentSelection.current.start < 1) { | ||
throw new Error('[react-native-live-markdown] invalid selection'); |
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 rephrase the error message to be more specific.
throw new Error('[react-native-live-markdown] invalid selection'); | |
throw new Error('[react-native-live-markdown] Trying to apply format command on empty selection'); |
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.
Done
src/MarkdownTextInput.web.tsx
Outdated
} | ||
|
||
const selectedText = parsedText.slice(contentSelection.current.start, contentSelection.current.end); | ||
const formattedText = formatSelection?.(selectedText, formatCommand) ?? selectedText; |
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.
Why do we need the ?? selectedText
part here? Maybe we should just return early if formatSelection
is undefined?
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.
@bernhardoj Thanks for this PR as well as for applying all code review changes. The code looks good to me now.
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.
LGTM
Details
Handle the context menu Format command for Bold and Italic.
Related Issues
Expensify/App#53145
Manual Tests
ios.mweb.mp4
web.mp4
Linked PRs