react-native-acoustic-connect-beta 18.0.7 → 18.0.9
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/Examples/SampleUI/.bundle/config +2 -0
- package/Examples/SampleUI/.eslintrc.js +4 -0
- package/Examples/SampleUI/.prettierrc.js +7 -0
- package/Examples/SampleUI/.watchmanconfig +1 -0
- package/Examples/SampleUI/App.js +1 -0
- package/Examples/SampleUI/Gemfile +10 -0
- package/Examples/SampleUI/Gemfile.lock +121 -0
- package/Examples/SampleUI/README.md +79 -0
- package/Examples/SampleUI/__tests__/App.test.tsx +17 -0
- package/Examples/SampleUI/android/app/build.gradle +124 -0
- package/Examples/SampleUI/android/app/debug.keystore +0 -0
- package/Examples/SampleUI/android/app/proguard-rules.pro +10 -0
- package/Examples/SampleUI/android/app/src/debug/AndroidManifest.xml +9 -0
- package/Examples/SampleUI/android/app/src/main/AndroidManifest.xml +26 -0
- package/Examples/SampleUI/android/app/src/main/java/com/sampleui/MainActivity.kt +23 -0
- package/Examples/SampleUI/android/app/src/main/java/com/sampleui/MainApplication.kt +53 -0
- package/Examples/SampleUI/android/app/src/main/res/drawable/rn_edit_text_material.xml +37 -0
- package/Examples/SampleUI/android/app/src/main/res/mipmap-hdpi/ic_launcher.png +0 -0
- package/Examples/SampleUI/android/app/src/main/res/mipmap-hdpi/ic_launcher_round.png +0 -0
- package/Examples/SampleUI/android/app/src/main/res/mipmap-mdpi/ic_launcher.png +0 -0
- package/Examples/SampleUI/android/app/src/main/res/mipmap-mdpi/ic_launcher_round.png +0 -0
- package/Examples/SampleUI/android/app/src/main/res/mipmap-xhdpi/ic_launcher.png +0 -0
- package/Examples/SampleUI/android/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png +0 -0
- package/Examples/SampleUI/android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png +0 -0
- package/Examples/SampleUI/android/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png +0 -0
- package/Examples/SampleUI/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png +0 -0
- package/Examples/SampleUI/android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png +0 -0
- package/Examples/SampleUI/android/app/src/main/res/values/strings.xml +3 -0
- package/Examples/SampleUI/android/app/src/main/res/values/styles.xml +9 -0
- package/Examples/SampleUI/android/build.gradle +21 -0
- package/Examples/SampleUI/android/gradle/wrapper/gradle-wrapper.jar +0 -0
- package/Examples/SampleUI/android/gradle/wrapper/gradle-wrapper.properties +7 -0
- package/Examples/SampleUI/android/gradle.properties +39 -0
- package/Examples/SampleUI/android/gradlew +252 -0
- package/Examples/SampleUI/android/gradlew.bat +94 -0
- package/Examples/SampleUI/android/settings.gradle +21 -0
- package/Examples/SampleUI/app.json +51 -0
- package/Examples/SampleUI/assets/fonts/Abel-Regular.ttf +0 -0
- package/Examples/SampleUI/assets/fonts/NotoSans-Regular.ttf +0 -0
- package/Examples/SampleUI/assets/images/android-icon.png +0 -0
- package/Examples/SampleUI/assets/images/artist-1.jpg +0 -0
- package/Examples/SampleUI/assets/images/artist-2.jpg +0 -0
- package/Examples/SampleUI/assets/images/avatar.png +0 -0
- package/Examples/SampleUI/assets/images/beach.jpg +0 -0
- package/Examples/SampleUI/assets/images/bridge.jpg +0 -0
- package/Examples/SampleUI/assets/images/chameleon.jpg +0 -0
- package/Examples/SampleUI/assets/images/city.jpg +0 -0
- package/Examples/SampleUI/assets/images/email-icon.png +0 -0
- package/Examples/SampleUI/assets/images/favorite.png +0 -0
- package/Examples/SampleUI/assets/images/forest.jpg +0 -0
- package/Examples/SampleUI/assets/images/paper-icon.png +0 -0
- package/Examples/SampleUI/assets/images/players-2.jpg +0 -0
- package/Examples/SampleUI/assets/images/players.jpg +0 -0
- package/Examples/SampleUI/assets/images/restaurant-1.jpg +0 -0
- package/Examples/SampleUI/assets/images/restaurant-2.jpg +0 -0
- package/Examples/SampleUI/assets/images/song-1.jpg +0 -0
- package/Examples/SampleUI/assets/images/song-2.jpg +0 -0
- package/Examples/SampleUI/assets/images/splash.png +0 -0
- package/Examples/SampleUI/assets/images/strawberries.jpg +0 -0
- package/Examples/SampleUI/assets/images/wrecked-ship.jpg +0 -0
- package/Examples/SampleUI/assets/styles/fonts.css +47 -0
- package/Examples/SampleUI/babel.config.js +22 -0
- package/Examples/SampleUI/index.js +9 -0
- package/Examples/SampleUI/ios/.xcode.env +11 -0
- package/Examples/SampleUI/ios/Podfile +58 -0
- package/Examples/SampleUI/ios/SampleUI/AppDelegate.h +7 -0
- package/Examples/SampleUI/ios/SampleUI/AppDelegate.mm +31 -0
- package/Examples/SampleUI/ios/SampleUI/Images.xcassets/AppIcon.appiconset/Contents.json +53 -0
- package/Examples/SampleUI/ios/SampleUI/Images.xcassets/Contents.json +6 -0
- package/Examples/SampleUI/ios/SampleUI/Info.plist +52 -0
- package/Examples/SampleUI/ios/SampleUI/LaunchScreen.storyboard +47 -0
- package/Examples/SampleUI/ios/SampleUI/PrivacyInfo.xcprivacy +48 -0
- package/Examples/SampleUI/ios/SampleUI/main.m +10 -0
- package/Examples/SampleUI/ios/SampleUI.xcodeproj/project.pbxproj +785 -0
- package/Examples/SampleUI/ios/SampleUI.xcodeproj/xcshareddata/xcschemes/SampleUI.xcscheme +88 -0
- package/Examples/SampleUI/ios/SampleUI.xcworkspace/contents.xcworkspacedata +10 -0
- package/Examples/SampleUI/ios/SampleUITests/Info.plist +24 -0
- package/Examples/SampleUI/ios/SampleUITests/SampleUITests.m +66 -0
- package/Examples/SampleUI/jest.config.js +3 -0
- package/Examples/SampleUI/metro.config.js +53 -0
- package/Examples/SampleUI/package-lock.json +17033 -0
- package/Examples/SampleUI/package.json +62 -0
- package/Examples/SampleUI/src/DrawerItems.tsx +322 -0
- package/Examples/SampleUI/src/ExampleList.tsx +175 -0
- package/Examples/SampleUI/src/Examples/ActivityIndicatorExample.tsx +73 -0
- package/Examples/SampleUI/src/Examples/AnimatedFABExample/AnimatedFABExample.tsx +220 -0
- package/Examples/SampleUI/src/Examples/AnimatedFABExample/CustomFAB.tsx +70 -0
- package/Examples/SampleUI/src/Examples/AnimatedFABExample/CustomFABControls.tsx +146 -0
- package/Examples/SampleUI/src/Examples/AnimatedFABExample/index.ts +1 -0
- package/Examples/SampleUI/src/Examples/AppbarExample.tsx +263 -0
- package/Examples/SampleUI/src/Examples/AvatarExample.tsx +76 -0
- package/Examples/SampleUI/src/Examples/BadgeExample.tsx +97 -0
- package/Examples/SampleUI/src/Examples/BannerExample.tsx +152 -0
- package/Examples/SampleUI/src/Examples/BottomNavigationBarExample.tsx +94 -0
- package/Examples/SampleUI/src/Examples/BottomNavigationExample.tsx +205 -0
- package/Examples/SampleUI/src/Examples/ButtonExample.tsx +405 -0
- package/Examples/SampleUI/src/Examples/CardExample.tsx +245 -0
- package/Examples/SampleUI/src/Examples/CheckboxExample.tsx +87 -0
- package/Examples/SampleUI/src/Examples/CheckboxItemExample.tsx +72 -0
- package/Examples/SampleUI/src/Examples/ChipExample.tsx +432 -0
- package/Examples/SampleUI/src/Examples/DataTableExample.tsx +128 -0
- package/Examples/SampleUI/src/Examples/DialogExample.tsx +131 -0
- package/Examples/SampleUI/src/Examples/Dialogs/DialogTextComponent.tsx +31 -0
- package/Examples/SampleUI/src/Examples/Dialogs/DialogWithCustomColors.tsx +57 -0
- package/Examples/SampleUI/src/Examples/Dialogs/DialogWithDismissableBackButton.tsx +38 -0
- package/Examples/SampleUI/src/Examples/Dialogs/DialogWithIcon.tsx +42 -0
- package/Examples/SampleUI/src/Examples/Dialogs/DialogWithLoadingIndicator.tsx +48 -0
- package/Examples/SampleUI/src/Examples/Dialogs/DialogWithLongText.tsx +81 -0
- package/Examples/SampleUI/src/Examples/Dialogs/DialogWithRadioBtns.tsx +111 -0
- package/Examples/SampleUI/src/Examples/Dialogs/UndismissableDialog.tsx +30 -0
- package/Examples/SampleUI/src/Examples/Dialogs/index.tsx +7 -0
- package/Examples/SampleUI/src/Examples/DividerExample.tsx +30 -0
- package/Examples/SampleUI/src/Examples/FABExample.tsx +226 -0
- package/Examples/SampleUI/src/Examples/IconButtonExample.tsx +230 -0
- package/Examples/SampleUI/src/Examples/IconExample.tsx +50 -0
- package/Examples/SampleUI/src/Examples/ListAccordionExample.tsx +65 -0
- package/Examples/SampleUI/src/Examples/ListAccordionGroupExample.tsx +80 -0
- package/Examples/SampleUI/src/Examples/ListItemExample.tsx +310 -0
- package/Examples/SampleUI/src/Examples/ListSectionExample.tsx +159 -0
- package/Examples/SampleUI/src/Examples/MaterialBottomTabNavigatorExample.tsx +87 -0
- package/Examples/SampleUI/src/Examples/MenuExample.tsx +211 -0
- package/Examples/SampleUI/src/Examples/ProgressBarExample.tsx +142 -0
- package/Examples/SampleUI/src/Examples/RadioButtonExample.tsx +104 -0
- package/Examples/SampleUI/src/Examples/RadioButtonGroupExample.tsx +67 -0
- package/Examples/SampleUI/src/Examples/RadioButtonItemExample.tsx +77 -0
- package/Examples/SampleUI/src/Examples/SearchbarExample.tsx +249 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtons/SegmentedButtonCustomColorCheck.tsx +93 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtons/SegmentedButtonDefault.tsx +49 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtons/SegmentedButtonDisabled.tsx +47 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtons/SegmentedButtonMultiselect.tsx +50 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtons/SegmentedButtonMultiselectIcons.tsx +49 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtons/SegmentedButtonMultiselectRealCase.tsx +114 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtons/SegmentedButtonOnlyIcons.tsx +40 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtons/SegmentedButtonOnlyIconsWithCheck.tsx +43 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtons/SegmentedButtonRealCase.tsx +97 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtons/SegmentedButtonWithDensity.tsx +47 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtons/SegmentedButtonWithSelectedCheck.tsx +52 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtons/index.ts +9 -0
- package/Examples/SampleUI/src/Examples/SegmentedButtonsExample.tsx +57 -0
- package/Examples/SampleUI/src/Examples/SnackbarExample.tsx +141 -0
- package/Examples/SampleUI/src/Examples/SurfaceExample.tsx +130 -0
- package/Examples/SampleUI/src/Examples/SwitchExample.tsx +105 -0
- package/Examples/SampleUI/src/Examples/TeamDetails.tsx +291 -0
- package/Examples/SampleUI/src/Examples/TeamsList.tsx +57 -0
- package/Examples/SampleUI/src/Examples/TextExample.tsx +132 -0
- package/Examples/SampleUI/src/Examples/TextInputExample.tsx +914 -0
- package/Examples/SampleUI/src/Examples/ThemeExample.tsx +41 -0
- package/Examples/SampleUI/src/Examples/ThemingWithReactNavigation.tsx +75 -0
- package/Examples/SampleUI/src/Examples/ToggleButtonExample.tsx +128 -0
- package/Examples/SampleUI/src/Examples/TooltipExample.tsx +191 -0
- package/Examples/SampleUI/src/Examples/TouchableRippleExample.tsx +37 -0
- package/Examples/SampleUI/src/RootNavigator.tsx +86 -0
- package/Examples/SampleUI/src/ScreenWrapper.tsx +67 -0
- package/Examples/SampleUI/src/components/ActivityIndicator.tsx +254 -0
- package/Examples/SampleUI/src/components/Appbar/Appbar.tsx +364 -0
- package/Examples/SampleUI/src/components/Appbar/AppbarAction.tsx +132 -0
- package/Examples/SampleUI/src/components/Appbar/AppbarBackAction.tsx +77 -0
- package/Examples/SampleUI/src/components/Appbar/AppbarBackIcon.tsx +52 -0
- package/Examples/SampleUI/src/components/Appbar/AppbarContent.tsx +249 -0
- package/Examples/SampleUI/src/components/Appbar/AppbarHeader.tsx +174 -0
- package/Examples/SampleUI/src/components/Appbar/index.ts +22 -0
- package/Examples/SampleUI/src/components/Appbar/utils.ts +188 -0
- package/Examples/SampleUI/src/components/Avatar/Avatar.tsx +8 -0
- package/Examples/SampleUI/src/components/Avatar/AvatarIcon.tsx +87 -0
- package/Examples/SampleUI/src/components/Avatar/AvatarImage.tsx +128 -0
- package/Examples/SampleUI/src/components/Avatar/AvatarText.tsx +127 -0
- package/Examples/SampleUI/src/components/Badge.tsx +138 -0
- package/Examples/SampleUI/src/components/Banner.tsx +302 -0
- package/Examples/SampleUI/src/components/BottomNavigation/BottomNavigation.tsx +643 -0
- package/Examples/SampleUI/src/components/BottomNavigation/BottomNavigationBar.tsx +1039 -0
- package/Examples/SampleUI/src/components/BottomNavigation/BottomNavigationRouteScreen.tsx +31 -0
- package/Examples/SampleUI/src/components/BottomNavigation/utils.ts +70 -0
- package/Examples/SampleUI/src/components/Button/Button.tsx +503 -0
- package/Examples/SampleUI/src/components/Button/utils.tsx +272 -0
- package/Examples/SampleUI/src/components/Card/Card.tsx +373 -0
- package/Examples/SampleUI/src/components/Card/CardActions.tsx +74 -0
- package/Examples/SampleUI/src/components/Card/CardContent.tsx +99 -0
- package/Examples/SampleUI/src/components/Card/CardCover.tsx +96 -0
- package/Examples/SampleUI/src/components/Card/CardTitle.tsx +236 -0
- package/Examples/SampleUI/src/components/Card/utils.tsx +110 -0
- package/Examples/SampleUI/src/components/Checkbox/Checkbox.tsx +78 -0
- package/Examples/SampleUI/src/components/Checkbox/CheckboxAndroid.tsx +180 -0
- package/Examples/SampleUI/src/components/Checkbox/CheckboxIOS.tsx +104 -0
- package/Examples/SampleUI/src/components/Checkbox/CheckboxItem.tsx +236 -0
- package/Examples/SampleUI/src/components/Checkbox/index.ts +19 -0
- package/Examples/SampleUI/src/components/Checkbox/utils.ts +183 -0
- package/Examples/SampleUI/src/components/Chip/Chip.tsx +513 -0
- package/Examples/SampleUI/src/components/Chip/helpers.tsx +322 -0
- package/Examples/SampleUI/src/components/CrossFadeIcon.tsx +140 -0
- package/Examples/SampleUI/src/components/DataTable/DataTable.tsx +138 -0
- package/Examples/SampleUI/src/components/DataTable/DataTableCell.tsx +132 -0
- package/Examples/SampleUI/src/components/DataTable/DataTableHeader.tsx +82 -0
- package/Examples/SampleUI/src/components/DataTable/DataTablePagination.tsx +407 -0
- package/Examples/SampleUI/src/components/DataTable/DataTableRow.tsx +107 -0
- package/Examples/SampleUI/src/components/DataTable/DataTableTitle.tsx +208 -0
- package/Examples/SampleUI/src/components/Dialog/Dialog.tsx +191 -0
- package/Examples/SampleUI/src/components/Dialog/DialogActions.tsx +94 -0
- package/Examples/SampleUI/src/components/Dialog/DialogContent.tsx +54 -0
- package/Examples/SampleUI/src/components/Dialog/DialogIcon.tsx +100 -0
- package/Examples/SampleUI/src/components/Dialog/DialogScrollArea.tsx +88 -0
- package/Examples/SampleUI/src/components/Dialog/DialogTitle.tsx +94 -0
- package/Examples/SampleUI/src/components/Divider.tsx +102 -0
- package/Examples/SampleUI/src/components/Drawer/Drawer.tsx +8 -0
- package/Examples/SampleUI/src/components/Drawer/DrawerCollapsedItem.tsx +282 -0
- package/Examples/SampleUI/src/components/Drawer/DrawerItem.tsx +209 -0
- package/Examples/SampleUI/src/components/Drawer/DrawerSection.tsx +138 -0
- package/Examples/SampleUI/src/components/FAB/AnimatedFAB.tsx +619 -0
- package/Examples/SampleUI/src/components/FAB/FAB.tsx +371 -0
- package/Examples/SampleUI/src/components/FAB/FABGroup.tsx +532 -0
- package/Examples/SampleUI/src/components/FAB/index.ts +13 -0
- package/Examples/SampleUI/src/components/FAB/utils.ts +474 -0
- package/Examples/SampleUI/src/components/HelperText/HelperText.tsx +168 -0
- package/Examples/SampleUI/src/components/HelperText/utils.ts +30 -0
- package/Examples/SampleUI/src/components/Icon.tsx +170 -0
- package/Examples/SampleUI/src/components/IconButton/IconButton.tsx +234 -0
- package/Examples/SampleUI/src/components/IconButton/utils.ts +190 -0
- package/Examples/SampleUI/src/components/List/List.tsx +20 -0
- package/Examples/SampleUI/src/components/List/ListAccordion.tsx +391 -0
- package/Examples/SampleUI/src/components/List/ListAccordionGroup.tsx +88 -0
- package/Examples/SampleUI/src/components/List/ListIcon.tsx +79 -0
- package/Examples/SampleUI/src/components/List/ListImage.tsx +89 -0
- package/Examples/SampleUI/src/components/List/ListItem.tsx +313 -0
- package/Examples/SampleUI/src/components/List/ListSection.tsx +87 -0
- package/Examples/SampleUI/src/components/List/ListSubheader.tsx +79 -0
- package/Examples/SampleUI/src/components/List/utils.ts +113 -0
- package/Examples/SampleUI/src/components/MaterialCommunityIcon.tsx +121 -0
- package/Examples/SampleUI/src/components/Menu/Menu.tsx +711 -0
- package/Examples/SampleUI/src/components/Menu/MenuItem.tsx +257 -0
- package/Examples/SampleUI/src/components/Menu/utils.ts +110 -0
- package/Examples/SampleUI/src/components/Modal.tsx +245 -0
- package/Examples/SampleUI/src/components/Portal/Portal.tsx +69 -0
- package/Examples/SampleUI/src/components/Portal/PortalConsumer.tsx +44 -0
- package/Examples/SampleUI/src/components/Portal/PortalHost.tsx +143 -0
- package/Examples/SampleUI/src/components/Portal/PortalManager.tsx +54 -0
- package/Examples/SampleUI/src/components/ProgressBar.tsx +292 -0
- package/Examples/SampleUI/src/components/RadioButton/RadioButton.tsx +86 -0
- package/Examples/SampleUI/src/components/RadioButton/RadioButtonAndroid.tsx +202 -0
- package/Examples/SampleUI/src/components/RadioButton/RadioButtonGroup.tsx +67 -0
- package/Examples/SampleUI/src/components/RadioButton/RadioButtonIOS.tsx +130 -0
- package/Examples/SampleUI/src/components/RadioButton/RadioButtonItem.tsx +268 -0
- package/Examples/SampleUI/src/components/RadioButton/index.ts +22 -0
- package/Examples/SampleUI/src/components/RadioButton/utils.ts +37 -0
- package/Examples/SampleUI/src/components/Searchbar.tsx +459 -0
- package/Examples/SampleUI/src/components/SegmentedButtons/SegmentedButtonItem.tsx +271 -0
- package/Examples/SampleUI/src/components/SegmentedButtons/SegmentedButtons.tsx +197 -0
- package/Examples/SampleUI/src/components/SegmentedButtons/utils.ts +179 -0
- package/Examples/SampleUI/src/components/Snackbar.tsx +430 -0
- package/Examples/SampleUI/src/components/Surface.tsx +376 -0
- package/Examples/SampleUI/src/components/Switch/Switch.tsx +108 -0
- package/Examples/SampleUI/src/components/Switch/utils.ts +113 -0
- package/Examples/SampleUI/src/components/TextInput/Addons/Outline.tsx +64 -0
- package/Examples/SampleUI/src/components/TextInput/Addons/Underline.tsx +78 -0
- package/Examples/SampleUI/src/components/TextInput/Adornment/TextInputAdornment.tsx +208 -0
- package/Examples/SampleUI/src/components/TextInput/Adornment/TextInputAffix.tsx +212 -0
- package/Examples/SampleUI/src/components/TextInput/Adornment/TextInputIcon.tsx +195 -0
- package/Examples/SampleUI/src/components/TextInput/Adornment/enums.tsx +12 -0
- package/Examples/SampleUI/src/components/TextInput/Adornment/types.tsx +11 -0
- package/Examples/SampleUI/src/components/TextInput/Adornment/utils.ts +48 -0
- package/Examples/SampleUI/src/components/TextInput/Label/InputLabel.tsx +217 -0
- package/Examples/SampleUI/src/components/TextInput/Label/LabelBackground.tsx +100 -0
- package/Examples/SampleUI/src/components/TextInput/TextInput.tsx +572 -0
- package/Examples/SampleUI/src/components/TextInput/TextInputFlat.tsx +475 -0
- package/Examples/SampleUI/src/components/TextInput/TextInputOutlined.tsx +450 -0
- package/Examples/SampleUI/src/components/TextInput/constants.tsx +48 -0
- package/Examples/SampleUI/src/components/TextInput/helpers.tsx +612 -0
- package/Examples/SampleUI/src/components/TextInput/types.tsx +155 -0
- package/Examples/SampleUI/src/components/ToggleButton/ToggleButton.tsx +187 -0
- package/Examples/SampleUI/src/components/ToggleButton/ToggleButtonGroup.tsx +72 -0
- package/Examples/SampleUI/src/components/ToggleButton/ToggleButtonRow.tsx +107 -0
- package/Examples/SampleUI/src/components/ToggleButton/index.ts +16 -0
- package/Examples/SampleUI/src/components/ToggleButton/utils.ts +26 -0
- package/Examples/SampleUI/src/components/Tooltip/Tooltip.tsx +249 -0
- package/Examples/SampleUI/src/components/Tooltip/utils.ts +129 -0
- package/Examples/SampleUI/src/components/TouchableRipple/Pressable.tsx +41 -0
- package/Examples/SampleUI/src/components/TouchableRipple/TouchableRipple.native.tsx +145 -0
- package/Examples/SampleUI/src/components/TouchableRipple/TouchableRipple.tsx +317 -0
- package/Examples/SampleUI/src/components/TouchableRipple/utils.ts +66 -0
- package/Examples/SampleUI/src/components/Typography/AnimatedText.tsx +107 -0
- package/Examples/SampleUI/src/components/Typography/Text.tsx +184 -0
- package/Examples/SampleUI/src/components/Typography/types.tsx +5 -0
- package/Examples/SampleUI/src/components/Typography/v2/Caption.tsx +50 -0
- package/Examples/SampleUI/src/components/Typography/v2/Headline.tsx +52 -0
- package/Examples/SampleUI/src/components/Typography/v2/Paragraph.tsx +49 -0
- package/Examples/SampleUI/src/components/Typography/v2/StyledText.tsx +56 -0
- package/Examples/SampleUI/src/components/Typography/v2/Subheading.tsx +50 -0
- package/Examples/SampleUI/src/components/Typography/v2/Text.tsx +62 -0
- package/Examples/SampleUI/src/components/Typography/v2/Title.tsx +49 -0
- package/Examples/SampleUI/src/components/Typography/v2/index.ts +5 -0
- package/Examples/SampleUI/src/core/PaperProvider.tsx +122 -0
- package/Examples/SampleUI/src/core/SafeAreaProviderCompat.tsx +65 -0
- package/Examples/SampleUI/src/core/settings.tsx +23 -0
- package/Examples/SampleUI/src/core/theming.tsx +170 -0
- package/Examples/SampleUI/src/index.native.tsx +270 -0
- package/Examples/SampleUI/src/index.tsx +226 -0
- package/Examples/SampleUI/src/react-navigation/index.tsx +19 -0
- package/Examples/SampleUI/src/react-navigation/navigators/createMaterialBottomTabNavigator.tsx +72 -0
- package/Examples/SampleUI/src/react-navigation/types.tsx +126 -0
- package/Examples/SampleUI/src/react-navigation/views/MaterialBottomTabView.tsx +140 -0
- package/Examples/SampleUI/src/styles/__tests__/fonts.test.js +290 -0
- package/Examples/SampleUI/src/styles/fonts.tsx +143 -0
- package/Examples/SampleUI/src/styles/overlay.tsx +70 -0
- package/Examples/SampleUI/src/styles/shadow.tsx +109 -0
- package/Examples/SampleUI/src/styles/themes/index.ts +4 -0
- package/Examples/SampleUI/src/styles/themes/v2/DarkTheme.tsx +30 -0
- package/Examples/SampleUI/src/styles/themes/v2/LightTheme.tsx +30 -0
- package/Examples/SampleUI/src/styles/themes/v2/colors.tsx +277 -0
- package/Examples/SampleUI/src/styles/themes/v3/DarkTheme.tsx +66 -0
- package/Examples/SampleUI/src/styles/themes/v3/LightTheme.tsx +69 -0
- package/Examples/SampleUI/src/styles/themes/v3/tokens.tsx +230 -0
- package/Examples/SampleUI/src/utils/BackHandler/BackHandler.native.tsx +3 -0
- package/Examples/SampleUI/src/utils/BackHandler/BackHandler.tsx +11 -0
- package/Examples/SampleUI/src/utils/addEventListener.tsx +56 -0
- package/Examples/SampleUI/src/utils/forwardRef.tsx +23 -0
- package/Examples/SampleUI/src/utils/getContrastingColor.tsx +15 -0
- package/Examples/SampleUI/src/utils/hasTouchHandler.tsx +23 -0
- package/Examples/SampleUI/src/utils/roundLayoutSize.ts +2 -0
- package/Examples/SampleUI/src/utils/splitStyles.ts +60 -0
- package/Examples/SampleUI/src/utils/useAnimatedValue.tsx +9 -0
- package/Examples/SampleUI/src/utils/useAnimatedValueArray.tsx +13 -0
- package/Examples/SampleUI/src/utils/useIsKeyboardShown.tsx +55 -0
- package/Examples/SampleUI/src/utils/useLayout.tsx +29 -0
- package/Examples/SampleUI/src/utils/useLazyRef.tsx +11 -0
- package/Examples/SampleUI/tsconfig.json +3 -0
- package/Examples/SampleUI/utils/index.ts +1426 -0
- package/lib/typescript/jslib/TLTRN.d.ts +5 -5
- package/lib/typescript/jslib/TLTRN.d.ts.map +1 -1
- package/package.json +7 -2
|
@@ -0,0 +1,711 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import {
|
|
3
|
+
Animated,
|
|
4
|
+
Dimensions,
|
|
5
|
+
Easing,
|
|
6
|
+
EmitterSubscription,
|
|
7
|
+
findNodeHandle,
|
|
8
|
+
I18nManager,
|
|
9
|
+
Keyboard,
|
|
10
|
+
KeyboardEvent as RNKeyboardEvent,
|
|
11
|
+
LayoutRectangle,
|
|
12
|
+
NativeEventSubscription,
|
|
13
|
+
Platform,
|
|
14
|
+
ScrollView,
|
|
15
|
+
ScrollViewProps,
|
|
16
|
+
StyleProp,
|
|
17
|
+
StyleSheet,
|
|
18
|
+
View,
|
|
19
|
+
ViewStyle,
|
|
20
|
+
Pressable,
|
|
21
|
+
} from 'react-native';
|
|
22
|
+
|
|
23
|
+
import { useSafeAreaInsets } from 'react-native-safe-area-context';
|
|
24
|
+
|
|
25
|
+
import MenuItem from './MenuItem';
|
|
26
|
+
import { useInternalTheme } from '../../core/theming';
|
|
27
|
+
import type { MD3Elevation, ThemeProp } from '../../types';
|
|
28
|
+
import { ElevationLevels } from '../../types';
|
|
29
|
+
import { addEventListener } from '../../utils/addEventListener';
|
|
30
|
+
import { BackHandler } from '../../utils/BackHandler/BackHandler';
|
|
31
|
+
import Portal from '../Portal/Portal';
|
|
32
|
+
import Surface from '../Surface';
|
|
33
|
+
|
|
34
|
+
export type Props = {
|
|
35
|
+
/**
|
|
36
|
+
* Whether the Menu is currently visible.
|
|
37
|
+
*/
|
|
38
|
+
visible: boolean;
|
|
39
|
+
/**
|
|
40
|
+
* The anchor to open the menu from. In most cases, it will be a button that opens the menu.
|
|
41
|
+
*/
|
|
42
|
+
anchor: React.ReactNode | { x: number; y: number };
|
|
43
|
+
/**
|
|
44
|
+
* Whether the menu should open at the top of the anchor or at its bottom.
|
|
45
|
+
* Applied only when anchor is a node, not an x/y position.
|
|
46
|
+
*/
|
|
47
|
+
anchorPosition?: 'top' | 'bottom';
|
|
48
|
+
/**
|
|
49
|
+
* Extra margin to add at the top of the menu to account for translucent status bar on Android.
|
|
50
|
+
* If you are using Expo, we assume translucent status bar and set a height for status bar automatically.
|
|
51
|
+
* Pass `0` or a custom value to and customize it.
|
|
52
|
+
* This is automatically handled on iOS.
|
|
53
|
+
*/
|
|
54
|
+
statusBarHeight?: number;
|
|
55
|
+
/**
|
|
56
|
+
* Callback called when Menu is dismissed. The `visible` prop needs to be updated when this is called.
|
|
57
|
+
*/
|
|
58
|
+
onDismiss?: () => void;
|
|
59
|
+
/**
|
|
60
|
+
* Accessibility label for the overlay. This is read by the screen reader when the user taps outside the menu.
|
|
61
|
+
*/
|
|
62
|
+
overlayAccessibilityLabel?: string;
|
|
63
|
+
/**
|
|
64
|
+
* Content of the `Menu`.
|
|
65
|
+
*/
|
|
66
|
+
children: React.ReactNode;
|
|
67
|
+
/**
|
|
68
|
+
* Style of menu's inner content.
|
|
69
|
+
*/
|
|
70
|
+
contentStyle?: Animated.WithAnimatedValue<StyleProp<ViewStyle>>;
|
|
71
|
+
style?: StyleProp<ViewStyle>;
|
|
72
|
+
/**
|
|
73
|
+
* Elevation level of the menu's content. Shadow styles are calculated based on this value. Default `backgroundColor` is taken from the corresponding `theme.colors.elevation` property. By default equals `2`.
|
|
74
|
+
* @supported Available in v5.x with theme version 3
|
|
75
|
+
*/
|
|
76
|
+
elevation?: MD3Elevation;
|
|
77
|
+
/**
|
|
78
|
+
* Mode of the menu's content.
|
|
79
|
+
* - `elevated` - Surface with a shadow and background color corresponding to set `elevation` value.
|
|
80
|
+
* - `flat` - Surface without a shadow, with the background color corresponding to set `elevation` value.
|
|
81
|
+
*
|
|
82
|
+
* @supported Available in v5.x with theme version 3
|
|
83
|
+
*/
|
|
84
|
+
mode?: 'flat' | 'elevated';
|
|
85
|
+
/**
|
|
86
|
+
* @optional
|
|
87
|
+
*/
|
|
88
|
+
theme?: ThemeProp;
|
|
89
|
+
/**
|
|
90
|
+
* Inner ScrollView prop
|
|
91
|
+
*/
|
|
92
|
+
keyboardShouldPersistTaps?: ScrollViewProps['keyboardShouldPersistTaps'];
|
|
93
|
+
/**
|
|
94
|
+
* testID to be used on tests.
|
|
95
|
+
*/
|
|
96
|
+
testID?: string;
|
|
97
|
+
};
|
|
98
|
+
|
|
99
|
+
// Minimum padding between the edge of the screen and the menu
|
|
100
|
+
const SCREEN_INDENT = 8;
|
|
101
|
+
// From https://material.io/design/motion/speed.html#duration
|
|
102
|
+
const ANIMATION_DURATION = 250;
|
|
103
|
+
// From the 'Standard easing' section of https://material.io/design/motion/speed.html#easing
|
|
104
|
+
const EASING = Easing.bezier(0.4, 0, 0.2, 1);
|
|
105
|
+
|
|
106
|
+
const WINDOW_LAYOUT = Dimensions.get('window');
|
|
107
|
+
|
|
108
|
+
const DEFAULT_ELEVATION: MD3Elevation = 2;
|
|
109
|
+
export const ELEVATION_LEVELS_MAP = Object.values(
|
|
110
|
+
ElevationLevels
|
|
111
|
+
) as ElevationLevels[];
|
|
112
|
+
|
|
113
|
+
const DEFAULT_MODE = 'elevated';
|
|
114
|
+
|
|
115
|
+
const focusFirstDOMNode = (el: View | null | undefined) => {
|
|
116
|
+
if (el && Platform.OS === 'web') {
|
|
117
|
+
// When in the browser, we want to focus the first focusable item on toggle
|
|
118
|
+
// For example, when menu is shown, focus the first item in the menu
|
|
119
|
+
// And when menu is dismissed, send focus back to the button to resume tabbing
|
|
120
|
+
const node: any = findNodeHandle(el);
|
|
121
|
+
const focusableNode = node.querySelector(
|
|
122
|
+
// This is a rough list of selectors that can be focused
|
|
123
|
+
'button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])'
|
|
124
|
+
);
|
|
125
|
+
|
|
126
|
+
focusableNode?.focus();
|
|
127
|
+
}
|
|
128
|
+
};
|
|
129
|
+
|
|
130
|
+
const isCoordinate = (anchor: any): anchor is { x: number; y: number } =>
|
|
131
|
+
!React.isValidElement(anchor) &&
|
|
132
|
+
typeof anchor?.x === 'number' &&
|
|
133
|
+
typeof anchor?.y === 'number';
|
|
134
|
+
|
|
135
|
+
const isBrowser = () => Platform.OS === 'web' && 'document' in global;
|
|
136
|
+
|
|
137
|
+
/**
|
|
138
|
+
* Menus display a list of choices on temporary elevated surfaces. Their placement varies based on the element that opens them.
|
|
139
|
+
*
|
|
140
|
+
* ## Usage
|
|
141
|
+
* ```js
|
|
142
|
+
* import * as React from 'react';
|
|
143
|
+
* import { View } from 'react-native';
|
|
144
|
+
* import { Button, Menu, Divider, PaperProvider } from 'react-native-paper';
|
|
145
|
+
*
|
|
146
|
+
* const MyComponent = () => {
|
|
147
|
+
* const [visible, setVisible] = React.useState(false);
|
|
148
|
+
*
|
|
149
|
+
* const openMenu = () => setVisible(true);
|
|
150
|
+
*
|
|
151
|
+
* const closeMenu = () => setVisible(false);
|
|
152
|
+
*
|
|
153
|
+
* return (
|
|
154
|
+
* <PaperProvider>
|
|
155
|
+
* <View
|
|
156
|
+
* style={{
|
|
157
|
+
* paddingTop: 50,
|
|
158
|
+
* flexDirection: 'row',
|
|
159
|
+
* justifyContent: 'center',
|
|
160
|
+
* }}>
|
|
161
|
+
* <Menu
|
|
162
|
+
* visible={visible}
|
|
163
|
+
* onDismiss={closeMenu}
|
|
164
|
+
* anchor={<Button onPress={openMenu}>Show menu</Button>}>
|
|
165
|
+
* <Menu.Item onPress={() => {}} title="Item 1" />
|
|
166
|
+
* <Menu.Item onPress={() => {}} title="Item 2" />
|
|
167
|
+
* <Divider />
|
|
168
|
+
* <Menu.Item onPress={() => {}} title="Item 3" />
|
|
169
|
+
* </Menu>
|
|
170
|
+
* </View>
|
|
171
|
+
* </PaperProvider>
|
|
172
|
+
* );
|
|
173
|
+
* };
|
|
174
|
+
*
|
|
175
|
+
* export default MyComponent;
|
|
176
|
+
* ```
|
|
177
|
+
*
|
|
178
|
+
* ### Note
|
|
179
|
+
* When using `Menu` within a React Native's `Modal` component, you need to wrap all
|
|
180
|
+
* `Modal` contents within a `PaperProvider` in order for the menu to show. This
|
|
181
|
+
* wrapping is not necessary if you use Paper's `Modal` instead.
|
|
182
|
+
*/
|
|
183
|
+
|
|
184
|
+
const Menu = ({
|
|
185
|
+
visible,
|
|
186
|
+
statusBarHeight,
|
|
187
|
+
overlayAccessibilityLabel = 'Close menu',
|
|
188
|
+
testID = 'menu',
|
|
189
|
+
anchor,
|
|
190
|
+
onDismiss,
|
|
191
|
+
anchorPosition,
|
|
192
|
+
contentStyle,
|
|
193
|
+
style,
|
|
194
|
+
elevation = DEFAULT_ELEVATION,
|
|
195
|
+
mode = DEFAULT_MODE,
|
|
196
|
+
children,
|
|
197
|
+
theme: themeOverrides,
|
|
198
|
+
keyboardShouldPersistTaps,
|
|
199
|
+
}: Props) => {
|
|
200
|
+
const theme = useInternalTheme(themeOverrides);
|
|
201
|
+
const insets = useSafeAreaInsets();
|
|
202
|
+
const [rendered, setRendered] = React.useState(visible);
|
|
203
|
+
const [left, setLeft] = React.useState(0);
|
|
204
|
+
const [top, setTop] = React.useState(0);
|
|
205
|
+
const [menuLayout, setMenuLayout] = React.useState({ width: 0, height: 0 });
|
|
206
|
+
const [anchorLayout, setAnchorLayout] = React.useState({
|
|
207
|
+
width: 0,
|
|
208
|
+
height: 0,
|
|
209
|
+
});
|
|
210
|
+
const [windowLayout, setWindowLayout] = React.useState({
|
|
211
|
+
width: WINDOW_LAYOUT.width,
|
|
212
|
+
height: WINDOW_LAYOUT.height,
|
|
213
|
+
});
|
|
214
|
+
|
|
215
|
+
const opacityAnimationRef = React.useRef(new Animated.Value(0));
|
|
216
|
+
const scaleAnimationRef = React.useRef(new Animated.ValueXY({ x: 0, y: 0 }));
|
|
217
|
+
const keyboardHeightRef = React.useRef(0);
|
|
218
|
+
const prevVisible = React.useRef<boolean | null>(null);
|
|
219
|
+
const anchorRef = React.useRef<View | null>(null);
|
|
220
|
+
const menuRef = React.useRef<View | null>(null);
|
|
221
|
+
const prevRendered = React.useRef(false);
|
|
222
|
+
|
|
223
|
+
const keyboardDidShow = React.useCallback((e: RNKeyboardEvent) => {
|
|
224
|
+
const keyboardHeight = e.endCoordinates.height;
|
|
225
|
+
keyboardHeightRef.current = keyboardHeight;
|
|
226
|
+
}, []);
|
|
227
|
+
|
|
228
|
+
const keyboardDidHide = React.useCallback(() => {
|
|
229
|
+
keyboardHeightRef.current = 0;
|
|
230
|
+
}, []);
|
|
231
|
+
|
|
232
|
+
const keyboardDidShowListenerRef: React.MutableRefObject<
|
|
233
|
+
EmitterSubscription | undefined
|
|
234
|
+
> = React.useRef();
|
|
235
|
+
const keyboardDidHideListenerRef: React.MutableRefObject<
|
|
236
|
+
EmitterSubscription | undefined
|
|
237
|
+
> = React.useRef();
|
|
238
|
+
|
|
239
|
+
const backHandlerSubscriptionRef: React.MutableRefObject<
|
|
240
|
+
NativeEventSubscription | undefined
|
|
241
|
+
> = React.useRef();
|
|
242
|
+
const dimensionsSubscriptionRef: React.MutableRefObject<
|
|
243
|
+
NativeEventSubscription | undefined
|
|
244
|
+
> = React.useRef();
|
|
245
|
+
|
|
246
|
+
const handleDismiss = React.useCallback(() => {
|
|
247
|
+
if (visible) {
|
|
248
|
+
onDismiss?.();
|
|
249
|
+
}
|
|
250
|
+
}, [onDismiss, visible]);
|
|
251
|
+
|
|
252
|
+
const handleKeypress = React.useCallback(
|
|
253
|
+
(e: KeyboardEvent) => {
|
|
254
|
+
if (e.key === 'Escape') {
|
|
255
|
+
onDismiss?.();
|
|
256
|
+
}
|
|
257
|
+
},
|
|
258
|
+
[onDismiss]
|
|
259
|
+
);
|
|
260
|
+
|
|
261
|
+
const removeListeners = React.useCallback(() => {
|
|
262
|
+
backHandlerSubscriptionRef.current?.remove();
|
|
263
|
+
dimensionsSubscriptionRef.current?.remove();
|
|
264
|
+
isBrowser() && document.removeEventListener('keyup', handleKeypress);
|
|
265
|
+
}, [handleKeypress]);
|
|
266
|
+
|
|
267
|
+
const attachListeners = React.useCallback(() => {
|
|
268
|
+
backHandlerSubscriptionRef.current = addEventListener(
|
|
269
|
+
BackHandler,
|
|
270
|
+
'hardwareBackPress',
|
|
271
|
+
handleDismiss
|
|
272
|
+
);
|
|
273
|
+
dimensionsSubscriptionRef.current = addEventListener(
|
|
274
|
+
Dimensions,
|
|
275
|
+
'change',
|
|
276
|
+
handleDismiss
|
|
277
|
+
);
|
|
278
|
+
Platform.OS === 'web' && document.addEventListener('keyup', handleKeypress);
|
|
279
|
+
}, [handleDismiss, handleKeypress]);
|
|
280
|
+
|
|
281
|
+
const measureMenuLayout = () =>
|
|
282
|
+
new Promise<LayoutRectangle>((resolve) => {
|
|
283
|
+
if (menuRef.current) {
|
|
284
|
+
menuRef.current.measureInWindow((x, y, width, height) => {
|
|
285
|
+
resolve({ x, y, width, height });
|
|
286
|
+
});
|
|
287
|
+
}
|
|
288
|
+
});
|
|
289
|
+
|
|
290
|
+
const measureAnchorLayout = React.useCallback(
|
|
291
|
+
() =>
|
|
292
|
+
new Promise<LayoutRectangle>((resolve) => {
|
|
293
|
+
if (isCoordinate(anchor)) {
|
|
294
|
+
resolve({ x: anchor.x, y: anchor.y, width: 0, height: 0 });
|
|
295
|
+
return;
|
|
296
|
+
}
|
|
297
|
+
|
|
298
|
+
if (anchorRef.current) {
|
|
299
|
+
anchorRef.current.measureInWindow((x, y, width, height) => {
|
|
300
|
+
resolve({ x, y, width, height });
|
|
301
|
+
});
|
|
302
|
+
}
|
|
303
|
+
}),
|
|
304
|
+
[anchor]
|
|
305
|
+
);
|
|
306
|
+
|
|
307
|
+
const show = React.useCallback(async () => {
|
|
308
|
+
const windowLayoutResult = Dimensions.get('window');
|
|
309
|
+
const [menuLayoutResult, anchorLayoutResult] = await Promise.all([
|
|
310
|
+
measureMenuLayout(),
|
|
311
|
+
measureAnchorLayout(),
|
|
312
|
+
]);
|
|
313
|
+
|
|
314
|
+
// When visible is true for first render
|
|
315
|
+
// native views can be still not rendered and
|
|
316
|
+
// measureMenuLayout/measureAnchorLayout functions
|
|
317
|
+
// return wrong values e.g { x:0, y: 0, width: 0, height: 0 }
|
|
318
|
+
// so we have to wait until views are ready
|
|
319
|
+
// and rerun this function to show menu
|
|
320
|
+
if (
|
|
321
|
+
!windowLayoutResult.width ||
|
|
322
|
+
!windowLayoutResult.height ||
|
|
323
|
+
!menuLayoutResult.width ||
|
|
324
|
+
!menuLayoutResult.height ||
|
|
325
|
+
(!anchorLayoutResult.width && !isCoordinate(anchor)) ||
|
|
326
|
+
(!anchorLayoutResult.height && !isCoordinate(anchor))
|
|
327
|
+
) {
|
|
328
|
+
requestAnimationFrame(show);
|
|
329
|
+
return;
|
|
330
|
+
}
|
|
331
|
+
|
|
332
|
+
setLeft(anchorLayoutResult.x);
|
|
333
|
+
setTop(anchorLayoutResult.y);
|
|
334
|
+
setAnchorLayout({
|
|
335
|
+
height: anchorLayoutResult.height,
|
|
336
|
+
width: anchorLayoutResult.width,
|
|
337
|
+
});
|
|
338
|
+
|
|
339
|
+
setMenuLayout({
|
|
340
|
+
height: menuLayoutResult.height,
|
|
341
|
+
width: menuLayoutResult.width,
|
|
342
|
+
});
|
|
343
|
+
|
|
344
|
+
setWindowLayout({
|
|
345
|
+
height: windowLayoutResult.height - keyboardHeightRef.current,
|
|
346
|
+
width: windowLayoutResult.width,
|
|
347
|
+
});
|
|
348
|
+
|
|
349
|
+
attachListeners();
|
|
350
|
+
const { animation } = theme;
|
|
351
|
+
Animated.parallel([
|
|
352
|
+
Animated.timing(scaleAnimationRef.current, {
|
|
353
|
+
toValue: { x: menuLayoutResult.width, y: menuLayoutResult.height },
|
|
354
|
+
duration: ANIMATION_DURATION * animation.scale,
|
|
355
|
+
easing: EASING,
|
|
356
|
+
useNativeDriver: true,
|
|
357
|
+
}),
|
|
358
|
+
Animated.timing(opacityAnimationRef.current, {
|
|
359
|
+
toValue: 1,
|
|
360
|
+
duration: ANIMATION_DURATION * animation.scale,
|
|
361
|
+
easing: EASING,
|
|
362
|
+
useNativeDriver: true,
|
|
363
|
+
}),
|
|
364
|
+
]).start(({ finished }) => {
|
|
365
|
+
if (finished) {
|
|
366
|
+
focusFirstDOMNode(menuRef.current);
|
|
367
|
+
prevRendered.current = true;
|
|
368
|
+
}
|
|
369
|
+
});
|
|
370
|
+
}, [anchor, attachListeners, measureAnchorLayout, theme]);
|
|
371
|
+
|
|
372
|
+
const hide = React.useCallback(() => {
|
|
373
|
+
removeListeners();
|
|
374
|
+
|
|
375
|
+
const { animation } = theme;
|
|
376
|
+
|
|
377
|
+
Animated.timing(opacityAnimationRef.current, {
|
|
378
|
+
toValue: 0,
|
|
379
|
+
duration: ANIMATION_DURATION * animation.scale,
|
|
380
|
+
easing: EASING,
|
|
381
|
+
useNativeDriver: true,
|
|
382
|
+
}).start(({ finished }) => {
|
|
383
|
+
if (finished) {
|
|
384
|
+
setMenuLayout({ width: 0, height: 0 });
|
|
385
|
+
setRendered(false);
|
|
386
|
+
prevRendered.current = false;
|
|
387
|
+
focusFirstDOMNode(anchorRef.current);
|
|
388
|
+
}
|
|
389
|
+
});
|
|
390
|
+
}, [removeListeners, theme]);
|
|
391
|
+
|
|
392
|
+
const updateVisibility = React.useCallback(
|
|
393
|
+
async (display: boolean) => {
|
|
394
|
+
// Menu is rendered in Portal, which updates items asynchronously
|
|
395
|
+
// We need to do the same here so that the ref is up-to-date
|
|
396
|
+
await Promise.resolve().then(() => {
|
|
397
|
+
if (display && !prevRendered.current) {
|
|
398
|
+
show();
|
|
399
|
+
return;
|
|
400
|
+
}
|
|
401
|
+
|
|
402
|
+
if (!display && prevRendered.current) {
|
|
403
|
+
hide();
|
|
404
|
+
}
|
|
405
|
+
|
|
406
|
+
return;
|
|
407
|
+
});
|
|
408
|
+
},
|
|
409
|
+
[hide, show]
|
|
410
|
+
);
|
|
411
|
+
|
|
412
|
+
React.useEffect(() => {
|
|
413
|
+
const opacityAnimation = opacityAnimationRef.current;
|
|
414
|
+
const scaleAnimation = scaleAnimationRef.current;
|
|
415
|
+
keyboardDidShowListenerRef.current = Keyboard.addListener(
|
|
416
|
+
'keyboardDidShow',
|
|
417
|
+
keyboardDidShow
|
|
418
|
+
);
|
|
419
|
+
keyboardDidHideListenerRef.current = Keyboard.addListener(
|
|
420
|
+
'keyboardDidHide',
|
|
421
|
+
keyboardDidHide
|
|
422
|
+
);
|
|
423
|
+
|
|
424
|
+
return () => {
|
|
425
|
+
removeListeners();
|
|
426
|
+
keyboardDidShowListenerRef.current?.remove();
|
|
427
|
+
keyboardDidHideListenerRef.current?.remove();
|
|
428
|
+
scaleAnimation.removeAllListeners();
|
|
429
|
+
opacityAnimation?.removeAllListeners();
|
|
430
|
+
};
|
|
431
|
+
}, [removeListeners, keyboardDidHide, keyboardDidShow]);
|
|
432
|
+
|
|
433
|
+
React.useEffect(() => {
|
|
434
|
+
if (prevVisible.current !== visible) {
|
|
435
|
+
prevVisible.current = visible;
|
|
436
|
+
|
|
437
|
+
if (visible !== rendered) {
|
|
438
|
+
setRendered(visible);
|
|
439
|
+
}
|
|
440
|
+
}
|
|
441
|
+
}, [visible, rendered]);
|
|
442
|
+
|
|
443
|
+
React.useEffect(() => {
|
|
444
|
+
updateVisibility(rendered);
|
|
445
|
+
}, [rendered, updateVisibility]);
|
|
446
|
+
|
|
447
|
+
// I don't know why but on Android measure function is wrong by 24
|
|
448
|
+
const additionalVerticalValue = Platform.select({
|
|
449
|
+
android: statusBarHeight ?? insets.top,
|
|
450
|
+
default: 0,
|
|
451
|
+
});
|
|
452
|
+
|
|
453
|
+
// We need to translate menu while animating scale to imitate transform origin for scale animation
|
|
454
|
+
const positionTransforms = [];
|
|
455
|
+
let leftTransformation = left;
|
|
456
|
+
let topTransformation =
|
|
457
|
+
!isCoordinate(anchorRef.current) && anchorPosition === 'bottom'
|
|
458
|
+
? top + anchorLayout.height
|
|
459
|
+
: top;
|
|
460
|
+
|
|
461
|
+
// Check if menu fits horizontally and if not align it to right.
|
|
462
|
+
if (left <= windowLayout.width - menuLayout.width - SCREEN_INDENT) {
|
|
463
|
+
positionTransforms.push({
|
|
464
|
+
translateX: scaleAnimationRef.current.x.interpolate({
|
|
465
|
+
inputRange: [0, menuLayout.width],
|
|
466
|
+
outputRange: [-(menuLayout.width / 2), 0],
|
|
467
|
+
}),
|
|
468
|
+
});
|
|
469
|
+
|
|
470
|
+
// Check if menu position has enough space from left side
|
|
471
|
+
if (leftTransformation < SCREEN_INDENT) {
|
|
472
|
+
leftTransformation = SCREEN_INDENT;
|
|
473
|
+
}
|
|
474
|
+
} else {
|
|
475
|
+
positionTransforms.push({
|
|
476
|
+
translateX: scaleAnimationRef.current.x.interpolate({
|
|
477
|
+
inputRange: [0, menuLayout.width],
|
|
478
|
+
outputRange: [menuLayout.width / 2, 0],
|
|
479
|
+
}),
|
|
480
|
+
});
|
|
481
|
+
|
|
482
|
+
leftTransformation += anchorLayout.width - menuLayout.width;
|
|
483
|
+
|
|
484
|
+
const right = leftTransformation + menuLayout.width;
|
|
485
|
+
// Check if menu position has enough space from right side
|
|
486
|
+
if (right > windowLayout.width - SCREEN_INDENT) {
|
|
487
|
+
leftTransformation =
|
|
488
|
+
windowLayout.width - SCREEN_INDENT - menuLayout.width;
|
|
489
|
+
}
|
|
490
|
+
}
|
|
491
|
+
|
|
492
|
+
// If the menu is larger than available vertical space,
|
|
493
|
+
// calculate the height of scrollable view
|
|
494
|
+
let scrollableMenuHeight = 0;
|
|
495
|
+
|
|
496
|
+
// Check if the menu should be scrollable
|
|
497
|
+
if (
|
|
498
|
+
// Check if the menu overflows from bottom side
|
|
499
|
+
topTransformation >=
|
|
500
|
+
windowLayout.height -
|
|
501
|
+
menuLayout.height -
|
|
502
|
+
SCREEN_INDENT -
|
|
503
|
+
additionalVerticalValue &&
|
|
504
|
+
// And bottom side of the screen has more space than top side
|
|
505
|
+
topTransformation <= windowLayout.height - topTransformation
|
|
506
|
+
) {
|
|
507
|
+
// Scrollable menu should be below the anchor (expands downwards)
|
|
508
|
+
scrollableMenuHeight =
|
|
509
|
+
windowLayout.height -
|
|
510
|
+
topTransformation -
|
|
511
|
+
SCREEN_INDENT -
|
|
512
|
+
additionalVerticalValue;
|
|
513
|
+
} else if (
|
|
514
|
+
// Check if the menu overflows from bottom side
|
|
515
|
+
topTransformation >=
|
|
516
|
+
windowLayout.height -
|
|
517
|
+
menuLayout.height -
|
|
518
|
+
SCREEN_INDENT -
|
|
519
|
+
additionalVerticalValue &&
|
|
520
|
+
// And top side of the screen has more space than bottom side
|
|
521
|
+
topTransformation >= windowLayout.height - top &&
|
|
522
|
+
// And menu overflows from top side
|
|
523
|
+
topTransformation <=
|
|
524
|
+
menuLayout.height -
|
|
525
|
+
anchorLayout.height +
|
|
526
|
+
SCREEN_INDENT -
|
|
527
|
+
additionalVerticalValue
|
|
528
|
+
) {
|
|
529
|
+
// Scrollable menu should be above the anchor (expands upwards)
|
|
530
|
+
scrollableMenuHeight =
|
|
531
|
+
topTransformation +
|
|
532
|
+
anchorLayout.height -
|
|
533
|
+
SCREEN_INDENT +
|
|
534
|
+
additionalVerticalValue;
|
|
535
|
+
}
|
|
536
|
+
|
|
537
|
+
// Scrollable menu max height
|
|
538
|
+
scrollableMenuHeight =
|
|
539
|
+
scrollableMenuHeight > windowLayout.height - 2 * SCREEN_INDENT
|
|
540
|
+
? windowLayout.height - 2 * SCREEN_INDENT
|
|
541
|
+
: scrollableMenuHeight;
|
|
542
|
+
|
|
543
|
+
// Menu is typically positioned below the element that generates it
|
|
544
|
+
// So first check if it fits below the anchor (expands downwards)
|
|
545
|
+
if (
|
|
546
|
+
// Check if menu fits vertically
|
|
547
|
+
topTransformation <=
|
|
548
|
+
windowLayout.height -
|
|
549
|
+
menuLayout.height -
|
|
550
|
+
SCREEN_INDENT -
|
|
551
|
+
additionalVerticalValue ||
|
|
552
|
+
// Or if the menu overflows from bottom side
|
|
553
|
+
(topTransformation >=
|
|
554
|
+
windowLayout.height -
|
|
555
|
+
menuLayout.height -
|
|
556
|
+
SCREEN_INDENT -
|
|
557
|
+
additionalVerticalValue &&
|
|
558
|
+
// And bottom side of the screen has more space than top side
|
|
559
|
+
topTransformation <= windowLayout.height - topTransformation)
|
|
560
|
+
) {
|
|
561
|
+
positionTransforms.push({
|
|
562
|
+
translateY: scaleAnimationRef.current.y.interpolate({
|
|
563
|
+
inputRange: [0, menuLayout.height],
|
|
564
|
+
outputRange: [-((scrollableMenuHeight || menuLayout.height) / 2), 0],
|
|
565
|
+
}),
|
|
566
|
+
});
|
|
567
|
+
|
|
568
|
+
// Check if menu position has enough space from top side
|
|
569
|
+
if (topTransformation < SCREEN_INDENT) {
|
|
570
|
+
topTransformation = SCREEN_INDENT;
|
|
571
|
+
}
|
|
572
|
+
} else {
|
|
573
|
+
positionTransforms.push({
|
|
574
|
+
translateY: scaleAnimationRef.current.y.interpolate({
|
|
575
|
+
inputRange: [0, menuLayout.height],
|
|
576
|
+
outputRange: [(scrollableMenuHeight || menuLayout.height) / 2, 0],
|
|
577
|
+
}),
|
|
578
|
+
});
|
|
579
|
+
|
|
580
|
+
topTransformation +=
|
|
581
|
+
anchorLayout.height - (scrollableMenuHeight || menuLayout.height);
|
|
582
|
+
|
|
583
|
+
const bottom =
|
|
584
|
+
topTransformation +
|
|
585
|
+
(scrollableMenuHeight || menuLayout.height) +
|
|
586
|
+
additionalVerticalValue;
|
|
587
|
+
|
|
588
|
+
// Check if menu position has enough space from bottom side
|
|
589
|
+
if (bottom > windowLayout.height - SCREEN_INDENT) {
|
|
590
|
+
topTransformation =
|
|
591
|
+
scrollableMenuHeight === windowLayout.height - 2 * SCREEN_INDENT
|
|
592
|
+
? -SCREEN_INDENT * 2
|
|
593
|
+
: windowLayout.height -
|
|
594
|
+
menuLayout.height -
|
|
595
|
+
SCREEN_INDENT -
|
|
596
|
+
additionalVerticalValue;
|
|
597
|
+
}
|
|
598
|
+
}
|
|
599
|
+
|
|
600
|
+
const shadowMenuContainerStyle = {
|
|
601
|
+
opacity: opacityAnimationRef.current,
|
|
602
|
+
transform: [
|
|
603
|
+
{
|
|
604
|
+
scaleX: scaleAnimationRef.current.x.interpolate({
|
|
605
|
+
inputRange: [0, menuLayout.width],
|
|
606
|
+
outputRange: [0, 1],
|
|
607
|
+
}),
|
|
608
|
+
},
|
|
609
|
+
{
|
|
610
|
+
scaleY: scaleAnimationRef.current.y.interpolate({
|
|
611
|
+
inputRange: [0, menuLayout.height],
|
|
612
|
+
outputRange: [0, 1],
|
|
613
|
+
}),
|
|
614
|
+
},
|
|
615
|
+
],
|
|
616
|
+
borderRadius: theme.roundness,
|
|
617
|
+
...(!theme.isV3 && { elevation: 8 }),
|
|
618
|
+
...(scrollableMenuHeight ? { height: scrollableMenuHeight } : {}),
|
|
619
|
+
};
|
|
620
|
+
|
|
621
|
+
const positionStyle = {
|
|
622
|
+
top: isCoordinate(anchor)
|
|
623
|
+
? topTransformation
|
|
624
|
+
: topTransformation + additionalVerticalValue,
|
|
625
|
+
...(I18nManager.getConstants().isRTL
|
|
626
|
+
? { right: leftTransformation }
|
|
627
|
+
: { left: leftTransformation }),
|
|
628
|
+
};
|
|
629
|
+
|
|
630
|
+
const pointerEvents = visible ? 'box-none' : 'none';
|
|
631
|
+
|
|
632
|
+
return (
|
|
633
|
+
<View ref={(ref) => (anchorRef.current = ref)} collapsable={false}>
|
|
634
|
+
{isCoordinate(anchor) ? null : anchor}
|
|
635
|
+
{rendered ? (
|
|
636
|
+
<Portal>
|
|
637
|
+
<Pressable
|
|
638
|
+
accessibilityLabel={overlayAccessibilityLabel}
|
|
639
|
+
accessibilityRole="button"
|
|
640
|
+
onPress={onDismiss}
|
|
641
|
+
style={styles.pressableOverlay}
|
|
642
|
+
/>
|
|
643
|
+
<View
|
|
644
|
+
ref={(ref) => (menuRef.current = ref)}
|
|
645
|
+
collapsable={false}
|
|
646
|
+
accessibilityViewIsModal={visible}
|
|
647
|
+
style={[styles.wrapper, positionStyle, style]}
|
|
648
|
+
pointerEvents={pointerEvents}
|
|
649
|
+
onAccessibilityEscape={onDismiss}
|
|
650
|
+
testID={`${testID}-view`}
|
|
651
|
+
>
|
|
652
|
+
<Animated.View
|
|
653
|
+
pointerEvents={pointerEvents}
|
|
654
|
+
style={{
|
|
655
|
+
transform: positionTransforms,
|
|
656
|
+
}}
|
|
657
|
+
>
|
|
658
|
+
<Surface
|
|
659
|
+
mode={mode}
|
|
660
|
+
pointerEvents={pointerEvents}
|
|
661
|
+
style={[
|
|
662
|
+
styles.shadowMenuContainer,
|
|
663
|
+
shadowMenuContainerStyle,
|
|
664
|
+
theme.isV3 && {
|
|
665
|
+
backgroundColor:
|
|
666
|
+
theme.colors.elevation[ELEVATION_LEVELS_MAP[elevation]],
|
|
667
|
+
},
|
|
668
|
+
contentStyle,
|
|
669
|
+
]}
|
|
670
|
+
{...(theme.isV3 && { elevation })}
|
|
671
|
+
testID={`${testID}-surface`}
|
|
672
|
+
theme={theme}
|
|
673
|
+
>
|
|
674
|
+
{(scrollableMenuHeight && (
|
|
675
|
+
<ScrollView
|
|
676
|
+
keyboardShouldPersistTaps={keyboardShouldPersistTaps}
|
|
677
|
+
>
|
|
678
|
+
{children}
|
|
679
|
+
</ScrollView>
|
|
680
|
+
)) || <React.Fragment>{children}</React.Fragment>}
|
|
681
|
+
</Surface>
|
|
682
|
+
</Animated.View>
|
|
683
|
+
</View>
|
|
684
|
+
</Portal>
|
|
685
|
+
) : null}
|
|
686
|
+
</View>
|
|
687
|
+
);
|
|
688
|
+
};
|
|
689
|
+
|
|
690
|
+
Menu.Item = MenuItem;
|
|
691
|
+
|
|
692
|
+
const styles = StyleSheet.create({
|
|
693
|
+
wrapper: {
|
|
694
|
+
position: 'absolute',
|
|
695
|
+
},
|
|
696
|
+
shadowMenuContainer: {
|
|
697
|
+
opacity: 0,
|
|
698
|
+
paddingVertical: 8,
|
|
699
|
+
},
|
|
700
|
+
pressableOverlay: {
|
|
701
|
+
...Platform.select({
|
|
702
|
+
web: {
|
|
703
|
+
cursor: 'auto',
|
|
704
|
+
},
|
|
705
|
+
}),
|
|
706
|
+
...StyleSheet.absoluteFillObject,
|
|
707
|
+
width: '100%',
|
|
708
|
+
},
|
|
709
|
+
});
|
|
710
|
+
|
|
711
|
+
export default Menu;
|