/** * The examples provided by Facebook are for non-commercial testing and * evaluation purposes only. * * Facebook reserves all rights not expressly granted. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NON INFRINGEMENT. IN NO EVENT SHALL * FACEBOOK BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN * AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #import #import #import #import "RCTEventDispatcher.h" #import "RCTBridge+Private.h" @interface RCTTestEvent : NSObject @property (atomic, assign, readwrite) BOOL canCoalesce; @end @implementation RCTTestEvent { NSDictionary *_body; } @synthesize viewTag = _viewTag; @synthesize eventName = _eventName; @synthesize coalescingKey = _coalescingKey; - (instancetype)initWithViewTag:(NSNumber *)viewTag eventName:(NSString *)eventName body:(NSDictionary *)body coalescingKey:(uint16_t)coalescingKey { if (self = [super init]) { _viewTag = viewTag; _eventName = eventName; _body = body; _canCoalesce = YES; _coalescingKey = coalescingKey; } return self; } - (id)coalesceWithEvent:(id)newEvent { return newEvent; } + (NSString *)moduleDotMethod { return @"MyCustomEventemitter.emit"; } - (NSArray *)arguments { return @[_eventName, _body]; } @end @interface RCTEventDispatcherTests : XCTestCase @end @implementation RCTEventDispatcherTests { id _bridge; RCTEventDispatcher *_eventDispatcher; NSString *_eventName; NSDictionary *_body; RCTTestEvent *_testEvent; NSString *_JSMethod; } - (void)setUp { [super setUp]; _bridge = [OCMockObject mockForClass:[RCTBatchedBridge class]]; _eventDispatcher = [RCTEventDispatcher new]; [_eventDispatcher setValue:_bridge forKey:@"bridge"]; _eventName = RCTNormalizeInputEventName(@"sampleEvent"); _body = @{ @"foo": @"bar" }; _testEvent = [[RCTTestEvent alloc] initWithViewTag:nil eventName:_eventName body:_body coalescingKey:0]; _JSMethod = [[_testEvent class] moduleDotMethod]; } - (void)testLegacyEventsAreImmediatelyDispatched { [[_bridge expect] enqueueJSCall:@"RCTDeviceEventEmitter" method:@"emit" args:[_testEvent arguments] completion:NULL]; #pragma clang diagnostic push #pragma clang diagnostic ignored "-Wdeprecated-declarations" [_eventDispatcher sendDeviceEventWithName:_eventName body:_body]; #pragma clang diagnostic pop [_bridge verify]; } - (void)testNonCoalescingEventIsImmediatelyDispatched { _testEvent.canCoalesce = NO; [[_bridge expect] dispatchBlock:OCMOCK_ANY queue:RCTJSThread]; [_eventDispatcher sendEvent:_testEvent]; [_bridge verify]; } - (void)testCoalescingEventIsImmediatelyDispatched { _testEvent.canCoalesce = YES; [[_bridge expect] dispatchBlock:OCMOCK_ANY queue:RCTJSThread]; [_eventDispatcher sendEvent:_testEvent]; [_bridge verify]; } - (void)testMultipleEventsResultInOnlyOneDispatchAfterTheFirstOne { [[_bridge expect] dispatchBlock:OCMOCK_ANY queue:RCTJSThread]; [_eventDispatcher sendEvent:_testEvent]; [_eventDispatcher sendEvent:_testEvent]; [_eventDispatcher sendEvent:_testEvent]; [_eventDispatcher sendEvent:_testEvent]; [_eventDispatcher sendEvent:_testEvent]; [_bridge verify]; } - (void)testRunningTheDispatchedBlockResultInANewOneBeingEnqueued { __block dispatch_block_t eventsEmittingBlock; [[_bridge expect] dispatchBlock:[OCMArg checkWithBlock:^(dispatch_block_t block) { eventsEmittingBlock = block; return YES; }] queue:RCTJSThread]; [_eventDispatcher sendEvent:_testEvent]; [_bridge verify]; // eventsEmittingBlock would be called when js is no longer busy, which will result in emitting events [[_bridge expect] enqueueJSCall:[[_testEvent class] moduleDotMethod] args:[_testEvent arguments]]; eventsEmittingBlock(); [_bridge verify]; [[_bridge expect] dispatchBlock:OCMOCK_ANY queue:RCTJSThread]; [_eventDispatcher sendEvent:_testEvent]; [_bridge verify]; } - (void)testBasicCoalescingReturnsLastEvent { __block dispatch_block_t eventsEmittingBlock; [[_bridge expect] dispatchBlock:[OCMArg checkWithBlock:^(dispatch_block_t block) { eventsEmittingBlock = block; return YES; }] queue:RCTJSThread]; [[_bridge expect] enqueueJSCall:[[_testEvent class] moduleDotMethod] args:[_testEvent arguments]]; RCTTestEvent *ignoredEvent = [[RCTTestEvent alloc] initWithViewTag:nil eventName:_eventName body:@{ @"other": @"body" } coalescingKey:0]; [_eventDispatcher sendEvent:ignoredEvent]; [_eventDispatcher sendEvent:_testEvent]; eventsEmittingBlock(); [_bridge verify]; } - (void)testDifferentEventTypesDontCoalesce { NSString *firstEventName = RCTNormalizeInputEventName(@"firstEvent"); RCTTestEvent *firstEvent = [[RCTTestEvent alloc] initWithViewTag:nil eventName:firstEventName body:_body coalescingKey:0]; __block dispatch_block_t eventsEmittingBlock; [[_bridge expect] dispatchBlock:[OCMArg checkWithBlock:^(dispatch_block_t block) { eventsEmittingBlock = block; return YES; }] queue:RCTJSThread]; [[_bridge expect] enqueueJSCall:[[_testEvent class] moduleDotMethod] args:[firstEvent arguments]]; [[_bridge expect] enqueueJSCall:[[_testEvent class] moduleDotMethod] args:[_testEvent arguments]]; [_eventDispatcher sendEvent:firstEvent]; [_eventDispatcher sendEvent:_testEvent]; eventsEmittingBlock(); [_bridge verify]; } - (void)testSameEventTypesWithDifferentCoalesceKeysDontCoalesce { NSString *eventName = RCTNormalizeInputEventName(@"firstEvent"); RCTTestEvent *firstEvent = [[RCTTestEvent alloc] initWithViewTag:nil eventName:eventName body:_body coalescingKey:0]; RCTTestEvent *secondEvent = [[RCTTestEvent alloc] initWithViewTag:nil eventName:eventName body:_body coalescingKey:1]; __block dispatch_block_t eventsEmittingBlock; [[_bridge expect] dispatchBlock:[OCMArg checkWithBlock:^(dispatch_block_t block) { eventsEmittingBlock = block; return YES; }] queue:RCTJSThread]; [[_bridge expect] enqueueJSCall:[[_testEvent class] moduleDotMethod] args:[firstEvent arguments]]; [[_bridge expect] enqueueJSCall:[[_testEvent class] moduleDotMethod] args:[secondEvent arguments]]; [_eventDispatcher sendEvent:firstEvent]; [_eventDispatcher sendEvent:secondEvent]; [_eventDispatcher sendEvent:firstEvent]; [_eventDispatcher sendEvent:secondEvent]; [_eventDispatcher sendEvent:secondEvent]; [_eventDispatcher sendEvent:firstEvent]; [_eventDispatcher sendEvent:firstEvent]; eventsEmittingBlock(); [_bridge verify]; } @end