Source

externalCode/navigation.js

  1. import * as React from "react";
  2. import {NavigationService} from "./types";
  3. /**
  4. * @typedef {"Auth" | "noAuth" | "Main" | "All"} NavigatorNames
  5. */
  6. /**
  7. * @typedef {"LoginScreen" | "SignupScreen" | "CodeVerificationScreen" | "LinkAccountScreen" | "SocialLoginDataInputScreen" | "ForgotScreen" | "SocialLoginScreen"} AuthStackScreenNames
  8. */
  9. /**
  10. * @typedef {"SettingsAboutScreen" | "ProfileGroupSingleScreen" | "SurveyScreen" | "ActivitiesCreatePostScreen" | "MessagesCreatePostScreen" | "QuizSingleScreen" | "QuizResults" | "QuizDetails" | "QuizReview" | "CourseSectionScreen" | "CourseQuizzesScreen" | "CourseMaterialsScreen" | "BBMediaFullView" | "ActivityMedia" | "MergeTopicScreen" | "LessonSingleScreen" | "LearnTopicSingleScreen" | "SubForumsSingleScreen" | "ForumsSingleScreen" | "CoursesSingleScreen" | "CoursesScreen" | "CoursesCategorySingleScreen" | "TopicsTagScreen" | "TopicsSingleScreen" | "AddTopicScreen" | "SelectScreen" | "MoveReplyScreen" | "SplitReplyScreen" | "NewReplyScreen" | "CreateNewAlbum" | "AddNewProfilePhoto" | "AddNewGroupPhoto" | "SettingsPushScreen" | "SettingsPrivacyScreen" | "GroupsSingleScreen" | "PageScreen" | "BlockScreen" | "ProfileScreen" | "PrivacySettingsScreen" | "GroupInviteSettingsScreen" | "LoginInfoScreen" | "AlbumSingleScreen" | "CreateNewPhotosScreen" | "AlbumAddPhotosScreen" | "HomeNotificationsScreen" | "HomeForumsScreen" | "HomeCourseCategoriesScreen" | "HomeTopicsScreen" | "HomeGroupsScreen" | "HomeMemberScreen" | "HomeActivityScreen" | "HomeMembersScreen" | "ReportFormScreen" | "ManageGroupDetails" | "EditNavigation" | "EditXprofile" | "ViewXprofile" | "PhotoXprofile" | "CoverXprofile" | "ProfileCourses" | "ProfilePhotos" | "GlobalPhotos" | "ProfileGamipress_achievements" | "ProfileGamipress_points" | "ProfileGamipress_ranks" | "ProfileBadgeos_achievements" | "ProfileBadgeos_points" | "ProfileBadgeos_ranks" | "ProfileActivities" | "ProfileCertificates" | "ProfileResults" | "ProfileBlog" | "ProfileXprofile" | "ProfileInvites" | "ProfileFriends" | "ProfileForums" | "ProfileGroups" | "GroupMessages" | "GroupMessageCreatePostScreen" | "GroupActivity" | "GroupPhotos" | "GroupAdmin" | "GroupManageSettings" | "GroupManageSettingsForm" | "GroupManageCoverPhoto" | "GroupManageMembers" | "GroupManagePhoto" | "GroupManageDelete" | "GroupManageDetails" | "GroupMembers" | "GroupSubgroups" | "GroupInvite" | "SelectGroupMembers" | "PendingInvites" | "SubmitInvite" | "GroupRequests" | "GroupCourses" | "ProductSingleScreen" | "ActivitySingleScreen" | "BlogSingleScreen" | "BlogSingleDeeplink" | "BlogReplyScreen" | "MyLibraryScreen" | "DownloadedCoursesScreen" | "BlogScreen" | "SettingsScreen" | "GroupsScreen" | "NotificationsScreen" | "TopicsScreen" | "CourseCategoriesScreen" | "ActivitiesScreen" | "MessagesScreen" | "MembersScreen" | "ForumsScreen" | "ProductsScreen" | "EmailInvitesSendScreen" | "EmailInvitesSentScreen" | "EmailInviteMessageScreen" | "SendFeedbackScreen" | "ReportBugScreen"} MainStackScreenNames
  11. */
  12. /**
  13. * @typedef {"my_library" | "blog" | "settings" | "groups" | "notifications" | "topics" | "courses" | "courses_category" | "courses_all" | "activity" | "documents" | "messages" | "members" | "forums" | "profile" | "iap_products" | "photos" | "videos" | "gamipress_achievements" | "badgeos_achievements" | "course_certificates" } MainMenuScreenNames
  14. */
  15. /**
  16. * @typedef {AuthStackScreenNames | MainStackScreenNames | MainMenuScreenNames} ScreenName
  17. */
  18. /**
  19. * @typedef {Object} Navigator
  20. * @property {NavigationService} navigation Navigation object
  21. * @property {Function} dispatch Dispatch function
  22. */
  23. /**
  24. * @typedef {Object} RoutesObject
  25. * @example
  26. * {
  27. * AppLockScreen: {screen: AppLockNavigator},
  28. * Main: {screen: MainScreenNavigator, navigationOptions: {header: null}},
  29. * ...
  30. * }
  31. */
  32. /**
  33. * @typedef {Function} AfterAuthRoutesFilter
  34. * @param {RoutesObject} routes Current routes
  35. * @return {RoutesObject} New routes
  36. */
  37. /**
  38. * @typedef {Function} ShouldUpdateCallback
  39. * @param {Object} thisProps Current props
  40. * @param {Object} nextProps Next props
  41. * @param {Object} thisState Current state
  42. * @param {Object} nextState Next state
  43. * @return {Boolean} Return `true` if component should update
  44. */
  45. /**
  46. * @typedef {Function} BottomTabsSettingsFilterCallback
  47. * @param {Object} tabsOptions `tabsOptions` include `tabBarOptions`, `initialRouteName` `tabBarComponent` etc..
  48. * @param {Object} colors App colors
  49. * @return {Object} New tab settings
  50. */
  51. /**
  52. * @typedef {Function} BottomTabsRoutesFilterCallback
  53. * @param {Object} routes Current bottom tab routes
  54. * @param {Function} constructTabbarAllScreens Returns the appropriate screens depending on value of "Tab Bar Visibility"
  55. * @param {Boolean} showTabBarOnAllScreens Returns `true` if "Tab Bar Visibility" is set to "Show on All Screens"
  56. * @return {Object} New routes
  57. */
  58. /**
  59. * @typedef {Function} InitialAfterAuthRouteCallback
  60. * @param {Object} props AppNavigator component props
  61. * @return {String} Route
  62. */
  63. /**
  64. * @typedef {Function} NavigatorEventListenerCallback
  65. * @param {Navigator} navigator Root navigator object
  66. */
  67. /**
  68. * @typedef {Function} HOCsFilter
  69. * @param {Array<Function>} hocs Initial HOCs array
  70. * @return {Array<Function>} New HOCs array
  71. */
  72. /**
  73. * @typedef {Function} MapStateToProps
  74. * @param {Object} state - Redux state
  75. * @return {Object} - Props
  76. */
  77. /**
  78. * @typedef {Object} animatedSwitchNavigatorProps
  79. * @property {Object} routes Default navigation routes
  80. * @property {Object} options Default navigation options
  81. * @property {Object} routeProps Contains information which can be utilized when creating custom navigators
  82. * @return {animatedSwitchNavigatorReturn}
  83. */
  84. /**
  85. * @typedef {Object} animatedSwitchNavigatorReturn
  86. * @property {Object} routes New navigation routes
  87. * @property {Object} options New navigation options
  88. */
  89. /**
  90. * @typedef {Object} IconProps
  91. * @property {Object} item Route information
  92. * @property {Boolean} focused Returns `true` if icon is currently active
  93. * @property {String} tintColor Icon color
  94. * @property {Function} calcFontSize Used to calculate font size based on the fixed value you assign and percent set on branding options
  95. * @property {Object} colors App colors
  96. */
  97. /**
  98. * @typedef {Object} LabelProps
  99. * @property {Object} props Route information
  100. * @property {Array<Object>} textStyle Default style for the text component
  101. */
  102. /**
  103. * @typedef {Object} BottomTabBarProps
  104. * @property {Object} state Route state information
  105. * @property {Object} descriptors Bottom tab bar menu items
  106. * @property {NavigationService} navigation
  107. */
  108. /**
  109. * @class
  110. * Navigation Hooks.
  111. * Instance name: navigationApi
  112. You can use this hook to customize app navigation aspects such as adding a new route to the app react-navigation tree, adding new HOC (Higher-Order Components) and more.
  113. * @example
  114. * externalCodeSetup.navigationApi.METHOD_NAME
  115. */
  116. export class NavigationApi {
  117. /**
  118. * @ignore
  119. * The app shows the SocialLoginScreen when this property is set to `true`
  120. * Note that social login needs to be enabled and configured at the backend
  121. * beforehand.
  122. * @private
  123. * @type {Boolean}
  124. */
  125. socialLoginEnabled = false;
  126. /**
  127. * @ignore Social login support removed
  128. * Enables social login (if social login is enabled and configured at the backend beforehand)
  129. * @method
  130. */
  131. enableSocialLogin = () => {
  132. this.socialLoginEnabled = true;
  133. };
  134. /**
  135. * @deprecated
  136. */
  137. filterMultisiteDrawerSettings = (switchSite, options) => options;
  138. setFilterMultisiteDrawerSettings = filterMultisiteDrawerSettings => {
  139. this.filterMultisiteDrawerSettings = filterMultisiteDrawerSettings;
  140. };
  141. /**
  142. * @deprecated
  143. */
  144. filterInitialAfterAuthRoute = (props, initialAfterAuthRoute) =>
  145. initialAfterAuthRoute;
  146. setFilterInitialAfterAuthRoute = filterInitialAfterAuthRoute => {
  147. this.filterInitialAfterAuthRoute = filterInitialAfterAuthRoute;
  148. };
  149. /**
  150. * @deprecated
  151. * We don't use default icons anymore. All icons are provided by plugin.
  152. */
  153. customIcons = {};
  154. addCustomIcons = iconsMap => {
  155. Object.assign(this.customIcons, iconsMap);
  156. };
  157. /**
  158. * @deprecated
  159. */
  160. setShowIosBottomTabsLabels = shouldShow => {
  161. console.warn(
  162. "This hook is deprecated. " +
  163. "Use globalSettings.app_menu.ios.show_labels and globalSettings.app_menu.android.show_labels instead"
  164. );
  165. };
  166. /**
  167. * @deprecated
  168. */
  169. composeHooks = newHooks => newHooks;
  170. /**
  171. * @deprecated
  172. */
  173. AddComposeHooks = composeHooks => {
  174. console.warn("This hook is deprecated. Use addComposeHocs");
  175. this.composeHooks = composeHooks;
  176. };
  177. /**
  178. * @deprecated
  179. * We use a custom TabBar component now.
  180. */
  181. iosBottomTabsIconRenderer = null;
  182. setIosBottomTabsIcon = renderFunction => {
  183. console.warn("This hook is depreciated");
  184. this.iosBottomTabsIconRenderer = renderFunction;
  185. };
  186. /**
  187. *
  188. * Display BlurView in IOS TabBar
  189. */
  190. isBlurViewInIOSTabBar = false;
  191. enableBlurViewInIOSTabBar = () => {
  192. this.isBlurViewInIOSTabBar = true;
  193. };
  194. /**
  195. * Used to provide additional component update logic
  196. * @private
  197. */
  198. shouldComponentUpdate = (thisProps, nextProps, thisState, nextState) =>
  199. undefined;
  200. /**
  201. * Used to provide custom update logic to AppNavigator.
  202. * This function is called in shouldComponentUpdate.
  203. * If this hook returns a boolean, the component will use this to determine if it should update.
  204. * For more info about shouldComponentUpdate function, refer to {@link https://reactjs.org/docs/react-component.html#shouldcomponentupdate}
  205. * @method
  206. * @param {ShouldUpdateCallback} updateFunc
  207. * @example
  208. * externalCodeSetup.navigationApi.setShouldComponentUpdate((thisProps, nextProps, thisState, nextState) => {
  209. * //If condition...
  210. * return true;
  211. * })
  212. */
  213. setShouldComponentUpdate = updateFunc => {
  214. this.shouldComponentUpdate = updateFunc;
  215. };
  216. /**
  217. * Used to provide a custom initial after auth route
  218. * Takes AppNavigator props and returns a route name
  219. * Return null to use default initial route logic
  220. * @private
  221. */
  222. initialAfterAuthRoute = props => null;
  223. /**
  224. * By default we navigate to the "Main" stack after navigation.
  225. * This hook is used to provide a different route after login.
  226. * It uses the callback function `func` that will receive AppNavigator props and return a route name as a string.
  227. * More examples at: {@link https://www.buddyboss.com/resources/dev-docs/app-development/extending-the-buddyboss-app/creating-new-screens/}
  228. * @method
  229. * @param {InitialAfterAuthRouteCallback} func
  230. * @return {ScreenName} Return string screen name
  231. * @example <caption> Show products screen after user authentication </caption>
  232. * externalCodeSetup.navigationApi.setInitialAfterAuthRoute( props => {
  233. * if (props.auth.isLoggedIn){
  234. * return "ProductsScreen";
  235. * }
  236. * })
  237. */
  238. setInitialAfterAuthRoute = func => {
  239. this.initialAfterAuthRoute = func;
  240. };
  241. /**
  242. * Used to provide custom state props
  243. * Takes redux state as an argument and returns an object mapping state to props
  244. * @private
  245. */
  246. mapStateToProps = state => ({});
  247. /**
  248. * Used to provide AppNavigator component custom redux props.
  249. * It uses the `mapStateFunc` callback function that should receive redux state as an argument and return state props as an object.
  250. * @method
  251. * @param {MapStateToProps} mapStateFunc
  252. * @example <caption> Add `blog` redux state and a localDate field to AppNavigator component props </caption>
  253. * externalCodeSetup.navigationApi.setMapStateToProps( props => {
  254. * return {...props, blog: props.blog, localDate: new Date()}
  255. * })
  256. *
  257. * //These new props can now be used in AppNavigator functions. For example:
  258. * externalCodeSetup.navigationApi.setInitialAfterAuthRoute( props => {
  259. * if (props.localDate < foo)
  260. * return "CustomScreen"
  261. * })
  262. *
  263. */
  264. setMapStateToProps = mapStateFunc => {
  265. this.mapStateToProps = mapStateFunc;
  266. };
  267. /**
  268. * @ignore
  269. * @private
  270. */
  271. defaultStackNavigatorConfig = {};
  272. /**
  273. * @ignore
  274. * Reason for ignoring: A lot of stackNavigator options can't be used in the app such as setting a header style
  275. * Each screen in the TabBarNavigator is wrapped in a StackNavigator.
  276. * This hook is used to provide a custom stackConfigs all of these StackNavigators.
  277. * For more info on how setting config for defaultStackNavigator, see {@link https://reactnavigation.org/docs/4.x/stack-navigator}
  278. * @method
  279. * @param {Object} config Configuration object
  280. */
  281. setDefaultStackNavigatorConfig = config => {
  282. this.defaultStackNavigatorConfig = config;
  283. };
  284. /**
  285. * @private
  286. */
  287. newNavigationRoutes = [];
  288. /**
  289. * Add a new route to react-navigation tree.
  290. * NOTE: The route will be added to the root navigator if the parentNavigator value is missing.
  291. * More information can be found at: {@link https://www.buddyboss.com/resources/dev-docs/app-development/extending-the-buddyboss-app/creating-new-screens/}
  292. * @method
  293. * @param {String} id Key of route in route map
  294. * @param {String} routeName Route name in react-navigation tree
  295. * @param {React.ComponentType<any>} component Route component
  296. * @param {NavigatorNames} parentNavigator Parent navigator name; can be one of the NavigatorNames values.
  297. * @example <caption> Register a new route and navigate to that route </caption>
  298. *
  299. * //After adding BookScreen as a new navigation route, you can now navigate to this screen using NavigationService
  300. * externalCodeSetup.navigationApi.addNavigationRoute(
  301. * "book",
  302. * "BookScreen",
  303. * () => <Text>Custom Book Screen</Text>,
  304. * "All"
  305. * );
  306. *
  307. * //Example on how to navigate to BookScreen
  308. * //Since "book" is now registered, app can now navigate to that route
  309. * const MyCustomLoginScreen = (props) => (
  310. * <View>
  311. * <Text> This is a custom login screen</Text>
  312. * <Button title="Go To Book Screen" onPress={() => props.navigation.navigate("book")}/>
  313. * </View>
  314. * )
  315. * externalCodeSetup.navigationApi.replaceScreenComponent(
  316. * "LoginScreen",
  317. * MyCustomLoginScreen
  318. * );
  319. */
  320. addNavigationRoute = (id, routeName, component, parentNavigator) => {
  321. this.newNavigationRoutes.push({id, routeName, component, parentNavigator});
  322. };
  323. /**
  324. * @private
  325. */
  326. screenReplacements = {};
  327. /**
  328. * Replaces a certain screen in the app.
  329. * More information at {@link https://www.buddyboss.com/resources/dev-docs/app-development/extending-the-buddyboss-app/creating-new-screens/}
  330. * @method
  331. * @param {ScreenName} componentName The name of the screen registered in the core BuddyBoss App navigation.
  332. * @param {React.ComponentType<any>} replaceWith Your custom screen component.
  333. * @example <caption> Replace LoginScreen with MyScreen </caption>
  334. *
  335. * const MyScreen = (props) => (
  336. * <View style={{flex: 1}}>
  337. * <Text>This is my screen</Text>
  338. * </View>
  339. * );
  340. *
  341. *
  342. * externalCodeSetup.navigationApi.replaceScreenComponent("LoginScreen", MyScreen);
  343. */
  344. replaceScreenComponent = (componentName, replaceWith) => {
  345. this.screenReplacements[componentName] = replaceWith;
  346. };
  347. /**
  348. * @private
  349. * @property {Array<NavigatorEventListenerCallback>} navigatorCreatedCallbacks
  350. */
  351. navigatorCreatedCallbacks = [];
  352. /**
  353. * Use to subscribe to root navigator created event.
  354. * This will return the navigator object.
  355. * Useful for dispatching navigation events outside of navigation tree components or getting state of navigator
  356. * @method
  357. * @param {NavigatorEventListenerCallback} onNavigatorCreated
  358. * @returns {UnsubscribeFunction} Unsubscribe function
  359. * @example
  360. * externalCodeSetup.navigationApi.addNavigatorCreatedCallback(props => {
  361. * Alert.alert("Navigator created!")
  362. * })
  363. */
  364. addNavigatorCreatedCallback = onNavigatorCreated => {
  365. this.navigatorCreatedCallbacks.push(onNavigatorCreated);
  366. return () => {
  367. const index = this.navigatorCreatedCallbacks.indexOf(onNavigatorCreated);
  368. if (index !== -1) {
  369. this.navigatorCreatedCallbacks.splice(index, 1);
  370. }
  371. };
  372. };
  373. /**
  374. *
  375. * @param {Object} tabsOptions
  376. * @param {Object} colors
  377. * @private
  378. */
  379. filterBottomTabsSettings = (tabsOptions, colors) => tabsOptions;
  380. /**
  381. * You can use this hook to configure `createBottomTabNavigator` from react-navigation.
  382. * `tabsOptions` and `colors` objects are provided which you can use to return a new settings object.
  383. * @method
  384. * @param {BottomTabsSettingsFilterCallback} filterBottomTabsSettings
  385. * @example <caption> Change bottom tab bar's active tint color </caption>
  386. * externalCodeSetup.navigationApi.setFilterBottomTabsSettings((tabsOptions, colors) => {
  387. * return {
  388. * ...tabsOptions,
  389. * tabBarOptions: {
  390. * ...tabsOptions.tabBarOptions,
  391. * tabBarActiveTintColor: colors.headerBg
  392. * }
  393. * }
  394. * })
  395. */
  396. setFilterBottomTabsSettings = filterBottomTabsSettings => {
  397. this.filterBottomTabsSettings = filterBottomTabsSettings;
  398. };
  399. /**
  400. *
  401. * @param {Object} routes
  402. * @private
  403. */
  404. filterBottomTabsRoutes = routes => routes;
  405. /**
  406. * You can use this hook to configure `createBottomTabNavigator` from react-navigation.
  407. * `routes` object is provided which you can use to return a new routes object.
  408. * @method
  409. * @param {BottomTabsRoutesFilterCallback} filterBottomTabsRoutes
  410. * @example <caption> Create a custom More Screen </caption>
  411. *
  412. * import React from "react";
  413. * import {Platform} from "react-native";
  414. * import MoreScreen from "@src/containers/Custom/MoreScreen";
  415. * import FontManager from "@src/FontManager";
  416. * import {NAV_HEIGHT, BARHEIGHT, isHaveDynamicIsland} from "@src/styles/global";
  417. * import {SEARCH_HEIGHT} from "@src/components/Search";
  418. * import {useSearchTransition} from "@src/components/listUtils";
  419. *
  420. * export const applyCustomCode = externalCodeSetup => {
  421. *
  422. * const MyCustomScreen = props => {
  423. * const showSearch = true;
  424. * const SPACING_TOP = FontManager.applyFontHeightAdjustment(NAV_HEIGHT + 17);
  425. * const INITIAL_SCROLL = -SPACING_TOP + SEARCH_HEIGHT;
  426. * const {listTopMargin} = useSearchTransition(
  427. * showSearch,
  428. * false,
  429. * Platform.select({
  430. * ios: NAV_HEIGHT - BARHEIGHT - (isHaveDynamicIsland() ? 20 : 16),
  431. * android: NAV_HEIGHT - 26 - BARHEIGHT
  432. * })
  433. * );
  434. *
  435. * const searchContainerPaddingTop = listTopMargin.value + (NAV_HEIGHT + 17);
  436. *
  437. * return (
  438. * <MoreScreen
  439. * screenTitle="More Menus"
  440. * containerPaddingTop={NAV_HEIGHT}
  441. * contentInsetTop={SPACING_TOP}
  442. * contentOffsetY={INITIAL_SCROLL}
  443. * searchContainerPaddingTop={searchContainerPaddingTop}
  444. * showSearch={showSearch}
  445. * />
  446. * );
  447. * };
  448. *
  449. * externalCodeSetup.navigationApi.setFilterBottomTabsRoutes(routes => {
  450. * return {
  451. * ...routes,
  452. * MoreScreen: {
  453. * ...routes.MoreScreen.options,
  454. * screen: {
  455. * ...routes.MoreScreen.screen,
  456. * component: MyCustomScreen
  457. * }
  458. * }
  459. * };
  460. * });
  461. * };
  462. *
  463. */
  464. setFilterBottomTabsRoutes = filterBottomTabsRoutes => {
  465. this.filterBottomTabsRoutes = filterBottomTabsRoutes;
  466. };
  467. filterAfterAuthRoutes = routes => routes;
  468. /**
  469. * You can use this hook to filter possible changes after authentication routes.
  470. * `filterAfterAuthRoutes` takes current routes object and returns a new one.
  471. * For more info, see {@link https://www.buddyboss.com/resources/dev-docs/app-development/extending-the-buddyboss-app/creating-new-screens/}
  472. * @method
  473. * @param {AfterAuthRoutesFilter} filterAfterAuthRoutes
  474. * @example <caption> Add custom screen to auth route then navigate to custom screen</caption>
  475. *
  476. * ...
  477. *
  478. * import {withNavigation} from "@src/components/hocs/withNavigation";
  479. *
  480. * export const applyCustomCode = externalCodeSetup => {
  481. * const MyWelcomeScreen = withNavigation(props => {
  482. * return (
  483. * <View style={{flex: 1, justifyContent: "center", alignSelf: "center"}}>
  484. * <Text> Welcome to our App! New adventure awaits!</Text>
  485. * <Button
  486. * title="Go to App"
  487. * onPress={() => props.navigation.navigate("Main")}
  488. * />
  489. * </View>
  490. * );
  491. * });
  492. *
  493. *
  494. * externalCodeSetup.navigationApi.setFilterAfterAuthRoutes(afterAuthRoutes => {
  495. * return {
  496. * ...afterAuthRoutes,
  497. * MyWelcomeScreen: {
  498. * screen: MyWelcomeScreen
  499. * }
  500. * };
  501. * });
  502. *
  503. * externalCodeSetup.navigationApi.setInitialAfterAuthRoute(props => {
  504. * return "MyWelcomeScreen";
  505. * });
  506. * };
  507. */
  508. setFilterAfterAuthRoutes = filterAfterAuthRoutes => {
  509. this.filterAfterAuthRoutes = filterAfterAuthRoutes;
  510. };
  511. composeHocs = hocs => hocs;
  512. /**
  513. * Use this hook to register new HOCs (Higher-Order Components) which will be applied to the AppNavigator component.
  514. * For more info on how to use HOCs, refer to: {@link https://reactjs.org/docs/higher-order-components.html}
  515. * @method
  516. * @param {HOCsFilter} composeHocs
  517. * @example <caption> Add new HOC </caption>
  518. * externalCodeSetup.navigationApi.addComposeHocs(hocs => {
  519. * return [
  520. * ...hocs,
  521. * withMyCustomHoc
  522. * ];
  523. * })
  524. */
  525. addComposeHocs = composeHocs => {
  526. this.composeHocs = composeHocs;
  527. };
  528. bottomTabBarIcon = (icon, props) => icon;
  529. /**
  530. * You can use this to replace the bottom tab bar icons to display your preferred icons.
  531. * @method
  532. * @param {React.ComponentType<Icon>} Icon Tab icon component
  533. * @param {IconProps} props
  534. * @example
  535. * import React from "react";
  536. * import Icon from "@src/components/Icon";
  537. * import BadgeIcon from "@src/components/BadgeIcon";
  538. * import {getIcon} from "@src/navigators/util";
  539. * import {glyphMap as bbIconGlyphMap} from "@src/components/BBIcon";
  540. * export const applyCustomCode = (externalCodeSetup) => {
  541. * externalCodeSetup.navigationApi.setBottomTabBarIcon(
  542. * (icon, props) => {
  543. * const {tintColor, focused, calcFontSize, colors, item} = props;
  544. *
  545. * if (!item) return null;
  546. *
  547. * const {color, menuIcon} = getIcon(
  548. * item,
  549. * bbIconGlyphMap,
  550. * focused,
  551. * tintColor
  552. * );
  553. *
  554. * if (item.object === "notifications" || item.object === "messages") {
  555. * return (
  556. * <BadgeIcon
  557. * calcFontSize={calcFontSize}
  558. * tintColor={color}
  559. * bottomTabsBg={colors.bottomTabsBg}
  560. * warningColor={colors.warningColor}
  561. * foregroundColor={item.icon.fg_color}
  562. * platform="ios"
  563. * inMore={false}
  564. * app={"learnerapp"}
  565. * type={item.object}
  566. * icon={menuIcon}
  567. * styles={{height: 25, width: 25}}
  568. * />
  569. * );
  570. * }
  571. *
  572. * return (
  573. * <Icon
  574. * icon={menuIcon}
  575. * foregroundColor={item.icon.fg_color}
  576. * tintColor={color}
  577. * styles={{height: 25, width: 25}}
  578. * />
  579. * );
  580. * }
  581. * );
  582. * }
  583. */
  584. setBottomTabBarIcon = bottomTabBarIcon => {
  585. this.bottomTabBarIcon = bottomTabBarIcon;
  586. };
  587. bottomTabBarLabel = (label, labelProps) => label;
  588. /**
  589. * You can use this to replace the bottom tab bar label to display your preferred labels.
  590. * @method
  591. * @param {React.ComponentType<Text>} Label Tab label component
  592. * @param {LabelProps} labelProps
  593. *
  594. * externalCodeSetup.navigationApi.setBottomTabBarLabel((label, labelProps) => (
  595. * <Text
  596. * style={labelProps.textStyle}
  597. * numberOfLines={1}
  598. * ellipsizeMode={"tail"}
  599. * allowFontScaling={false}
  600. * >
  601. * {labelProps.props.item.label}
  602. * </Text>
  603. * ));
  604. */
  605. setBottomTabBarLabel = bottomTabBarLabel => {
  606. this.bottomTabBarLabel = bottomTabBarLabel;
  607. };
  608. bottomTabBar = null;
  609. /**
  610. * You can use this to replace the bottom tab bar component according to your preference.
  611. * For more information regarding the BottomTabBar, you can use the following link: {@link https://reactnavigation.org}
  612. * @method
  613. * @param {React.ComponentType<BottomTabBarProps>} BottomTabBarProps
  614. * @example <caption> Create your own bottom tab bar </caption>
  615. *
  616. * import React from "react";
  617. * import {Text, View, TouchableOpacity} from "react-native";
  618. *
  619. * import {isTabletOrIPad} from "@src/utils";
  620. * import {
  621. * FontWeights,
  622. * textRTLStyleFix,
  623. * correctBottomSafeArea
  624. * } from "@src/styles/global";
  625. * import {useSafeAreaInsets} from "react-native-safe-area-context";
  626. *
  627. * import Icon from "@src/components/Icon";
  628. * import BadgeIcon from "@src/components/BadgeIcon";
  629. * import {getMenuFontIconVariant} from "@src/navigators/util";
  630. * import {glyphMap as bbIconGlyphMap} from "@src/components/BBIcon";
  631. * import withGlobalStyles from "@src/components/hocs/withGlobalStyles";
  632. * import {useScreenProps} from "@src/navigators/v5/ScreenPropsProvider";
  633. *
  634. * export const applyCustomCode = externalCodeSetup => {
  635. * const renderIcon = props => {
  636. * const {calcFontSize} = useScreenProps(["calcFontSize"]);
  637. *
  638. * const {focused, item, tabBarOptions, colors} = props;
  639. * const {tabBarActiveTintColor} = tabBarOptions;
  640. *
  641. * if (!item) return null;
  642. *
  643. * let color;
  644. * let menuIcon;
  645. * if (item.icon.type === "buddyboss") {
  646. * if (bbIconGlyphMap[item.icon.id]) {
  647. * menuIcon = {
  648. * fontIconName: item.icon.id,
  649. * fontIconVariant: getMenuFontIconVariant(item.icon, focused)
  650. * };
  651. * } else {
  652. * menuIcon = {
  653. * uri: focused ? item.icon.uri_active || item.icon.uri : item.icon.uri
  654. * };
  655. * }
  656. * color = focused ? tabBarActiveTintColor : item.icon.color;
  657. * } else {
  658. * menuIcon = {
  659. * uri: focused ? item.icon.uri_active || item.icon.uri : item.icon.uri
  660. * };
  661. * if (item.icon.type !== "custom") {
  662. * color = focused ? tabBarActiveTintColor : item.icon.color;
  663. * } else if (item.icon.fill_color) {
  664. * color = focused ? tabBarActiveTintColor : item.icon.color;
  665. * }
  666. * }
  667. *
  668. * if (item.object === "notifications" || item.object === "messages") {
  669. * return (
  670. * <BadgeIcon
  671. * calcFontSize={calcFontSize}
  672. * tintColor={color}
  673. * bottomTabsBg={colors.bottomTabsBg}
  674. * warningColor={colors.warningColor}
  675. * foregroundColor={item.icon.fg_color}
  676. * platform="ios"
  677. * inMore={false}
  678. * app={"learnerapp"}
  679. * type={item.object}
  680. * icon={menuIcon}
  681. * styles={{height: 25, width: 25}}
  682. * />
  683. * );
  684. * }
  685. *
  686. * return (
  687. * <Icon
  688. * icon={menuIcon}
  689. * foregroundColor={item.icon.fg_color}
  690. * tintColor={color}
  691. * styles={{height: 25, width: 25}}
  692. * />
  693. * );
  694. * };
  695. *
  696. * const MyTabBar = withGlobalStyles(props => {
  697. * const {state, descriptors, navigation, global, colors} = props;
  698. *
  699. * const insets = useSafeAreaInsets();
  700. * const bottomSafeArea = correctBottomSafeArea(insets.bottom);
  701. *
  702. * const textStyle = [
  703. * global.menuLabelStyle,
  704. * {
  705. * marginHorizontal: 5,
  706. * marginTop: isTabletOrIPad() ? -(bottomSafeArea / 5) : 2,
  707. * fontWeight: FontWeights["medium"],
  708. * ...textRTLStyleFix(),
  709. * opacity: 1,
  710. * textAlign: "center"
  711. * }
  712. * ];
  713. *
  714. * return (
  715. * <View style={{flexDirection: "row"}}>
  716. * {state.routes.map((route, index) => {
  717. * const {options} = descriptors[route.key];
  718. *
  719. * const item = route.params.item;
  720. *
  721. * const isFocused = state.index === index;
  722. *
  723. * const onPress = () => {
  724. * const event = navigation.emit({
  725. * type: "tabPress",
  726. * target: route.key,
  727. * canPreventDefault: true
  728. * });
  729. *
  730. * if (!isFocused && !event.defaultPrevented) {
  731. * navigation.navigate({name: route.name, merge: true});
  732. * }
  733. * };
  734. *
  735. * return (
  736. * <TouchableOpacity
  737. * onPress={onPress}
  738. * style={{
  739. * flex: 1,
  740. * height: 40,
  741. * alignItems: "center",
  742. * alignSelf: "center",
  743. * marginBottom: 33
  744. * }}
  745. * activeOpacity={1}
  746. * >
  747. * <View
  748. * style={{
  749. * height: 30,
  750. * alignItems: "center",
  751. * justifyContent: "center"
  752. * }}
  753. * >
  754. * {renderIcon({
  755. * focused: isFocused,
  756. * item,
  757. * colors,
  758. * tabBarOptions: options
  759. * })}
  760. * </View>
  761. *
  762. * <Text
  763. * style={[
  764. * textStyle,
  765. * {color: isFocused ? colors.linkColor : colors.textColor}
  766. * ]}
  767. * numberOfLines={1}
  768. * ellipsizeMode="tail"
  769. * allowFontScaling={false}
  770. * >
  771. * {item.label}
  772. * </Text>
  773. * </TouchableOpacity>
  774. * );
  775. * })}
  776. * </View>
  777. * );
  778. * });
  779. *
  780. * externalCodeSetup.navigationApi.setBottomTabBar(props => (
  781. * <MyTabBar {...props} />
  782. * ));
  783. * }
  784. */
  785. setBottomTabBar = bottomTabBar => {
  786. this.bottomTabBar = bottomTabBar;
  787. };
  788. animatedSwitchNavigator = (routes, options) => ({
  789. routes: routes,
  790. options: options
  791. });
  792. /**
  793. * This hook allows you to create your own animated switch navigator.
  794. * For example, you can add a new route and use it as your initial switch route instead of the default "Auth" route which will show you the login screen.
  795. * @method
  796. * @param {animatedSwitchNavigatorProps} animatedSwitchNavigator
  797. * @returns {animatedSwitchNavigatorReturn}
  798. *
  799. * @example <caption> Add a custom screen to be used as initial switch route </caption>
  800. *
  801. * //In custom_code/MyCustomScreen.js...
  802. *
  803. * import React from 'react';
  804. * import { View, Text, Button } from 'react-native';
  805. *
  806. * const MyCustomScreen = props => {
  807. *
  808. * return <View style={{ flex: 1, justifyContent: "center", alignSelf: "center" }}>
  809. * <Text> Welcome to our App! New adventure awaits!</Text>
  810. * <Button title="Go to App" onPress={() => props.navigation.navigate("Auth")} />
  811. * </View>
  812. * }
  813. *
  814. * export default MyCustomScreen;
  815. *
  816. *
  817. * //In custom_code/index.js...
  818. *
  819. * ...
  820. *
  821. * import MyCustomScreen from "./components/MyCustomScreen";
  822. * export const applyCustomCode = externalCodeSetup => {
  823. *
  824. * externalCodeSetup.navigationApi.setAnimatedSwitchNavigator((routes, options, routeProps) => {
  825. *
  826. * const feature = routeProps.settings.features.multisite_network;
  827. * const hasMultiSite = Platform.select({
  828. * ios: feature.is_enabled_ios,
  829. * android: feature.is_enabled_android
  830. * })
  831. *
  832. * // Get the initial switch route based on state data.
  833. * // getInitialSwitchRoute() is based on BB App's AppNavigator.js
  834. * // Feel free to copy and paste this to your own code
  835. *
  836. * const getInitialSwitchRoute = () => {
  837. *
  838. * const defaultInitialRoute = "Auth";
  839. *
  840. * const myCustomRoute = "MyCustomScreen"
  841. *
  842. * if (!routeProps.hasValidSigning) {
  843. * return "InvalidSigningScreen";
  844. * }
  845. *
  846. * if (routeProps.shouldEnforceVersionControl) {
  847. * return "VersionControlScreen";
  848. * } else if (routeProps.isLoggedIn) {
  849. * if (
  850. * routeProps.isFeatureEnabled(hasMultiSite) &&
  851. * routeProps.sites.selectedSite === null
  852. * ) {
  853. * return "AuthSiteSelectionScreen";
  854. * } else {
  855. * return routeProps.shouldLockApp ? "AppLockScreen" : "noAuth";
  856. * }
  857. * }
  858. * else {
  859. * return myCustomRoute; //Use my own custom route instead of the default "Auth" route
  860. * // return defaultInitialRoute;
  861. * }
  862. *
  863. * };
  864. *
  865. * const newRoutes = {
  866. * ...routes,
  867. * MyCustomScreen: {
  868. * screen: MyCustomScreen
  869. * }
  870. * }
  871. *
  872. * const newOptions = {
  873. * ...options,
  874. * initialRouteName: getInitialSwitchRoute()
  875. * }
  876. *
  877. * return {
  878. * routes: newRoutes,
  879. * options: newOptions,
  880. * }
  881. *
  882. * })
  883. * }
  884. */
  885. setAnimatedSwitchNavigator = animatedSwitchNavigator => {
  886. this.animatedSwitchNavigator = animatedSwitchNavigator;
  887. };
  888. screensWithoutTabBar = [];
  889. /**
  890. * This hook allows you to customise which screens will not have a tab bar even when the "Show on All Screens" option is selected.
  891. * Therefore you can use this hook to specifically select the pages where you don’t want to display tab bars.
  892. * @method
  893. * @param {Array} screensWithoutTabBar Screen names: {@link https://www.buddyboss.com/resources/app-codex/global.html#MainStackScreenNames}
  894. * @example
  895. *
  896. * externalCodeSetup.navigationApi.setScreensWithoutTabBar(["BlogSingleScreen", "CoursesSingleScreen"])
  897. */
  898. setScreensWithoutTabBar = screensWithoutTabBar => {
  899. this.screensWithoutTabBar = screensWithoutTabBar;
  900. };
  901. }