Skip to content

Commit

Permalink
[Sheets] [Pt. 12] Add event for selected snap point change (#5584)
Browse files Browse the repository at this point in the history
Co-authored-by: Eric Bailey <[email protected]>
Co-authored-by: Samuel Newman <[email protected]>
  • Loading branch information
3 people authored Oct 3, 2024
1 parent 79996ac commit 14e3567
Show file tree
Hide file tree
Showing 11 changed files with 86 additions and 21 deletions.
13 changes: 11 additions & 2 deletions modules/bottom-sheet/index.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,13 @@
import {BottomSheetState, BottomSheetViewProps} from './src/BottomSheet.types'
import {
BottomSheetSnapPoint,
BottomSheetState,
BottomSheetViewProps,
} from './src/BottomSheet.types'
import {BottomSheetView} from './src/BottomSheetView'

export {type BottomSheetState, BottomSheetView, type BottomSheetViewProps}
export {
BottomSheetSnapPoint,
type BottomSheetState,
BottomSheetView,
type BottomSheetViewProps,
}
11 changes: 6 additions & 5 deletions modules/bottom-sheet/ios/BottomSheetModule.swift
Original file line number Diff line number Diff line change
Expand Up @@ -10,8 +10,9 @@ public class BottomSheetModule: Module {

View(SheetView.self) {
Events([
"onAttemptDismiss",
"onSnapPointChange",
"onStateChange",
"onAttemptDismiss"
])

AsyncFunction("dismiss") { (view: SheetView) in
Expand All @@ -26,10 +27,6 @@ public class BottomSheetModule: Module {
view.cornerRadius = CGFloat(prop)
}

Prop("preventDismiss") { (view: SheetView, prop: Bool) in
view.preventDismiss = prop
}

Prop("minHeight") { (view: SheetView, prop: Double) in
view.minHeight = prop
}
Expand All @@ -38,6 +35,10 @@ public class BottomSheetModule: Module {
view.maxHeight = prop
}

Prop("preventDismiss") { (view: SheetView, prop: Bool) in
view.preventDismiss = prop
}

Prop("preventExpansion") { (view: SheetView, prop: Bool) in
view.preventExpansion = prop
}
Expand Down
23 changes: 21 additions & 2 deletions modules/bottom-sheet/ios/SheetView.swift
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,9 @@ class SheetView: ExpoView, UISheetPresentationControllerDelegate {
private var innerView: UIView?

// Events
private let onStateChange = EventDispatcher()
private let onAttemptDismiss = EventDispatcher()
private let onSnapPointChange = EventDispatcher()
private let onStateChange = EventDispatcher()

// Open event firing
private var isOpen: Bool = false {
Expand Down Expand Up @@ -51,6 +52,19 @@ class SheetView: ExpoView, UISheetPresentationControllerDelegate {
}
}
}
private var selectedDetentIdentifier: UISheetPresentationController.Detent.Identifier? {
didSet {
if selectedDetentIdentifier == .large {
onSnapPointChange([
"snapPoint": 2
])
} else {
onSnapPointChange([
"snapPoint": 1
])
}
}
}

// MARK: - Lifecycle

Expand Down Expand Up @@ -103,11 +117,12 @@ class SheetView: ExpoView, UISheetPresentationControllerDelegate {
}

let sheetVc = SheetViewController()
sheetVc.setDetents(contentHeight: self.clampHeight(contentHeight), preventExpansion: self.preventExpansion)
if let sheet = sheetVc.sheetPresentationController {
sheet.delegate = self
sheet.preferredCornerRadius = self.cornerRadius
self.selectedDetentIdentifier = sheet.selectedDetentIdentifier
}
sheetVc.setDetents(contentHeight: self.clampHeight(contentHeight), preventExpansion: self.preventExpansion)
sheetVc.view.addSubview(innerView)

self.sheetVc = sheetVc
Expand Down Expand Up @@ -158,4 +173,8 @@ class SheetView: ExpoView, UISheetPresentationControllerDelegate {
func presentationControllerDidDismiss(_ presentationController: UIPresentationController) {
self.destroy()
}

func sheetPresentationControllerDidChangeSelectedDetentIdentifier(_ sheetPresentationController: UISheetPresentationController) {
self.selectedDetentIdentifier = sheetPresentationController.selectedDetentIdentifier
}
}
10 changes: 10 additions & 0 deletions modules/bottom-sheet/src/BottomSheet.types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,12 @@ import {ColorValue, NativeSyntheticEvent} from 'react-native'

export type BottomSheetState = 'closed' | 'closing' | 'open' | 'opening'

export enum BottomSheetSnapPoint {
Hidden,
Partial,
Full,
}

export interface BottomSheetViewProps {
children: React.ReactNode
cornerRadius?: number
Expand All @@ -15,6 +21,10 @@ export interface BottomSheetViewProps {
minHeight?: number
maxHeight?: number

onAttemptDismiss?: (event: NativeSyntheticEvent<object>) => void
onSnapPointChange?: (
event: NativeSyntheticEvent<{snapPoint: BottomSheetSnapPoint}>,
) => void
onStateChange?: (
event: NativeSyntheticEvent<{state: BottomSheetState}>,
) => void
Expand Down
Empty file.
12 changes: 7 additions & 5 deletions src/components/Button.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -122,11 +122,13 @@ export const Button = React.forwardRef<View, ButtonProps>(
},
ref,
) => {
// This will pick the correct default pressable to use. If we are inside a dialog, we need to use the RNGH
// pressable so that it is usable inside the dialog.
const {insideDialog} = useDialogContext()
/*
* This will pick the correct default pressable to use. If we are inside a
* native dialog, we need to use the RNGH pressable.
*/
const {isNativeDialog} = useDialogContext()
if (!PressableComponent) {
if (insideDialog) {
if (isNativeDialog) {
PressableComponent = NormalizedRNGHPressable
} else {
PressableComponent = Pressable
Expand Down Expand Up @@ -468,7 +470,7 @@ export const Button = React.forwardRef<View, ButtonProps>(
role="button"
accessibilityHint={undefined} // optional
{...rest}
// @ts-expect-error ref type
// @ts-ignore - this will always be a pressable
ref={ref}
aria-label={label}
aria-pressed={state.pressed}
Expand Down
4 changes: 3 additions & 1 deletion src/components/Dialog/context.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,10 +6,12 @@ import {
DialogControlRefProps,
DialogOuterProps,
} from '#/components/Dialog/types'
import {BottomSheetSnapPoint} from '../../../modules/bottom-sheet/src/BottomSheet.types'

export const Context = React.createContext<DialogContextProps>({
close: () => {},
insideDialog: false,
isNativeDialog: false,
nativeSnapPoint: BottomSheetSnapPoint.Hidden,
})

export function useDialogContext() {
Expand Down
25 changes: 21 additions & 4 deletions src/components/Dialog/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,15 +8,18 @@ import {isIOS} from '#/platform/detection'
import {useDialogStateControlContext} from '#/state/dialogs'
import {List, ListMethods, ListProps} from '#/view/com/util/List'
import {atoms as a, flatten, useTheme} from '#/alf'
import {Context} from '#/components/Dialog/context'
import {Context, useDialogContext} from '#/components/Dialog/context'
import {
DialogControlProps,
DialogInnerProps,
DialogOuterProps,
} from '#/components/Dialog/types'
import {createInput} from '#/components/forms/TextField'
import {Portal} from '#/components/Portal'
import {BottomSheetView} from '../../../modules/bottom-sheet'
import {
BottomSheetSnapPoint,
BottomSheetView,
} from '../../../modules/bottom-sheet'

export {useDialogContext, useDialogControl} from '#/components/Dialog/context'
export * from '#/components/Dialog/types'
Expand Down Expand Up @@ -57,6 +60,10 @@ export function OuterWithoutPortal({
const closeCallbacks = React.useRef<(() => void)[]>([])
const {setDialogIsOpen} = useDialogStateControlContext()

const [snapPoint, setSnapPoint] = React.useState<BottomSheetSnapPoint>(
BottomSheetSnapPoint.Partial,
)

const callQueuedCallbacks = React.useCallback(() => {
for (const cb of closeCallbacks.current) {
try {
Expand Down Expand Up @@ -103,7 +110,10 @@ export function OuterWithoutPortal({
[open, close],
)

const context = React.useMemo(() => ({close, insideDialog: true}), [close])
const context = React.useMemo(
() => ({close, isNativeDialog: true, nativeSnapPoint: snapPoint}),
[close, snapPoint],
)

const Wrapper = isIOS ? View : GestureHandlerRootView

Expand All @@ -113,6 +123,9 @@ export function OuterWithoutPortal({
ref={ref}
topInset={30}
bottomInset={insets.bottom}
onSnapPointChange={e => {
setSnapPoint(e.nativeEvent.snapPoint)
}}
onStateChange={e => {
if (e.nativeEvent.state === 'closed') {
onCloseAnimationComplete()
Expand Down Expand Up @@ -147,8 +160,12 @@ export function Inner({children, style}: DialogInnerProps) {
export const ScrollableInner = React.forwardRef<ScrollView, DialogInnerProps>(
function ScrollableInner({children, style}, ref) {
const insets = useSafeAreaInsets()
const {nativeSnapPoint} = useDialogContext()
return (
<ScrollView style={[a.px_xl, style]} ref={ref}>
<ScrollView
style={[a.px_xl, style]}
ref={ref}
bounces={nativeSnapPoint === BottomSheetSnapPoint.Full}>
{children}
<View style={{height: insets.bottom + a.pt_5xl.paddingTop}} />
</ScrollView>
Expand Down
3 changes: 2 additions & 1 deletion src/components/Dialog/index.web.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -103,7 +103,8 @@ export function Outer({
const context = React.useMemo(
() => ({
close,
insideDialog: true,
isNativeDialog: false,
nativeSnapPoint: 0,
}),
[close],
)
Expand Down
4 changes: 3 additions & 1 deletion src/components/Dialog/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import type {

import {ViewStyleProp} from '#/alf'
import {BottomSheetViewProps} from '../../../modules/bottom-sheet'
import {BottomSheetSnapPoint} from '../../../modules/bottom-sheet/src/BottomSheet.types'

type A11yProps = Required<AccessibilityProps>

Expand Down Expand Up @@ -37,7 +38,8 @@ export type DialogControlProps = DialogControlRefProps & {

export type DialogContextProps = {
close: DialogControlProps['close']
insideDialog: boolean
isNativeDialog: boolean
nativeSnapPoint: BottomSheetSnapPoint
}

export type DialogControlOpenOptions = {
Expand Down
2 changes: 2 additions & 0 deletions src/components/NormalizedRNGHPressable.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -115,6 +115,8 @@ export class NormalizedRNGHPressable extends React.Component<PressableProps> {
render() {
return (
<BSPressable
accessible={true}
accessibilityRole="button"
{...this.props}
onPress={this.onPress}
onLongPress={this.onLongPress}
Expand Down

0 comments on commit 14e3567

Please sign in to comment.