Skip to content

Commit 87af42a

Browse files
manicanticguycayogevbd
committed
Added modalAttemptedToDismiss event with tests and docs (#5832)
* Added modalAttemptedToDismiss event with tests and docs * Typo on pageSheet modal button label * Update ModalScreen.js * Fixed button testID * Revert label to fix test cases Co-authored-by: Guy Carmeli <[email protected]> Co-authored-by: Yogev Ben David <[email protected]>
1 parent cf591d9 commit 87af42a

16 files changed

+111
-11
lines changed

docs/docs/events.md

+16
Original file line numberDiff line numberDiff line change
@@ -145,6 +145,22 @@ const modalDismissedListener = Navigation.events().registerModalDismissedListene
145145
modalDismissedListener.remove();
146146
```
147147

148+
## registerModalAttemptedToDismissListener(iOS 13+ only)
149+
Invoked only on iOS pageSheet modal when swipeToDismiss flag is set to true and modal swiped down to dismiss.
150+
151+
```js
152+
// Subscribe
153+
const modalAttemptedToDismissListener = Navigation.events().registerModalAttemptedToDismissListener(({ componentId }) => {
154+
155+
});
156+
...
157+
// Unsubscribe
158+
modalDismissedListener.remove();
159+
```
160+
| Parameter | Description |
161+
|:--------------------:|:-----|
162+
|**componentId** | Id of the modal tried to dismiss|
163+
148164
## registerScreenPoppedListener
149165
Invoked when screen is popped.
150166

lib/ios/RNNCommandsHandler.m

+4
Original file line numberDiff line numberDiff line change
@@ -354,6 +354,10 @@ - (void)dismissedModal:(UIViewController *)viewController {
354354
[_eventEmitter sendModalsDismissedEvent:viewController.layoutInfo.componentId numberOfModalsDismissed:@(1)];
355355
}
356356

357+
- (void)attemptedToDismissModal:(UIViewController *)viewController {
358+
[_eventEmitter sendModalAttemptedToDismissEvent:viewController.layoutInfo.componentId];
359+
}
360+
357361
- (void)dismissedMultipleModals:(NSArray *)viewControllers {
358362
if (viewControllers && viewControllers.count) {
359363
[_eventEmitter sendModalsDismissedEvent:((UIViewController *)viewControllers.lastObject).layoutInfo.componentId numberOfModalsDismissed:@(viewControllers.count)];

lib/ios/RNNEventEmitter.h

+2
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,8 @@
2424

2525
- (void)sendModalsDismissedEvent:(NSString *)componentId numberOfModalsDismissed:(NSNumber *)modalsDismissed;
2626

27+
- (void)sendModalAttemptedToDismissEvent:(NSString *)componentId;
28+
2729
- (void)sendScreenPoppedEvent:(NSString *)componentId;
2830

2931

lib/ios/RNNEventEmitter.m

+9-1
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ @implementation RNNEventEmitter {
1515
static NSString* const ComponentDidDisappear = @"RNN.ComponentDidDisappear";
1616
static NSString* const NavigationButtonPressed = @"RNN.NavigationButtonPressed";
1717
static NSString* const ModalDismissed = @"RNN.ModalDismissed";
18+
static NSString* const ModalAttemptedToDismiss = @"RNN.ModalAttemptedToDismiss";
1819
static NSString* const SearchBarUpdated = @"RNN.SearchBarUpdated";
1920
static NSString* const SearchBarCancelPressed = @"RNN.SearchBarCancelPressed";
2021
static NSString* const PreviewCompleted = @"RNN.PreviewCompleted";
@@ -31,7 +32,8 @@ @implementation RNNEventEmitter {
3132
SearchBarUpdated,
3233
SearchBarCancelPressed,
3334
PreviewCompleted,
34-
ScreenPopped];
35+
ScreenPopped,
36+
ModalAttemptedToDismiss];
3537
}
3638

3739
# pragma mark public
@@ -113,6 +115,12 @@ - (void)sendModalsDismissedEvent:(NSString *)componentId numberOfModalsDismissed
113115
}];
114116
}
115117

118+
- (void)sendModalAttemptedToDismissEvent:(NSString *)componentId {
119+
[self send:ModalAttemptedToDismiss body:@{
120+
@"componentId": componentId,
121+
}];
122+
}
123+
116124
- (void)sendScreenPoppedEvent:(NSString *)componentId {
117125
[self send:ScreenPopped body:@{
118126
@"componentId": componentId

lib/ios/RNNModalManager.h

+1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ typedef void (^RNNTransitionRejectionBlock)(NSString *code, NSString *message, N
88
@protocol RNNModalManagerDelegate <NSObject>
99

1010
- (void)dismissedModal:(UIViewController *)viewController;
11+
- (void)attemptedToDismissModal:(UIViewController *)viewController;
1112
- (void)dismissedMultipleModals:(NSArray *)viewControllers;
1213

1314
@end

lib/ios/RNNModalManager.m

+4
Original file line numberDiff line numberDiff line change
@@ -127,6 +127,10 @@ - (void)presentationControllerDidDismiss:(UIPresentationController *)presentatio
127127
[_delegate dismissedModal:presentationController.presentedViewController.presentedComponentViewController];
128128
}
129129

130+
- (void)presentationControllerDidAttemptToDismiss:(UIPresentationController *)presentationController {
131+
[_delegate attemptedToDismissModal:presentationController.presentedViewController.presentedComponentViewController];
132+
}
133+
130134
-(UIViewController*)topPresentedVC {
131135
UIViewController *root = UIApplication.sharedApplication.delegate.window.rootViewController;
132136
while(root.presentedViewController) {

lib/src/adapters/NativeEventsReceiver.ts

+6-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,8 @@ import {
77
SearchBarCancelPressedEvent,
88
PreviewCompletedEvent,
99
ModalDismissedEvent,
10-
ScreenPoppedEvent
10+
ScreenPoppedEvent,
11+
ModalAttemptedToDismissEvent
1112
} from '../interfaces/ComponentEvents';
1213
import { CommandCompletedEvent, BottomTabSelectedEvent } from '../interfaces/Events';
1314

@@ -49,6 +50,10 @@ export class NativeEventsReceiver {
4950
return this.emitter.addListener('RNN.ModalDismissed', callback);
5051
}
5152

53+
public registerModalAttemptedToDismissListener(callback: (event: ModalAttemptedToDismissEvent) => void): EmitterSubscription {
54+
return this.emitter.addListener('RNN.ModalAttemptedToDismiss', callback);
55+
}
56+
5257
public registerSearchBarUpdatedListener(callback: (event: SearchBarUpdatedEvent) => void): EmitterSubscription {
5358
return this.emitter.addListener('RNN.SearchBarUpdated', callback);
5459
}

lib/src/events/ComponentEventsObserver.test.tsx

+16-3
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ describe('ComponentEventsObserver', () => {
1818
const searchBarCancelPressedFn = jest.fn();
1919
const previewCompletedFn = jest.fn();
2020
const modalDismissedFn = jest.fn();
21+
const modalAttemptedToDismissFn = jest.fn();
2122
const screenPoppedFn = jest.fn();
2223
let subscription: EventSubscription;
2324
let uut: ComponentEventsObserver;
@@ -57,6 +58,10 @@ describe('ComponentEventsObserver', () => {
5758
modalDismissedFn(event);
5859
}
5960

61+
modalAttemptedToDismiss(event: any) {
62+
modalAttemptedToDismissFn(event);
63+
}
64+
6065
searchBarUpdated(event: any) {
6166
searchBarUpdatedFn(event);
6267
}
@@ -108,6 +113,10 @@ describe('ComponentEventsObserver', () => {
108113
modalDismissedFn(event);
109114
}
110115

116+
modalAttemptedToDismiss(event: any) {
117+
modalAttemptedToDismissFn(event);
118+
}
119+
111120
searchBarUpdated(event: any) {
112121
searchBarUpdatedFn(event);
113122
}
@@ -153,14 +162,14 @@ describe('ComponentEventsObserver', () => {
153162
});
154163

155164
it(`bindComponent should use optional componentId if component has a componentId in props`, () => {
156-
const tree = renderer.create(<UnboundScreen componentId={'doNotUseThisId'} />);
165+
const tree = renderer.create(<UnboundScreen componentId={'doNotUseThisId'} />);
157166
uut.bindComponent(tree.getInstance() as any, 'myCompId')
158167

159168
expect(tree.toJSON()).toBeDefined();
160-
169+
161170
uut.notifyComponentDidAppear({ componentId: 'dontUseThisId', componentName: 'doesnt matter', componentType: 'Component' });
162171
expect(didAppearFn).not.toHaveBeenCalled();
163-
172+
164173

165174
uut.notifyComponentDidAppear({ componentId: 'myCompId', componentName: 'doesnt matter', componentType: 'Component' });
166175
expect(didAppearFn).toHaveBeenCalledTimes(1);
@@ -188,6 +197,10 @@ describe('ComponentEventsObserver', () => {
188197
expect(modalDismissedFn).toHaveBeenCalledTimes(1);
189198
expect(modalDismissedFn).toHaveBeenLastCalledWith({ componentId: 'myCompId', modalsDismissed: 1 })
190199

200+
uut.notifyModalAttemptedToDismiss({ componentId: 'myCompId' });
201+
expect(modalAttemptedToDismissFn).toHaveBeenCalledTimes(1);
202+
expect(modalAttemptedToDismissFn).toHaveBeenLastCalledWith({ componentId: 'myCompId' })
203+
191204
uut.notifySearchBarUpdated({ componentId: 'myCompId', text: 'theText', isFocused: true });
192205
expect(searchBarUpdatedFn).toHaveBeenCalledTimes(1);
193206
expect(searchBarUpdatedFn).toHaveBeenCalledWith({ componentId: 'myCompId', text: 'theText', isFocused: true });

lib/src/events/ComponentEventsObserver.ts

+8-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,8 @@ import {
1313
ComponentEvent,
1414
PreviewCompletedEvent,
1515
ModalDismissedEvent,
16-
ScreenPoppedEvent
16+
ScreenPoppedEvent,
17+
ModalAttemptedToDismissEvent
1718
} from '../interfaces/ComponentEvents';
1819
import { NativeEventsReceiver } from '../adapters/NativeEventsReceiver';
1920
import { Store } from '../components/Store';
@@ -32,6 +33,7 @@ export class ComponentEventsObserver {
3233
this.notifyComponentDidDisappear = this.notifyComponentDidDisappear.bind(this);
3334
this.notifyNavigationButtonPressed = this.notifyNavigationButtonPressed.bind(this);
3435
this.notifyModalDismissed = this.notifyModalDismissed.bind(this);
36+
this.notifyModalAttemptedToDismiss = this.notifyModalAttemptedToDismiss.bind(this);
3537
this.notifySearchBarUpdated = this.notifySearchBarUpdated.bind(this);
3638
this.notifySearchBarCancelPressed = this.notifySearchBarCancelPressed.bind(this);
3739
this.notifyPreviewCompleted = this.notifyPreviewCompleted.bind(this);
@@ -45,6 +47,7 @@ export class ComponentEventsObserver {
4547
this.nativeEventsReceiver.registerComponentDidDisappearListener(this.notifyComponentDidDisappear);
4648
this.nativeEventsReceiver.registerNavigationButtonPressedListener(this.notifyNavigationButtonPressed);
4749
this.nativeEventsReceiver.registerModalDismissedListener(this.notifyModalDismissed);
50+
this.nativeEventsReceiver.registerModalAttemptedToDismissListener(this.notifyModalAttemptedToDismiss);
4851
this.nativeEventsReceiver.registerSearchBarUpdatedListener(this.notifySearchBarUpdated);
4952
this.nativeEventsReceiver.registerSearchBarCancelPressedListener(this.notifySearchBarCancelPressed);
5053
this.nativeEventsReceiver.registerPreviewCompletedListener(this.notifyPreviewCompleted);
@@ -87,6 +90,10 @@ export class ComponentEventsObserver {
8790
this.triggerOnAllListenersByComponentId(event, 'modalDismissed');
8891
}
8992

93+
notifyModalAttemptedToDismiss(event: ModalAttemptedToDismissEvent) {
94+
this.triggerOnAllListenersByComponentId(event, 'modalAttemptedToDismiss');
95+
}
96+
9097
notifySearchBarUpdated(event: SearchBarUpdatedEvent) {
9198
this.triggerOnAllListenersByComponentId(event, 'searchBarUpdated');
9299
}

lib/src/events/EventsRegistry.test.tsx

+7
Original file line numberDiff line numberDiff line change
@@ -68,6 +68,13 @@ describe('EventsRegistry', () => {
6868
expect(mockNativeEventsReceiver.registerModalDismissedListener).toHaveBeenCalledWith(cb);
6969
});
7070

71+
it('delegates modalAttemptedToDimiss to nativeEventsReceiver', () => {
72+
const cb = jest.fn();
73+
uut.registerModalAttemptedToDismissListener(cb);
74+
expect(mockNativeEventsReceiver.registerModalAttemptedToDismissListener).toHaveBeenCalledTimes(1);
75+
expect(mockNativeEventsReceiver.registerModalAttemptedToDismissListener).toHaveBeenCalledWith(cb);
76+
});
77+
7178
it('delegates searchBarUpdated to nativeEventsReceiver', () => {
7279
const cb = jest.fn();
7380
uut.registerSearchBarUpdatedListener(cb);

lib/src/events/EventsRegistry.ts

+6-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,8 @@ import {
1212
SearchBarCancelPressedEvent,
1313
PreviewCompletedEvent,
1414
ModalDismissedEvent,
15-
ScreenPoppedEvent
15+
ScreenPoppedEvent,
16+
ModalAttemptedToDismissEvent
1617
} from '../interfaces/ComponentEvents';
1718
import { CommandCompletedEvent, BottomTabSelectedEvent } from '../interfaces/Events';
1819

@@ -47,6 +48,10 @@ export class EventsRegistry {
4748
return this.nativeEventsReceiver.registerModalDismissedListener(callback);
4849
}
4950

51+
public registerModalAttemptedToDismissListener(callback: (event: ModalAttemptedToDismissEvent) => void): EmitterSubscription {
52+
return this.nativeEventsReceiver.registerModalAttemptedToDismissListener(callback);
53+
}
54+
5055
public registerSearchBarUpdatedListener(callback: (event: SearchBarUpdatedEvent) => void): EmitterSubscription {
5156
return this.nativeEventsReceiver.registerSearchBarUpdatedListener(callback);
5257
}

lib/src/interfaces/ComponentEvents.ts

+4
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,10 @@ export interface ModalDismissedEvent extends ComponentEvent {
2424
modalsDismissed: number;
2525
}
2626

27+
export interface ModalAttemptedToDismissEvent extends ComponentEvent {
28+
componentId: string;
29+
}
30+
2731
export interface SearchBarUpdatedEvent extends ComponentEvent {
2832
text: string;
2933
isFocused: boolean;

playground/ios/NavigationTests/RNNCommandsHandlerTest.m

+1
Original file line numberDiff line numberDiff line change
@@ -103,6 +103,7 @@ -(NSArray*) getPublicMethodNamesForObject:(NSObject*)obj{
103103
[skipMethods addObject:@"readyToReceiveCommands"];
104104
[skipMethods addObject:@".cxx_destruct"];
105105
[skipMethods addObject:@"dismissedModal:"];
106+
[skipMethods addObject:@"attemptedToDismissModal:"];
106107
[skipMethods addObject:@"dismissedMultipleModals:"];
107108

108109
NSMutableArray* result = [NSMutableArray new];

playground/src/screens/ModalScreen.js

+12
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,18 @@ class ModalScreen extends React.Component {
3434
};
3535
}
3636

37+
state = {
38+
dimissCounter: 0,
39+
}
40+
41+
componentDidMount() {
42+
Navigation.events().bindComponent(this);
43+
}
44+
45+
modalAttemptedToDismiss() {
46+
return this.setState(state => ({ dimissCounter: state.dimissCounter + 1 }))
47+
}
48+
3749
render() {
3850
return (
3951
<Root componentId={this.props.componentId} footer={`Modal Stack Position: ${this.getModalPosition()}`}>

playground/src/screens/NavigationScreen.js

+13-3
Original file line numberDiff line numberDiff line change
@@ -2,18 +2,20 @@ const React = require('react');
22
const Root = require('../components/Root');
33
const Button = require('../components/Button')
44
const Navigation = require('./../services/Navigation');
5+
const { Platform } = require('react-native');
56
const {
67
NAVIGATION_TAB,
78
MODAL_BTN,
89
OVERLAY_BTN,
910
EXTERNAL_COMP_BTN,
1011
SHOW_STATIC_EVENTS_SCREEN,
1112
SHOW_ORIENTATION_SCREEN,
12-
SET_ROOT_BTN
13+
SET_ROOT_BTN,
14+
PAGE_SHEET_MODAL_BTN
1315
} = require('../testIDs');
1416
const Screens = require('./Screens');
1517

16-
class NavigationScreen extends React.Component {
18+
class NavigationScreen extends React.Component {
1719
static options() {
1820
return {
1921
topBar: {
@@ -34,6 +36,7 @@ class NavigationScreen extends React.Component {
3436
<Root componentId={this.props.componentId}>
3537
<Button label='Set Root' testID={SET_ROOT_BTN} onPress={this.setRoot} />
3638
<Button label='Modal' testID={MODAL_BTN} onPress={this.showModal} />
39+
{Platform.OS === 'ios' && <Button label='PageSheet modal' testID={PAGE_SHEET_MODAL_BTN} onPress={this.showPageSheetModal} />}
3740
<Button label='Overlay' testID={OVERLAY_BTN} onPress={this.showOverlay} />
3841
<Button label='External Component' testID={EXTERNAL_COMP_BTN} onPress={this.externalComponent} />
3942
<Button label='Static Events' testID={SHOW_STATIC_EVENTS_SCREEN} onPress={this.pushStaticEventsScreen} />
@@ -50,13 +53,20 @@ class NavigationScreen extends React.Component {
5053

5154
setRoot = () => Navigation.showModal(Screens.SetRoot);
5255
showModal = () => Navigation.showModal(Screens.Modal);
56+
57+
showPageSheetModal = () => Navigation.showModal(Screens.Modal, {
58+
modalPresentationStyle: 'pageSheet',
59+
modal: {
60+
swipeToDismiss: false,
61+
}
62+
});
5363
showOverlay = () => Navigation.showModal(Screens.Overlay);
5464
externalComponent = () => Navigation.showModal(Screens.ExternalComponent);
5565
pushStaticEventsScreen = () => Navigation.showModal(Screens.EventsScreen)
5666
orientation = () => Navigation.showModal(Screens.Orientation);
5767
pushContextScreen = () => Navigation.push(this, Screens.ContextScreen);
5868
sharedElement = () => Navigation.showModal(Screens.CocktailsListScreen)
59-
preview = ({reactTag}) => {
69+
preview = ({ reactTag }) => {
6070
Navigation.push(this.props.componentId, {
6171
component: {
6272
name: Screens.Pushed,

playground/src/testIDs.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ module.exports = {
1212
OVERLAY_BTN: 'OVERLAY_BTN',
1313
SIDE_MENU_BTN: 'SIDE_MENU_BTN',
1414
MODAL_BTN: 'SHOW_MODAL_BUTTON',
15+
PAGE_SHEET_MODAL_BTN: 'SHOW_PAGE_SHEET_MODAL_BUTTON',
1516
DISMISS_MODAL_BTN: 'DISMISS_MODAL_BUTTON',
1617
MODAL_SCREEN_HEADER: 'MODAL_SCREEN_HEADER',
1718
ALERT_BUTTON: 'ALERT_BUTTON',
@@ -129,7 +130,7 @@ module.exports = {
129130
SET_INTERCEPT_TOUCH: `SET_INTERCEPT_TOUCH`,
130131
PUSH_BOTTOM_TABS_BUTTON: `PUSH_BOTTOM_TABS_BUTTON`,
131132
SET_STACK_ROOT_BUTTON: `SET_STACK_ROOT_BUTTON`,
132-
SET_ROOT:'SET_ROOT',
133+
SET_ROOT: 'SET_ROOT',
133134
RESET_BUTTONS: 'RESET_BUTTONS',
134135
SHOW_LIFECYCLE_BTN: 'SHOW_LIFECYCLE_BTN',
135136
CHANGE_BUTTON_PROPS: 'CHANGE_BUTTON_PROPS',

0 commit comments

Comments
 (0)