Add support for events
Summary: LegacyViewManagerInterop layer can now handle events as well. We expose `eventInterceptor` from `NSComponentData`, that way we can hook into event dispatching that happens within `NSComponentData`. Coordinator has a map of `UIView -> reactTag` which it uses to figure out to which component view to forward the event. New `LegacyViewManagerInteropViewEventEmitter` exposes APIs to send `folly::dynamic` to javascript. Reviewed By: shergin Differential Revision: D17765834 fbshipit-source-id: 25e75e445b32c69ec9ab0189c4ab7fba5f8c7b5d
This commit is contained in:
Родитель
0a98be65e3
Коммит
5dbd62c7ff
|
@ -7,6 +7,7 @@
|
|||
|
||||
#import "RCTLegacyViewManagerInteropComponentView.h"
|
||||
|
||||
#import <React/UIView+React.h>
|
||||
#import <react/components/legacyviewmanagerinterop/LegacyViewManagerInteropComponentDescriptor.h>
|
||||
#import <react/components/legacyviewmanagerinterop/LegacyViewManagerInteropViewProps.h>
|
||||
#import <react/components/legacyviewmanagerinterop/RCTLegacyViewManagerInteropCoordinator.h>
|
||||
|
@ -31,19 +32,22 @@ using namespace facebook::react;
|
|||
|
||||
+ (BOOL)isSupported:(NSString *)componentName
|
||||
{
|
||||
static NSSet<NSString *> *supportedComponents = [NSSet setWithObjects:@"ActivityIndicatorView", nil];
|
||||
static NSSet<NSString *> *supportedComponents = [NSSet setWithObjects:@"Picker", @"DatePicker", nil];
|
||||
return [supportedComponents containsObject:componentName];
|
||||
}
|
||||
|
||||
#pragma mark - RCTComponentViewProtocol
|
||||
|
||||
- (void)prepareForRecycle
|
||||
- (RCTLegacyViewManagerInteropCoordinator *)coordinator
|
||||
{
|
||||
[super prepareForRecycle];
|
||||
[_view removeFromSuperview];
|
||||
_view = NULL;
|
||||
if (_state != nullptr) {
|
||||
const auto &state = _state->getData();
|
||||
return unwrapManagedObject(state.coordinator);
|
||||
} else {
|
||||
return NULL;
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - RCTComponentViewProtocol
|
||||
|
||||
+ (ComponentDescriptorProvider)componentDescriptorProvider
|
||||
{
|
||||
return concreteComponentDescriptorProvider<LegacyViewManagerInteropComponentDescriptor>();
|
||||
|
@ -58,18 +62,24 @@ using namespace facebook::react;
|
|||
{
|
||||
[super finalizeUpdates:updateMask];
|
||||
assert(_props && _state);
|
||||
const auto &state = _state->getData();
|
||||
RCTLegacyViewManagerInteropCoordinator *coordinator = unwrapManagedObject(state.coordinator);
|
||||
|
||||
if (!_view) {
|
||||
UIView *view = [coordinator view];
|
||||
__weak __typeof(self) weakSelf = self;
|
||||
UIView *view = [self.coordinator viewWithInterceptor:^(std::string eventName, folly::dynamic event) {
|
||||
if (weakSelf) {
|
||||
__typeof(self) strongSelf = weakSelf;
|
||||
auto eventEmitter =
|
||||
std::static_pointer_cast<LegacyViewManagerInteropViewEventEmitter const>(strongSelf->_eventEmitter);
|
||||
eventEmitter->dispatchEvent(eventName, event);
|
||||
}
|
||||
}];
|
||||
self.contentView = view;
|
||||
_view = view;
|
||||
}
|
||||
|
||||
if (updateMask & RNComponentViewUpdateMaskProps) {
|
||||
const auto &newProps = *std::static_pointer_cast<const LegacyViewManagerInteropViewProps>(_props);
|
||||
[coordinator setProps:newProps.otherProps forView:_view];
|
||||
[self.coordinator setProps:newProps.otherProps forView:_view];
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -29,6 +29,8 @@
|
|||
- (void)setProps:(NSDictionary<NSString *, id> *)props forView:(id<RCTComponent>)view;
|
||||
- (void)setProps:(NSDictionary<NSString *, id> *)props forShadowView:(RCTShadowView *)shadowView;
|
||||
|
||||
@property (nonatomic, copy, nullable) void (^eventInterceptor)(NSString *eventName, NSDictionary *event, id sender);
|
||||
|
||||
- (NSDictionary<NSString *, id> *)viewConfig;
|
||||
|
||||
@end
|
||||
|
|
|
@ -20,6 +20,7 @@
|
|||
|
||||
typedef void (^RCTPropBlock)(id<RCTComponent> view, id json);
|
||||
typedef NSMutableDictionary<NSString *, RCTPropBlock> RCTPropBlockDictionary;
|
||||
typedef void (^InterceptorBlock)(NSString *eventName, NSDictionary *event, id sender);
|
||||
|
||||
/**
|
||||
* Get the converter function for the specified type
|
||||
|
@ -101,7 +102,7 @@ RCT_NOT_IMPLEMENTED(- (instancetype)init)
|
|||
}
|
||||
}
|
||||
|
||||
static RCTPropBlock createEventSetter(NSString *propName, SEL setter, RCTBridge *bridge)
|
||||
static RCTPropBlock createEventSetter(NSString *propName, SEL setter,InterceptorBlock eventInterceptor, RCTBridge *bridge)
|
||||
{
|
||||
__weak RCTBridge *weakBridge = bridge;
|
||||
return ^(id target, id json) {
|
||||
|
@ -115,10 +116,14 @@ static RCTPropBlock createEventSetter(NSString *propName, SEL setter, RCTBridge
|
|||
return;
|
||||
}
|
||||
|
||||
RCTComponentEvent *componentEvent = [[RCTComponentEvent alloc] initWithName:propName
|
||||
viewTag:strongTarget.reactTag
|
||||
body:event];
|
||||
[weakBridge.eventDispatcher sendEvent:componentEvent];
|
||||
if (eventInterceptor) {
|
||||
eventInterceptor(propName, event, strongTarget);
|
||||
} else {
|
||||
RCTComponentEvent *componentEvent = [[RCTComponentEvent alloc] initWithName:propName
|
||||
viewTag:strongTarget.reactTag
|
||||
body:event];
|
||||
[weakBridge.eventDispatcher sendEvent:componentEvent];
|
||||
}
|
||||
};
|
||||
}
|
||||
((void (*)(id, SEL, id))objc_msgSend)(target, setter, eventHandler);
|
||||
|
@ -244,7 +249,7 @@ static RCTPropBlock createNSInvocationSetter(NSMethodSignature *typeSignature, S
|
|||
if (type == NSSelectorFromString(@"RCTBubblingEventBlock:") ||
|
||||
type == NSSelectorFromString(@"RCTDirectEventBlock:")) {
|
||||
// Special case for event handlers
|
||||
setterBlock = createEventSetter(name, setter, _bridge);
|
||||
setterBlock = createEventSetter(name, setter, self.eventInterceptor, _bridge);
|
||||
} else {
|
||||
// Ordinary property handlers
|
||||
NSMethodSignature *typeSignature = [[RCTConvert class] methodSignatureForSelector:type];
|
||||
|
|
|
@ -8,6 +8,7 @@
|
|||
#pragma once
|
||||
|
||||
#include <react/components/legacyviewmanagerinterop/LegacyViewManagerInteropState.h>
|
||||
#include <react/components/legacyviewmanagerinterop/LegacyViewManagerInteropViewEventEmitter.h>
|
||||
#include <react/components/legacyviewmanagerinterop/LegacyViewManagerInteropViewProps.h>
|
||||
#include <react/components/view/ConcreteViewShadowNode.h>
|
||||
|
||||
|
@ -19,7 +20,7 @@ extern const char LegacyViewManagerInteropComponentName[];
|
|||
using LegacyViewManagerInteropShadowNode = ConcreteViewShadowNode<
|
||||
LegacyViewManagerInteropComponentName,
|
||||
LegacyViewManagerInteropViewProps,
|
||||
ViewEventEmitter,
|
||||
LegacyViewManagerInteropViewEventEmitter,
|
||||
LegacyViewManagerInteropState>;
|
||||
|
||||
} // namespace react
|
||||
|
|
|
@ -0,0 +1,20 @@
|
|||
/**
|
||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
||||
*
|
||||
* This source code is licensed under the MIT license found in the
|
||||
* LICENSE file in the root directory of this source tree.
|
||||
*/
|
||||
|
||||
#include "LegacyViewManagerInteropViewEventEmitter.h"
|
||||
#include <iostream>
|
||||
|
||||
namespace facebook {
|
||||
namespace react {
|
||||
void LegacyViewManagerInteropViewEventEmitter::dispatchEvent(
|
||||
std::string const &type,
|
||||
folly::dynamic const &payload) const {
|
||||
EventEmitter::dispatchEvent(type, payload);
|
||||
}
|
||||
|
||||
} // namespace react
|
||||
} // namespace facebook
|
|
@ -0,0 +1,32 @@
|
|||
/**
|
||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
||||
*
|
||||
* This source code is licensed under the MIT license found in the
|
||||
* LICENSE file in the root directory of this source tree.
|
||||
*/
|
||||
#pragma once
|
||||
|
||||
#include <memory>
|
||||
|
||||
#include <folly/dynamic.h>
|
||||
#include <react/components/view/ViewEventEmitter.h>
|
||||
#include <react/core/EventEmitter.h>
|
||||
|
||||
namespace facebook {
|
||||
namespace react {
|
||||
|
||||
class LegacyViewManagerInteropViewEventEmitter;
|
||||
|
||||
using SharedLegacyViewManagerInteropViewEventEmitter =
|
||||
std::shared_ptr<const LegacyViewManagerInteropViewEventEmitter>;
|
||||
|
||||
class LegacyViewManagerInteropViewEventEmitter : public ViewEventEmitter {
|
||||
public:
|
||||
using ViewEventEmitter::ViewEventEmitter;
|
||||
|
||||
void dispatchEvent(std::string const &type, folly::dynamic const &payload)
|
||||
const;
|
||||
};
|
||||
|
||||
} // namespace react
|
||||
} // namespace facebook
|
|
@ -13,11 +13,13 @@ NS_ASSUME_NONNULL_BEGIN
|
|||
|
||||
@class RCTComponentData;
|
||||
|
||||
typedef void (^InterceptorBlock)(std::string eventName, folly::dynamic event);
|
||||
|
||||
@interface RCTLegacyViewManagerInteropCoordinator : NSObject
|
||||
|
||||
- (instancetype)initWithComponentData:(RCTComponentData *)componentData;
|
||||
|
||||
- (UIView *)view;
|
||||
- (UIView *)viewWithInterceptor:(InterceptorBlock)block;
|
||||
|
||||
- (void)setProps:(folly::dynamic const &)props forView:(UIView *)view;
|
||||
|
||||
|
|
|
@ -5,38 +5,52 @@
|
|||
* LICENSE file in the root directory of this source tree.
|
||||
*/
|
||||
|
||||
#import "RCTLegacyViewManagerInteropCoordinator.h"
|
||||
#include "RCTLegacyViewManagerInteropCoordinator.h"
|
||||
#include <React/RCTComponentData.h>
|
||||
#include <React/RCTFollyConvert.h>
|
||||
#include <folly/json.h>
|
||||
|
||||
static NSDictionary<NSString *, id> *something(folly::dynamic const &props)
|
||||
{
|
||||
auto json = folly::toJson(props);
|
||||
NSData *data = [NSData dataWithBytes:json.c_str() length:json.size()];
|
||||
NSDictionary<NSString *, id> *result = [NSJSONSerialization JSONObjectWithData:data options:0 error:NULL];
|
||||
return result;
|
||||
}
|
||||
using namespace facebook::react;
|
||||
|
||||
@implementation RCTLegacyViewManagerInteropCoordinator {
|
||||
RCTComponentData *_componentData;
|
||||
/*
|
||||
Each instnace of `RCTLegacyViewManagerInteropComponentView` registers a block to which events are dispatched.
|
||||
This is the container that maps unretained UIView pointer to a block to which the event is dispatched.
|
||||
*/
|
||||
NSMutableDictionary<NSValue *, InterceptorBlock> *_eventInterceptors;
|
||||
}
|
||||
|
||||
- (instancetype)initWithComponentData:(RCTComponentData *)componentData;
|
||||
{
|
||||
if (self = [super init]) {
|
||||
_componentData = componentData;
|
||||
|
||||
_eventInterceptors = [NSMutableDictionary new];
|
||||
|
||||
__weak __typeof(self) weakSelf = self;
|
||||
_componentData.eventInterceptor = ^(NSString *eventName, NSDictionary *event, id sender) {
|
||||
__typeof(self) strongSelf = weakSelf;
|
||||
InterceptorBlock block =
|
||||
[strongSelf->_eventInterceptors objectForKey:[NSValue valueWithNonretainedObject:sender]];
|
||||
if (block) {
|
||||
block(std::string([RCTNormalizeInputEventName(eventName) UTF8String]), convertIdToFollyDynamic(event));
|
||||
}
|
||||
};
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (UIView *)view
|
||||
- (UIView *)viewWithInterceptor:(InterceptorBlock)block
|
||||
{
|
||||
return [_componentData createViewWithTag:NULL];
|
||||
UIView *view = [_componentData createViewWithTag:NULL];
|
||||
[_eventInterceptors setObject:block forKey:[NSValue valueWithNonretainedObject:view]];
|
||||
return view;
|
||||
}
|
||||
|
||||
- (void)setProps:(folly::dynamic const &)props forView:(UIView *)view
|
||||
{
|
||||
NSDictionary<NSString *, id> *convertedProps = something(props);
|
||||
NSDictionary<NSString *, id> *convertedProps = convertFollyDynamicToId(props);
|
||||
[_componentData setProps:convertedProps forView:view];
|
||||
}
|
||||
|
||||
|
|
Загрузка…
Ссылка в новой задаче