Skip to content

Commit e2b081e

Browse files
zhongwuzwreact-native-bot
authored andcommitted
Fixes TextInput crashes when any text is entered while running as iOS app on apple silicon mac (#49320)
Summary: Fixes #48544. We can make `RCTWeakEventEmitterWrapper` to subclass `NSDictionary` that Textinput supports encode it. System allowed classes are: ``` Allowed classes are: {( "'NSMorphology' (0x20088b6d8) [/System/Library/Frameworks/Foundation.framework]", "'NSString' (0x2003f4738) [/System/Library/Frameworks/Foundation.framework]", "'NSInflectionRule' (0x20088d348) [/System/Library/Frameworks/Foundation.framework]", "'UIColor' (0x2006c0520) [/System/iOSSupport/System/Library/PrivateFrameworks/UIKitCore.framework]", "'NSTextAttachment' (0x20042af98) [/System/Library/PrivateFrameworks/UIFoundation.framework]", "'NSShadow' (0x20042ae30) [/System/Library/PrivateFrameworks/UIFoundation.framework]", "'NSTextEncapsulation' (0x200897e50) [/System/Library/Frameworks/CoreText.framework]", "'NSTextAlternatives' (0x20042af70) [/System/Library/PrivateFrameworks/UIFoundation.framework]", "'NSFont' (0x20042a9f8) [/System/Library/PrivateFrameworks/UIFoundation.framework]", "'NSAttributedString' (0x2003f3838) [/System/Library/Frameworks/Foundation.framework]", "'NSData' (0x2003ed528) [/System/Library/Frameworks/CoreFoundation.framework]", "'NSURL' (0x2003ed938) [/System/Library/Frameworks/CoreFoundation.framework]", "'NSAdaptiveImageGlyph' (0x2008ae538) [/System/Library/PrivateFrameworks/UIFoundation.framework]", "'NSNumber' (0x2003f4238) [/System/Library/Frameworks/Foundation.framework]", "'NSParagraphStyle' (0x20042ad40) [/System/Library/PrivateFrameworks/UIFoundation.framework]", "'NSDictionary' (0x2003ed5a0) [/System/Library/Frameworks/CoreFoundation.framework]", "'UIFont' (0x20042c668) [/System/Library/PrivateFrameworks/UIFoundation.framework]", "'NSColor' (0x200412350) [/System/Library/Frameworks/AppKit.framework]", "'NSGlyphInfo' (0x20042aa98) [/System/Library/PrivateFrameworks/UIFoundation.framework]", "'NSArray' (0x2003ed460) [/System/Library/Frameworks/CoreFoundation.framework]", "'NSPresentationIntent' (0x20088da28) [/System/Library/Frameworks/Foundation.framework]" )} ``` ## Changelog: [IOS] [FIXED] - Fixes TextInput crashes when any text is entered while running as iOS app on apple silicon mac Pull Request resolved: #49320 Test Plan: Run RNTester on apple silicon mac, and entered some text in textinput, no crash occured. Also, verified that the caret was not jumping around, thus preserving the original fix. https://github.com/user-attachments/assets/6304f6e7-c663-4351-ace8-ab1842ee545f Reviewed By: javache Differential Revision: D69981500 Pulled By: cipolleschi fbshipit-source-id: 2af9b280e42f621446efda9b101af50525e8fef7
1 parent 68055f2 commit e2b081e

File tree

4 files changed

+30
-58
lines changed

4 files changed

+30
-58
lines changed

packages/react-native/React/Fabric/Mounting/ComponentViews/TextInput/RCTTextInputComponentView.mm

+1-7
Original file line numberDiff line numberDiff line change
@@ -99,11 +99,7 @@ - (void)updateEventEmitter:(const EventEmitter::Shared &)eventEmitter
9999
NSMutableDictionary<NSAttributedStringKey, id> *defaultAttributes =
100100
[_backedTextInputView.defaultTextAttributes mutableCopy];
101101

102-
#if !TARGET_OS_MACCATALYST
103-
RCTWeakEventEmitterWrapper *eventEmitterWrapper = [RCTWeakEventEmitterWrapper new];
104-
eventEmitterWrapper.eventEmitter = _eventEmitter;
105-
defaultAttributes[RCTAttributedStringEventEmitterKey] = eventEmitterWrapper;
106-
#endif
102+
defaultAttributes[RCTAttributedStringEventEmitterKey] = RCTWrapEventEmitter(_eventEmitter);
107103

108104
_backedTextInputView.defaultTextAttributes = defaultAttributes;
109105
}
@@ -263,10 +259,8 @@ - (void)updateProps:(const Props::Shared &)props oldProps:(const Props::Shared &
263259
if (newTextInputProps.textAttributes != oldTextInputProps.textAttributes) {
264260
NSMutableDictionary<NSAttributedStringKey, id> *defaultAttributes =
265261
RCTNSTextAttributesFromTextAttributes(newTextInputProps.getEffectiveTextAttributes(RCTFontSizeMultiplier()));
266-
#if !TARGET_OS_MACCATALYST
267262
defaultAttributes[RCTAttributedStringEventEmitterKey] =
268263
_backedTextInputView.defaultTextAttributes[RCTAttributedStringEventEmitterKey];
269-
#endif
270264
_backedTextInputView.defaultTextAttributes = defaultAttributes;
271265
}
272266

packages/react-native/ReactCommon/react/renderer/textlayoutmanager/platform/ios/react/renderer/textlayoutmanager/RCTAttributedTextUtils.h

+24-3
Original file line numberDiff line numberDiff line change
@@ -52,8 +52,29 @@ BOOL RCTIsAttributedStringEffectivelySame(
5252
NSDictionary<NSAttributedStringKey, id> *insensitiveAttributes,
5353
const facebook::react::TextAttributes &baseTextAttributes);
5454

55-
@interface RCTWeakEventEmitterWrapper : NSObject
56-
@property (nonatomic, assign) facebook::react::SharedEventEmitter eventEmitter;
57-
@end
55+
static inline NSData *RCTWrapEventEmitter(const facebook::react::SharedEventEmitter &eventEmitter)
56+
{
57+
auto eventEmitterPtr = new std::weak_ptr<const facebook::react::EventEmitter>(eventEmitter);
58+
return [[NSData alloc] initWithBytesNoCopy:eventEmitterPtr
59+
length:sizeof(eventEmitterPtr)
60+
deallocator:^(void *ptrToDelete, NSUInteger) {
61+
delete (std::weak_ptr<facebook::react::EventEmitter> *)ptrToDelete;
62+
}];
63+
}
64+
65+
static inline facebook::react::SharedEventEmitter RCTUnwrapEventEmitter(NSData *data)
66+
{
67+
if (data.length == 0) {
68+
return nullptr;
69+
}
70+
71+
auto weakPtr = dynamic_cast<std::weak_ptr<const facebook::react::EventEmitter> *>(
72+
(std::weak_ptr<const facebook::react::EventEmitter> *)data.bytes);
73+
if (weakPtr) {
74+
return weakPtr->lock();
75+
}
76+
77+
return nullptr;
78+
}
5879

5980
NS_ASSUME_NONNULL_END

packages/react-native/ReactCommon/react/renderer/textlayoutmanager/platform/ios/react/renderer/textlayoutmanager/RCTAttributedTextUtils.mm

+1-43
Original file line numberDiff line numberDiff line change
@@ -16,45 +16,6 @@
1616

1717
using namespace facebook::react;
1818

19-
@implementation RCTWeakEventEmitterWrapper {
20-
std::weak_ptr<const EventEmitter> _weakEventEmitter;
21-
}
22-
23-
- (void)setEventEmitter:(SharedEventEmitter)eventEmitter
24-
{
25-
_weakEventEmitter = eventEmitter;
26-
}
27-
28-
- (SharedEventEmitter)eventEmitter
29-
{
30-
return _weakEventEmitter.lock();
31-
}
32-
33-
- (void)dealloc
34-
{
35-
_weakEventEmitter.reset();
36-
}
37-
38-
- (BOOL)isEqual:(id)object
39-
{
40-
// We consider the underlying EventEmitter as the identity
41-
if (![object isKindOfClass:[self class]]) {
42-
return NO;
43-
}
44-
45-
auto thisEventEmitter = [self eventEmitter];
46-
auto otherEventEmitter = [((RCTWeakEventEmitterWrapper *)object) eventEmitter];
47-
return thisEventEmitter == otherEventEmitter;
48-
}
49-
50-
- (NSUInteger)hash
51-
{
52-
// We consider the underlying EventEmitter as the identity
53-
return (NSUInteger)_weakEventEmitter.lock().get();
54-
}
55-
56-
@end
57-
5819
inline static UIFontWeight RCTUIFontWeightFromInteger(NSInteger fontWeight)
5920
{
6021
assert(fontWeight > 50);
@@ -405,18 +366,15 @@ void RCTApplyBaselineOffset(NSMutableAttributedString *attributedText)
405366
{
406367
auto nsAttributedStringFragment = RCTNSAttributedStringFragmentFromFragment(fragment, placeholderImage);
407368

408-
#if !TARGET_OS_MACCATALYST
409369
if (fragment.parentShadowView.componentHandle) {
410-
RCTWeakEventEmitterWrapper *eventEmitterWrapper = [RCTWeakEventEmitterWrapper new];
411-
eventEmitterWrapper.eventEmitter = fragment.parentShadowView.eventEmitter;
370+
auto eventEmitterWrapper = RCTWrapEventEmitter(fragment.parentShadowView.eventEmitter);
412371

413372
NSDictionary<NSAttributedStringKey, id> *additionalTextAttributes =
414373
@{RCTAttributedStringEventEmitterKey : eventEmitterWrapper};
415374

416375
[nsAttributedStringFragment addAttributes:additionalTextAttributes
417376
range:NSMakeRange(0, nsAttributedStringFragment.length)];
418377
}
419-
#endif
420378

421379
return nsAttributedStringFragment;
422380
}

packages/react-native/ReactCommon/react/renderer/textlayoutmanager/platform/ios/react/renderer/textlayoutmanager/RCTTextLayoutManager.mm

+4-5
Original file line numberDiff line numberDiff line change
@@ -280,11 +280,10 @@ - (SharedEventEmitter)getEventEmitterWithAttributeString:(AttributedString)attri
280280
// after (fraction == 1.0) the last character, then the attribute is valid.
281281
if (textStorage.length > 0 && (fraction > 0 || characterIndex > 0) &&
282282
(fraction < 1 || characterIndex < textStorage.length - 1)) {
283-
RCTWeakEventEmitterWrapper *eventEmitterWrapper =
284-
(RCTWeakEventEmitterWrapper *)[textStorage attribute:RCTAttributedStringEventEmitterKey
285-
atIndex:characterIndex
286-
effectiveRange:NULL];
287-
return eventEmitterWrapper.eventEmitter;
283+
NSData *eventEmitterWrapper = (NSData *)[textStorage attribute:RCTAttributedStringEventEmitterKey
284+
atIndex:characterIndex
285+
effectiveRange:NULL];
286+
return RCTUnwrapEventEmitter(eventEmitterWrapper);
288287
}
289288

290289
return nil;

0 commit comments

Comments
 (0)