-
Notifications
You must be signed in to change notification settings - Fork 24.6k
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
Add onPaste to TextInput #45425
base: main
Are you sure you want to change the base?
Add onPaste to TextInput #45425
Conversation
PR for docs update: facebook/react-native-website#4161 |
Base commit: fcd526d |
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.
Hi @s77rt, thanks for this contributions.
I see that there are some failures in CI for the JS side. This is happening because there is a new method in the spec and this is changing the public API of the component. We have a check that warn us about API changes.
In this case is a safe change, but can you update the test case? The failure is here.
Also, to set expectation, this kind of changes, that touches at the same time JS and native code, can take a little more to be merged and we might have to split the JS part from the Native part to land them properly.
@cipolleschi Updated! Thanks! |
@cipolleschi has imported this pull request. If you are a Meta employee, you can view this diff on Phabricator. |
.../ReactAndroid/src/main/java/com/facebook/react/views/textinput/ReactTextInputPasteEvent.java
Outdated
Show resolved
Hide resolved
Hi @s77rt, we were evaluating the PR internally and we figured a couple of things:
Thank you so much for your patience. |
@cipolleschi Thanks for your quick actions on this.
The new arch is already supported and tested with RNTester. Can you please elaborate if I'm missing something here?
|
Making this a draft until we land on an approved proposal react-native-community/discussions-and-proposals#804 |
This reverts commit 5baad1a.
will there be any new progress? 😳 |
@huhuanming Code-wise this is ready. You can port this as a patch for your needs. Feedback is much appreciated. For now this is still marked as draft because there is no approval on the proposal yet (and code could change) |
NSString *fileName = [NSString stringWithFormat:@"%@.%@", [[NSUUID UUID] UUIDString], fileExtension]; | ||
NSString *filePath = [NSTemporaryDirectory() stringByAppendingPathComponent:fileName]; |
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.
Replace with RCTTempFilePath
NSString *fileName = [NSString stringWithFormat:@"%@.%@", [[NSUUID UUID] UUIDString], fileExtension]; | ||
NSString *filePath = [NSTemporaryDirectory() stringByAppendingPathComponent:fileName]; |
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.
Same ^
Hey @s77rt, Thanks a lot for your work ! |
You are very likely using a precompiled artifact. You'd have to build react-native from source.
Ah I think I missed that, it's probably something to raise on react-native-community/discussions-and-proposals#804. BTW This is also missing the paste event from the keyboard suggestion list, check unresolved questions. |
Hey @s77rt, you're absolutely right! Since we're in an Expo app, we'll keep the feature Android-only for now and wait for your PR to be merged 🙂 In case you'd like to include iOS file pasting support in your PR, I’ve added some code to enable file pasting on iOS alongside image pasting. I created a small utility class to handle it: RCTPasterHandlerUtil.h #import <Foundation/Foundation.h>
#import <UIKit/UIKit.h>
NS_ASSUME_NONNULL_BEGIN
@interface RCTPasteHandlerUtil : NSObject
/**
Process the UIPasteboard and returns an array of info dictionaries for pasted items.
Each dictionary contains:
- fileName: (NSString) the temporary file name.
- fileSize: (NSNumber) length of the file data.
- type: (NSString) the MIME type.
- uri: (NSString) the temporary file’s URI.
This method will process both images and non‑image files.
@param pasteboard The UIPasteboard instance to process.
@return An array of info dictionaries.
*/
+ (NSArray<NSDictionary *> *)itemInfosFromPasteboard:(UIPasteboard *)pasteboard;
/**
Checks if the provided pasteboard has pasteable content.
@param pasteboard The UIPasteboard instance to check.
@return YES if there is content that can be pasted, NO otherwise.
*/
+ (BOOL)hasPasteableContent:(UIPasteboard *)pasteboard;
@end
NS_ASSUME_NONNULL_END RCTPasterHandlerUtil.mm #import "RCTPasteHandlerUtil.h"
#import <MobileCoreServices/MobileCoreServices.h>
#import <UniformTypeIdentifiers/UniformTypeIdentifiers.h>
@implementation RCTPasteHandlerUtil
+ (NSArray<NSDictionary *> *)itemInfosFromPasteboard:(UIPasteboard *)pasteboard {
NSMutableArray<NSDictionary *> *itemInfos = [NSMutableArray array];
NSArray<NSDictionary *> *items = pasteboard.items;
// Generic UTIs to ignore if a more specific UTI is available.
NSArray<NSString *> *genericUTIs = @[@"public.data", @"public.item"];
for (NSDictionary *item in items) {
// Pick a UTI: try to choose one that isn't generic.
NSString *selectedUTI = nil;
for (NSString *uti in item.allKeys) {
if (![genericUTIs containsObject:uti]) {
selectedUTI = uti;
break;
}
}
if (!selectedUTI) {
selectedUTI = item.allKeys.firstObject;
}
@try {
NSData *fileData = nil;
id value = item[selectedUTI];
// If the value is NSData, use it directly.
if ([value isKindOfClass:[NSData class]]) {
fileData = value;
}
// If it's a UIImage, convert it to data.
else if ([value isKindOfClass:[UIImage class]]) {
UIImage *image = (UIImage *)value;
// Choose PNG or JPEG based on UTI if possible.
if (@available(iOS 14.0, *)) {
UTType *utType = [UTType typeWithIdentifier:selectedUTI];
if (utType && [utType conformsToType:UTTypePNG]) {
fileData = UIImagePNGRepresentation(image);
} else {
fileData = UIImageJPEGRepresentation(image, 1.0);
}
} else {
// Fallback: check the UTI string.
if (UTTypeConformsTo((__bridge CFStringRef)selectedUTI, kUTTypePNG)) {
fileData = UIImagePNGRepresentation(image);
} else {
fileData = UIImageJPEGRepresentation(image, 1.0);
}
}
}
// If the value is an NSString, try to retrieve NSData from the pasteboard.
else if ([value isKindOfClass:[NSString class]]) {
fileData = [pasteboard dataForPasteboardType:selectedUTI];
}
// If we couldn't extract data, skip.
if (!fileData) {
continue;
}
// Determine MIME type and extension.
NSString *mimeType = @"application/octet-stream";
NSString *extension = @"bin";
if (@available(iOS 14.0, *)) {
UTType *utType = [UTType typeWithIdentifier:selectedUTI];
if (utType) {
mimeType = utType.preferredMIMEType ?: mimeType;
extension = utType.preferredFilenameExtension ?: extension;
}
} else {
CFStringRef utiRef = (__bridge CFStringRef)selectedUTI;
CFStringRef mimeTypeRef = UTTypeCopyPreferredTagWithClass(utiRef, kUTTagClassMIMEType);
mimeType = mimeTypeRef ? [NSString stringWithString:(__bridge_transfer NSString *)mimeTypeRef] : mimeType;
CFStringRef extRef = UTTypeCopyPreferredTagWithClass(utiRef, kUTTagClassFilenameExtension);
extension = extRef ? [NSString stringWithString:(__bridge_transfer NSString *)extRef] : extension;
}
if (extension.length == 0) {
extension = @"bin";
}
// Create a unique file name and write the data to a temporary file.
NSString *fileName = [NSString stringWithFormat:@"%@.%@", [[NSUUID UUID] UUIDString], extension];
NSString *filePath = [NSTemporaryDirectory() stringByAppendingPathComponent:fileName];
NSURL *fileURL = [NSURL fileURLWithPath:filePath];
BOOL success = [fileData writeToURL:fileURL atomically:YES];
if (success) {
NSDictionary *info = @{
@"fileName": fileName,
@"fileSize": @(fileData.length),
@"type": mimeType,
@"uri": fileURL.absoluteString
};
[itemInfos addObject:info];
}
} @catch (NSException *exception) {
NSDictionary *errorInfo = @{@"type": selectedUTI, @"error": exception.reason ?: @""};
[itemInfos addObject:errorInfo];
}
}
return itemInfos;
}
+ (BOOL)hasPasteableContent:(UIPasteboard *)pasteboard {
if (pasteboard.hasStrings || pasteboard.hasImages) {
return YES;
}
NSArray<NSDictionary *> *infos = [self itemInfosFromPasteboard:pasteboard];
return infos.count > 0;
}
@end And then in RCTUITextView.mm and RCTUITextField.mm we can do: - (BOOL)canPerformAction:(SEL)action withSender:(id)sender {
if (action == @selector(paste:)) {
return [RCTPasteHandlerUtil hasPasteableContent:[UIPasteboard generalPasteboard]];
}
return [super canPerformAction:action withSender:sender];
} - (void)paste:(id)sender {
if (clipboard.hasStrings) {
[_textInputDelegateAdapter didPaste:@"text/plain" withData:clipboard.string];
_textWasPasted = YES;
} else {
UIPasteboard *clipboard = [UIPasteboard generalPasteboard];
NSArray<NSDictionary *> *infos = [RCTPasteHandlerUtil itemInfosFromPasteboard:clipboard];
if (infos.count > 0) {
for (NSDictionary *info in infos) {
NSString *mime = info[@"type"];
NSString *uri = info[@"uri"];
[_textInputDelegateAdapter didPaste:mime withData:uri];
}
[self resignFirstResponder];
return;
}
}
[super paste:sender];
} |
Summary:
Detect
Paste
event. In our use case this is needed to unlock the possibility to support image pasting.Changelog:
[GENERAL] [ADDED] - Add onPaste prop to TextInput component
Test Plan:
onPaste
is logged beforeonChange
Screen.Recording.2024-07-13.at.7.08.40.PM.mov
Screen.Recording.2024-07-21.at.3.40.31.PM.mov
Screen.Recording.2024-07-25.at.6.16.22.PM.mov
Screen.Recording.2024-07-25.at.6.28.29.PM.mov