2015-03-23 13:28:42 -07:00
|
|
|
/**
|
|
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This source code is licensed under the BSD-style license found in the
|
|
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
|
|
*/
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
#import <UIKit/UIKit.h>
|
|
|
|
|
2015-03-26 14:22:12 +00:00
|
|
|
#import "../Base/RCTBridgeModule.h"
|
|
|
|
#import "../Base/RCTConvert.h"
|
|
|
|
#import "../Base/RCTLog.h"
|
2015-02-19 20:10:52 -08:00
|
|
|
|
2015-03-01 15:33:55 -08:00
|
|
|
@class RCTBridge;
|
2015-02-19 20:10:52 -08:00
|
|
|
@class RCTEventDispatcher;
|
|
|
|
@class RCTShadowView;
|
|
|
|
@class RCTSparseArray;
|
|
|
|
@class RCTUIManager;
|
|
|
|
|
|
|
|
typedef void (^RCTViewManagerUIBlock)(RCTUIManager *uiManager, RCTSparseArray *viewRegistry);
|
|
|
|
|
2015-03-01 15:33:55 -08:00
|
|
|
@interface RCTViewManager : NSObject <RCTBridgeModule>
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
/**
|
2015-03-01 15:33:55 -08:00
|
|
|
* The bridge can be used to access both the RCTUIIManager and the RCTEventDispatcher,
|
|
|
|
* allowing the manager (or the views that it manages) to manipulate the view
|
|
|
|
* hierarchy and send events back to the JS context.
|
2015-02-19 20:10:52 -08:00
|
|
|
*/
|
2015-03-01 15:33:55 -08:00
|
|
|
@property (nonatomic, strong) RCTBridge *bridge;
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The module name exposed to React JS. If omitted, this will be inferred
|
|
|
|
* automatically by using the view module's class name. It is better to not
|
|
|
|
* override this, and just follow standard naming conventions for your view
|
|
|
|
* module subclasses.
|
|
|
|
*/
|
|
|
|
+ (NSString *)moduleName;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This method instantiates a native view to be managed by the module. Override
|
|
|
|
* this to return a custom view instance, which may be preconfigured with default
|
|
|
|
* properties, subviews, etc. This method will be called many times, and should
|
|
|
|
* return a fresh instance each time. The view module MUST NOT cache the returned
|
|
|
|
* view and return the same instance for subsequent calls.
|
|
|
|
*/
|
|
|
|
- (UIView *)view;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This method instantiates a shadow view to be managed by the module. If omitted,
|
|
|
|
* an ordinary RCTShadowView instance will be created, which is typically fine for
|
|
|
|
* most view types. As with the -view method, the -shadowView method should return
|
|
|
|
* a fresh instance each time it is called.
|
|
|
|
*/
|
|
|
|
- (RCTShadowView *)shadowView;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a dictionary of config data passed to JS that defines eligible events
|
|
|
|
* that can be placed on native views. This should return bubbling
|
|
|
|
* directly-dispatched event types and specify what names should be used to
|
|
|
|
* subscribe to either form (bubbling/capturing).
|
|
|
|
*
|
|
|
|
* Returned dictionary should be of the form: @{
|
|
|
|
* @"onTwirl": {
|
|
|
|
* @"phasedRegistrationNames": @{
|
|
|
|
* @"bubbled": @"onTwirl",
|
|
|
|
* @"captured": @"onTwirlCaptured"
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* Note that this method is not inherited when you subclass a view module, and
|
|
|
|
* you should not call [super customBubblingEventTypes] when overriding it.
|
|
|
|
*/
|
2015-03-24 17:37:03 -07:00
|
|
|
- (NSDictionary *)customBubblingEventTypes;
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a dictionary of config data passed to JS that defines eligible events
|
|
|
|
* that can be placed on native views. This should return non-bubbling
|
|
|
|
* directly-dispatched event types.
|
|
|
|
*
|
|
|
|
* Returned dictionary should be of the form: @{
|
|
|
|
* @"onTwirl": {
|
|
|
|
* @"registrationName": @"onTwirl"
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* Note that this method is not inherited when you subclass a view module, and
|
|
|
|
* you should not call [super customDirectEventTypes] when overriding it.
|
|
|
|
*/
|
2015-03-24 17:37:03 -07:00
|
|
|
- (NSDictionary *)customDirectEventTypes;
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
/**
|
2015-03-01 15:33:55 -08:00
|
|
|
* Called to notify manager that layout has finished, in case any calculated
|
|
|
|
* properties need to be copied over from shadow view to view.
|
2015-02-19 20:10:52 -08:00
|
|
|
*/
|
2015-03-01 15:33:55 -08:00
|
|
|
- (RCTViewManagerUIBlock)uiBlockToAmendWithShadowView:(RCTShadowView *)shadowView;
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
/**
|
2015-03-01 15:33:55 -08:00
|
|
|
* Called after view hierarchy manipulation has finished, and all shadow props
|
|
|
|
* have been set, but before layout has been performed. Useful for performing
|
|
|
|
* custo layout logic or tasks that involve walking the view hierarchy.
|
|
|
|
* To be deprecated, hopefully.
|
2015-02-19 20:10:52 -08:00
|
|
|
*/
|
|
|
|
- (RCTViewManagerUIBlock)uiBlockToAmendWithShadowViewRegistry:(RCTSparseArray *)shadowViewRegistry;
|
|
|
|
|
|
|
|
/**
|
2015-03-25 21:29:28 -07:00
|
|
|
* This handles the simple case, where JS and native property names match.
|
2015-02-19 20:10:52 -08:00
|
|
|
*/
|
2015-03-25 21:29:28 -07:00
|
|
|
#define RCT_EXPORT_VIEW_PROPERTY(name, type) RCT_REMAP_VIEW_PROPERTY(name, name, type)
|
|
|
|
|
|
|
|
#define RCT_EXPORT_SHADOW_PROPERTY(name, type) RCT_REMAP_SHADOW_PROPERTY(name, name, type)
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This macro maps a named property on the module to an arbitrary key path
|
2015-03-25 21:29:28 -07:00
|
|
|
* within the view or shadowView.
|
2015-02-19 20:10:52 -08:00
|
|
|
*/
|
2015-03-25 21:29:28 -07:00
|
|
|
#define RCT_REMAP_VIEW_PROPERTY(name, keyPath, type) \
|
2015-02-19 20:10:52 -08:00
|
|
|
- (void)set_##name:(id)json forView:(id)view withDefaultView:(id)defaultView { \
|
2015-03-25 21:29:28 -07:00
|
|
|
if ((json && !RCTSetProperty(view, @#keyPath, @selector(type:), json)) || \
|
|
|
|
(!json && !RCTCopyProperty(view, defaultView, @#keyPath))) { \
|
|
|
|
RCTLogError(@"%@ does not have setter for `%s` property", [view class], #name); \
|
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define RCT_REMAP_SHADOW_PROPERTY(name, keyPath, type) \
|
|
|
|
- (void)set_##name:(id)json forShadowView:(id)view withDefaultView:(id)defaultView { \
|
|
|
|
if ((json && !RCTSetProperty(view, @#keyPath, @selector(type:), json)) || \
|
|
|
|
(!json && !RCTCopyProperty(view, defaultView, @#keyPath))) { \
|
2015-03-01 15:33:55 -08:00
|
|
|
RCTLogError(@"%@ does not have setter for `%s` property", [view class], #name); \
|
2015-02-19 20:10:52 -08:00
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
2015-03-01 15:33:55 -08:00
|
|
|
/**
|
|
|
|
* These macros can be used when you need to provide custom logic for setting
|
|
|
|
* view properties. The macro should be followed by a method body, which can
|
|
|
|
* refer to "json", "view" and "defaultView" to implement the required logic.
|
|
|
|
*/
|
2015-03-25 21:29:28 -07:00
|
|
|
#define RCT_CUSTOM_VIEW_PROPERTY(name, type, viewClass) \
|
2015-03-15 16:08:42 -07:00
|
|
|
- (void)set_##name:(id)json forView:(viewClass *)view withDefaultView:(viewClass *)defaultView
|
2015-03-01 15:33:55 -08:00
|
|
|
|
2015-03-25 21:29:28 -07:00
|
|
|
#define RCT_CUSTOM_SHADOW_PROPERTY(name, type, viewClass) \
|
2015-03-15 16:08:42 -07:00
|
|
|
- (void)set_##name:(id)json forShadowView:(viewClass *)view withDefaultView:(viewClass *)defaultView
|
2015-03-01 15:33:55 -08:00
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
/**
|
|
|
|
* These are useful in cases where the module's superclass handles a
|
|
|
|
* property, but you wish to "unhandle" it, so it will be ignored.
|
|
|
|
*/
|
|
|
|
#define RCT_IGNORE_VIEW_PROPERTY(name) \
|
|
|
|
- (void)set_##name:(id)value forView:(id)view withDefaultView:(id)defaultView {}
|
|
|
|
|
|
|
|
#define RCT_IGNORE_SHADOW_PROPERTY(name) \
|
|
|
|
- (void)set_##name:(id)value forShadowView:(id)view withDefaultView:(id)defaultView {}
|
|
|
|
|
2015-03-30 04:58:02 -07:00
|
|
|
/**
|
|
|
|
* Used for when view property names change. Will log an error when used.
|
|
|
|
*/
|
|
|
|
#define RCT_DEPRECATED_VIEW_PROPERTY(oldName, newName) \
|
|
|
|
- (void)set_##oldName:(id)json forView:(id)view withDefaultView:(id)defaultView { \
|
|
|
|
RCTLogError(@"Property '%s' has been replaced by '%s'.", #oldName, #newName); \
|
|
|
|
[self set_##newName:json forView:view withDefaultView:defaultView]; \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define RCT_DEPRECATED_SHADOW_PROPERTY(oldName, newName) \
|
|
|
|
- (void)set_##oldName:(id)json forShadowView:(id)view withDefaultView:(id)defaultView { \
|
|
|
|
RCTLogError(@"Property '%s' has been replaced by '%s'.", #oldName, #newName); \
|
|
|
|
[self set_##newName:json forView:view withDefaultView:defaultView]; \
|
|
|
|
}
|
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
@end
|