Source

externalCode/courseSingle.ts

  1. import * as React from "react";
  2. import {TCourseViewModel, TTranslationFunction, TQuizViewModel} from "./types";
  3. import {NavigationService} from "./types";
  4. import {NavigationProp} from "@react-navigation/native";
  5. export const API_NAME = "courseSingleApi";
  6. /**
  7. * CoursePropsFilterCallback
  8. */
  9. type CoursePropsFilterCallback = {
  10. /**
  11. * Course props
  12. */
  13. oldProps: Record<any, any>
  14. };
  15. /**
  16. * CourseStatusHiddenCallback
  17. */
  18. type CourseStatusHiddenCallback = {
  19. course: TCourseViewModel,
  20. /**
  21. * Returns `true` if course progress has started
  22. */
  23. hasStarted: boolean,
  24. /**
  25. * Course structure
  26. */
  27. courseTree: Record<any, any>
  28. };
  29. /**
  30. * CourseDescriptionHiddenCallback
  31. */
  32. type CourseDescriptionHiddenCallback = {
  33. course: TCourseViewModel
  34. };
  35. /**
  36. * CategoryTagsHiddenCallback
  37. */
  38. type CategoryTagsHiddenCallback = {
  39. course: TCourseViewModel
  40. };
  41. /**
  42. * TransformCourseActionButtonsCallback
  43. */
  44. type TransformCourseActionButtonsCallback = {
  45. CourseActionBtn: JSX.Element,
  46. course: TCourseViewModel,
  47. t: Function,
  48. /**
  49. * App global style
  50. */
  51. global: Record<any, any>,
  52. /**
  53. * App colors
  54. */
  55. colors: Record<any, any>,
  56. /**
  57. * Products associated to the course
  58. */
  59. products: Record<any, any>[],
  60. navigation: NavigationService,
  61. /**
  62. * Starts the course when called
  63. */
  64. startCourse: Function,
  65. /**
  66. * Continues the course when called for a course in progress
  67. */
  68. continueCourse: Function,
  69. /**
  70. * Renders the price component when called
  71. */
  72. priceComponentRender: Function
  73. };
  74. /**
  75. * CourseMaterialsProps
  76. */
  77. type CourseMaterialsProps = {
  78. /**
  79. * App global style
  80. */
  81. global: Record<any, any>,
  82. /**
  83. * App colors
  84. */
  85. colors: Record<any, any>,
  86. t: TTranslationFunction,
  87. /**
  88. * Contents of course materials field
  89. */
  90. materials: string,
  91. /**
  92. * Default styles for different tags
  93. */
  94. tagsStyles: Record<any, any>,
  95. /**
  96. * Default styles for course materials
  97. */
  98. materialsStyles: Record<any, any>,
  99. /**
  100. * Function to set a font style
  101. */
  102. baseFontStyle: Function,
  103. navigation: NavigationProp<any, any>,
  104. /**
  105. * Helper function that handles link press for the HTML component
  106. */
  107. onLinkPress: Function
  108. };
  109. /**
  110. * HeaderAuthorComponentProps
  111. */
  112. type HeaderAuthorComponentProps = {
  113. /**
  114. * Author details
  115. */
  116. user: Record<any, any>,
  117. /**
  118. * Returns `true` if course is using cover image
  119. */
  120. lightStyle: boolean,
  121. /**
  122. * App global style
  123. */
  124. global: Record<any, any>,
  125. /**
  126. * User id of course author
  127. */
  128. userId: number,
  129. onUserProfilePress: Function,
  130. loadingInProgress: Function,
  131. t: TTranslationFunction
  132. };
  133. /**
  134. * CourseTitleProps
  135. */
  136. type CourseTitleProps = {
  137. course: TCourseViewModel,
  138. /**
  139. * App global style
  140. */
  141. _globalStyles: Record<any, any>,
  142. /**
  143. * App colors
  144. */
  145. colors: Record<any, any>,
  146. /**
  147. * Returns `true` if course is using cover image
  148. */
  149. lightStyle: boolean,
  150. /**
  151. * Default title styles
  152. */
  153. styles: Record<any, any>,
  154. /**
  155. * Returns an object generated by React Native Animated
  156. */
  157. animatedOpacity: Record<any, any>
  158. };
  159. /**
  160. * CourseHeaderDetailsProps
  161. */
  162. type CourseHeaderDetailsProps = {
  163. courseVM: TCourseViewModel,
  164. /**
  165. * App global style
  166. */
  167. global: Record<any, any>,
  168. /**
  169. * App colors
  170. */
  171. colors: Record<any, any>,
  172. labels: Record<any, any>,
  173. t: TTranslationFunction,
  174. /**
  175. * Lessons count
  176. */
  177. lCount: number,
  178. /**
  179. * Topics count
  180. */
  181. tCount: number,
  182. /**
  183. * Quizzes count
  184. */
  185. qCount: number,
  186. /**
  187. * Can be used to determine if participants should be shown in the component
  188. */
  189. learndashCourseParticipants: number,
  190. navigation: NavigationProp<any, any>,
  191. /**
  192. * Returns `true` if participants should be shown
  193. */
  194. shouldShowParticipants: Boolean,
  195. /**
  196. * Returns `true` if course image should be shown
  197. */
  198. shouldShowImageBackground: Boolean,
  199. /**
  200. * Returns `true` if course video should be shown
  201. */
  202. shouldShowVideo: Boolean,
  203. /**
  204. * Returns `true` if "Course Includes" label should be shown
  205. */
  206. shouldShowCourseIncludesTitle: Boolean,
  207. /**
  208. * Returns `true` if lessons label should be shown
  209. */
  210. shouldShowLessonsText: Boolean,
  211. /**
  212. * Returns `true` if topics label should be shown
  213. */
  214. shouldShowTopicsText: Boolean,
  215. /**
  216. * Returns `true` if quizzes label should be shown
  217. */
  218. shouldShowQuizzesText: Boolean,
  219. /**
  220. * Returns `true` if course certificate label should be shown
  221. */
  222. shouldShowCertificatesText: Boolean,
  223. /**
  224. * Returns the lessons label
  225. */
  226. lessonsText: String,
  227. /**
  228. * Returns the topics label
  229. */
  230. topicsText: String,
  231. /**
  232. * Returns the quizzes label
  233. */
  234. quizzesText: String,
  235. /**
  236. * Returns the enrolled label
  237. */
  238. enrolledText: String,
  239. /**
  240. * Default styles
  241. */
  242. styles: Record<any, any>
  243. };
  244. /**
  245. * DownloadProgressComponentProps
  246. */
  247. type DownloadProgressComponentProps = {
  248. /**
  249. * Cancels the download progress
  250. */
  251. onPress: Function,
  252. /**
  253. * App colors
  254. */
  255. colors: Record<any, any>,
  256. /**
  257. * Current progress of download
  258. */
  259. progress: number,
  260. /**
  261. * Size of progress circle
  262. */
  263. size: number,
  264. /**
  265. * Returns `true` if component should use lightStyle
  266. */
  267. lightStyle: boolean,
  268. /**
  269. * Default hitSlop for component
  270. */
  271. hitSlop: Record<string, number>,
  272. /**
  273. * Default container style
  274. */
  275. containerStyle: Record<any, any>
  276. };
  277. /**
  278. * CourseHeaderRightProps
  279. */
  280. type CourseHeaderRightProps = {
  281. /**
  282. * App global style
  283. */
  284. global: Record<any, any>,
  285. /**
  286. * App colors
  287. */
  288. colors: Record<any, any>,
  289. t: TTranslationFunction,
  290. courseVM: TCourseViewModel,
  291. /**
  292. * Returns `true` if course should be open for guest user
  293. */
  294. isShowingOpenCourseForGuestUser: boolean,
  295. /**
  296. * Can be used to indicate what DownloadStatusIndicator component style should use
  297. */
  298. contentStyle: Record<any, any>
  299. };
  300. /*
  301. *
  302. * DownloadIconProps
  303. */
  304. type DownloadIconProps = {
  305. /**
  306. * Returns the status of the course being downloaded. Status includes `Complete`, `Pending`, `Failed`, etc.
  307. */
  308. status: string,
  309. /**
  310. * Starts the download process
  311. */
  312. start: Function,
  313. /**
  314. * Cancels the download process
  315. */
  316. cancel: Function,
  317. /**
  318. * Opens a modal that contains actions available for the downloaded
  319. */
  320. openOptions: Function,
  321. /**
  322. * Icon size
  323. */
  324. size: number,
  325. /**
  326. * Returns `true` if component should use lightStyle
  327. */
  328. lightStyle: boolean,
  329. /**
  330. * App colors
  331. */
  332. colors: Record<any, any>,
  333. /**
  334. * Default color of icon
  335. */
  336. idleIconColor: string,
  337. /**
  338. * Default hitSlop for component
  339. */
  340. hitSlop: Record<string, number>
  341. };
  342. /**
  343. * CourseStartedHeaderDetailsProps
  344. */
  345. type CourseStartedHeaderDetailsProps = {
  346. /**
  347. * App global style
  348. */
  349. global: Record<any, any>,
  350. /**
  351. * App colors
  352. */
  353. colors: Record<any, any>,
  354. t: TTranslationFunction,
  355. courseVM: TCourseViewModel,
  356. /**
  357. * Default styles for the header component
  358. */
  359. styles: Record<any, any>,
  360. /**
  361. * Helper function for parsing dates
  362. */
  363. formatDateFunc: Function
  364. };
  365. /**
  366. * CourseContentTitleProps
  367. */
  368. type CourseContentTitleProps = {
  369. /**
  370. * App global style
  371. */
  372. global: Record<any, any>,
  373. t: TTranslationFunction,
  374. labels: Record<any, any>,
  375. /**
  376. * Returns `true` if course content is collapsable
  377. */
  378. isCollapsable: boolean,
  379. /**
  380. * Returns `true` if collapsable items are already expanded
  381. */
  382. expanded: boolean,
  383. /**
  384. * Hides all the collapsable items
  385. */
  386. hideAll: Function,
  387. /**
  388. * Expands all the collapsable items
  389. */
  390. expand: Function
  391. };
  392. /**
  393. * CourseSectionItemProps
  394. */
  395. type CourseSectionItemProps = {
  396. /**
  397. * Data about item to be rendered
  398. */
  399. item: Record<any, any>,
  400. /**
  401. * Returns progression of the item
  402. */
  403. progression: number,
  404. /**
  405. * App global style
  406. */
  407. global: Record<any, any>,
  408. /**
  409. * App colors
  410. */
  411. colors: Record<any, any>,
  412. /**
  413. * Returns `true` if navigation to the item is allowed
  414. */
  415. allowNavigation: boolean,
  416. /**
  417. * Navigates to the lesson; topic or quiz
  418. */
  419. onItemPress: Function,
  420. /**
  421. * Returns number of topics and quizzes for the top-level item.
  422. * This can be used when determining if the lesson should be expandable or not.
  423. */
  424. count: number,
  425. /**
  426. * Result of using require() when fetching the appropriate icon for each item
  427. */
  428. icon: number,
  429. /**
  430. * Toggle or expand the item. This can be called when creating a custom IconComponent
  431. */
  432. toggleExpand: Function,
  433. /**
  434. * Returs the type of section item
  435. */
  436. blockType: string | undefined,
  437. /**
  438. * Calls an alert component which shows an error message
  439. */
  440. showAlertMessage: Function,
  441. /**
  442. * Default component for displaying the steps label in the item
  443. */
  444. StepsComponent: JSX.Element,
  445. /**
  446. * Default component for displaying the left-most icon in the item
  447. */
  448. IconComponent: JSX.Element
  449. };
  450. /**
  451. * QuizSectionTitleProps
  452. */
  453. type QuizSectionTitleProps = {
  454. /**
  455. * App global style
  456. */
  457. global: Record<any, any>,
  458. t: TTranslationFunction
  459. };
  460. /**
  461. * CourseDescriptionSectionTitleProps
  462. */
  463. type CourseDescriptionSectionTitleProps = {
  464. /**
  465. * Returns `true` if there are blocks to be rendered natively
  466. */
  467. shouldRenderBlocks: boolean,
  468. courseVM: TCourseViewModel,
  469. /**
  470. * App global style
  471. */
  472. global: Record<any, any>,
  473. t: TTranslationFunction,
  474. labels: Record<any, any>
  475. };
  476. /**
  477. * CourseMaterialsSectionTitleProps
  478. */
  479. type CourseMaterialsSectionTitleProps = {
  480. /**
  481. * App global style
  482. */
  483. global: Record<any, any>,
  484. t: TTranslationFunction,
  485. labels: Record<any, any>
  486. };
  487. /**
  488. * CourseDescriptionReadMoreComponentProps
  489. */
  490. type CourseDescriptionReadMoreComponentProps = {
  491. courseVM: TCourseViewModel,
  492. /**
  493. * Opens the desription modal
  494. */
  495. openDescriptionModal: Function,
  496. /**
  497. * App global style
  498. */
  499. global: Record<any, any>,
  500. t: TTranslationFunction
  501. };
  502. /**
  503. * WebViewOfflineComponentProps
  504. */
  505. type WebViewOfflineComponentProps = {
  506. /**
  507. * App global style
  508. */
  509. global: Record<any, any>,
  510. t: TTranslationFunction,
  511. /**
  512. * Default container style
  513. */
  514. containerStyle: Record<any, any>,
  515. /**
  516. * Default style of EmptyList component
  517. */
  518. emptyListStyle: Record<any, any>
  519. };
  520. /**
  521. * WebViewDescriptionComponentProps
  522. */
  523. type WebViewDescriptionComponentProps = {
  524. /**
  525. * Returns `true` if device is connected to an internet network
  526. */
  527. online: boolean,
  528. t: TTranslationFunction,
  529. /**
  530. * The default function used for determining how to handle webview requests.
  531. */
  532. onShouldStartLoadWithRequest: Function,
  533. /**
  534. * Default height limit
  535. */
  536. heightLimit: number,
  537. /**
  538. * Contains data of the web site to be loaded in the webview
  539. */
  540. source: Record<any, any>,
  541. /**
  542. * App global style
  543. */
  544. global: Record<any, any>,
  545. /**
  546. * App colors
  547. */
  548. colors: Record<any, any>,
  549. /**
  550. * Modal which shows up when the "Read More" component is pressed
  551. */
  552. ModalHeaderComponent: React.FC
  553. };
  554. /**
  555. * CourseQuizItemProps
  556. */
  557. type CourseQuizItemProps = {
  558. item: TQuizViewModel,
  559. style: Record<any, any>,
  560. /**
  561. * App global style
  562. */
  563. global: Record<any, any>,
  564. /**
  565. * App colors
  566. */
  567. colors: Record<any, any>,
  568. /**
  569. * Can be used to determine if actual progress should be shown or if item should be indicated as completed
  570. */
  571. skipProgress: boolean,
  572. /**
  573. * Navigates to the quiz
  574. */
  575. onPress: Function,
  576. /**
  577. * Returns `true` if navigation to the item is allowed
  578. */
  579. allowNavigation: boolean,
  580. /**
  581. * Default component for displaying the left-most icon in the item
  582. */
  583. IconComponent: JSX.Element
  584. };
  585. /**
  586. * LessonProgressStepsProps
  587. */
  588. type LessonProgressStepsProps = {
  589. t: TTranslationFunction,
  590. /**
  591. * Item details
  592. */
  593. item: Record<any, any>,
  594. /**
  595. * App global style
  596. */
  597. global: Record<any, any>,
  598. /**
  599. * App colors
  600. */
  601. colors: Record<any, any>,
  602. /**
  603. * Number of items inside the item
  604. */
  605. count: number,
  606. course: TCourseViewModel,
  607. /**
  608. * Describes the type of item being generated
  609. */
  610. blockType: string,
  611. /**
  612. * Number of completed steps inside the lesson
  613. */
  614. completedSteps: number
  615. };
  616. /**
  617. * @class
  618. * Single Course Screen Hooks.
  619. * Instance name: courseSingleApi
  620. Hooks to modify the individual courses’ pages.
  621. * @example
  622. * externalCodeSetup.courseSingleApi.METHOD_NAME
  623. */
  624. export class CourseSingleApi {
  625. /**
  626. * @deprecated
  627. */
  628. // show number of lessons on course single screen beside the lesson title
  629. setShowLessonsCount = () => (this.showLessonsCount = true);
  630. /**
  631. * @deprecated
  632. */
  633. // show number of quizzes on course single screen beside the quizzes title
  634. setShowQuizzesCount = () => (this.showQuizzesCount = true);
  635. /**
  636. * @deprecated
  637. * @private
  638. * @property {boolean} showLessonsCount
  639. */
  640. showLessonsCount = true;
  641. /**
  642. * @deprecated
  643. * @private
  644. * @property {boolean} showQuizzesCount
  645. */
  646. showQuizzesCount = true;
  647. headerAuthorRenderer: React.ComponentType<
  648. HeaderAuthorComponentProps
  649. > | null = null;
  650. /**
  651. * Sets component for overriding the default course author component.
  652. * @method
  653. * @param {React.ComponentType<HeaderAuthorComponentProps>} renderer
  654. * @example <caption>Add more details besides the author name</caption>
  655. *
  656. * //In custom_code/components/CourseAuthor.js
  657. *
  658. * import React from 'react';
  659. * import { View, Text } from 'react-native';
  660. * import AppAvatar from "@src/components/AppAvatar";
  661. * import AppTouchableOpacity from "@src/components/AppTouchableOpacity";
  662. * import { getBestImageVariant } from "@src/utils/CCDataUtil";
  663. * import { withUserClickHandler } from "@src/components/hocs/withUserClickHandler"; //Use BuddyBoss HOC for easier navigation
  664. *
  665. * const CourseAuthor = (props) => {
  666. *
  667. * const { user, global, lightStyle, toUserBasedOnSettings } = props;
  668. *
  669. * //Able navigate to profile of `user` because of wrapping component with HOC
  670. * const boundCallback = React.useMemo(
  671. * () => toUserBasedOnSettings.bind(null, null, user),
  672. * [user]
  673. * );
  674. *
  675. * //Use BuddyBoss getBestImageVariant helper function to get best image size for the component
  676. * const userAvatarUrl = React.useMemo(
  677. * () =>
  678. * !!user
  679. * ? user.avatar_urls
  680. * ? getBestImageVariant(user.avatar_urls, 96)
  681. * : user?.avatar?.thumb
  682. * : "",
  683. * [user]
  684. * );
  685. *
  686. * return <View style={[global.row]}>
  687. * <AppTouchableOpacity onPress={boundCallback}>
  688. * <View>
  689. * {user && (
  690. * <AppAvatar
  691. * size={26}
  692. * name={user.name}
  693. * source={{
  694. * uri: userAvatarUrl
  695. * }}
  696. * />
  697. * )}
  698. * </View>
  699. * </AppTouchableOpacity>
  700. * <View style={{ marginLeft: 8 }}>
  701. * {user && (
  702. *
  703. * <>
  704. * <Text
  705. * style={[
  706. * global.itemAuthorName,
  707. * { marginBottom: 1 },
  708. * lightStyle ? { color: "white" } : {}
  709. * ]}
  710. * >
  711. * {user?.name}
  712. * </Text>
  713. *
  714. * <Text> {user?.member_rest.user_email} </Text>
  715. * </>
  716. *
  717. * )}
  718. * </View>
  719. * </View>
  720. * }
  721. *
  722. * export default withUserClickHandler(CourseAuthor);
  723. *
  724. * //In custom_code/index.js
  725. *
  726. * ...
  727. *
  728. * import CourseAuthor from "./components/CourseAuthor";
  729. * export const applyCustomCode = externalCodeSetup => {
  730. * externalCodeSetup.courseSingleApi.setHeaderAuthorRenderer((props) => <CourseAuthor {...props} />)
  731. * }
  732. */
  733. setHeaderAuthorRenderer = (
  734. renderer: React.ComponentType<HeaderAuthorComponentProps> | null
  735. ) => {
  736. this.headerAuthorRenderer = renderer;
  737. };
  738. filterIncomingCourseProps = (props: Record<any, any>) => props;
  739. /**
  740. * Sets the callback function `filterIncomingCourseProps` used for filtering props that are coming into Courses Single Screen component
  741. * @param {CoursePropsFilterCallback} filterIncomingCourseProps
  742. * @method
  743. * @example <caption>Add a date object to incoming course props</caption>
  744. * externalCodeSetup.courseSingleApi.setFilterIncomingCourseProps((props) => {
  745. * return {
  746. * ...props,
  747. * date: new Date()
  748. * }
  749. * });
  750. */
  751. setFilterIncomingCourseProps = (
  752. filterIncomingCourseProps: (
  753. filterIncomingCourseProps: Record<any, any>
  754. ) => Record<any, any>
  755. ) => {
  756. this.filterIncomingCourseProps = filterIncomingCourseProps;
  757. };
  758. isCategoryTagsHidden = (course: TCourseViewModel) => false;
  759. /**
  760. * You can use this to show or hide the category tags in the course single screen.
  761. * @param {CategoryTagsHiddenCallback} CategoryTagsHiddenCallback
  762. * @method
  763. * @example
  764. *
  765. * externalCodeSetup.courseSingleApi.setIsCategoryTagsHidden((course) => {
  766. * return false;
  767. * })
  768. */
  769. setIsCategoryTagsHidden = (
  770. isCategoryTagsHidden: (course: TCourseViewModel) => boolean
  771. ) => {
  772. this.isCategoryTagsHidden = isCategoryTagsHidden;
  773. };
  774. CourseMaterialsComponent: React.ComponentType<
  775. CourseMaterialsProps
  776. > | null = null;
  777. /**
  778. * Sets a component that overrides the content added in course materials field found in BuddyBoss site > Learndash LMS > Courses > [Selected Course] > Settings.
  779. * @method
  780. * @param {React.ComponentType<CourseMaterialsProps>} CourseMaterialsComponent
  781. * @example <caption>Use a different component depending on the logged-in user</caption>
  782. *
  783. * ...
  784. *
  785. * const DEVICE_WIDTH = Dimensions.get("window").width;
  786. * import { useSelector } from "react-redux";
  787. * import WebView from "react-native-webview";
  788. * const ent = require("ent");
  789. *
  790. * export const applyCustomCode = (externalCodeSetup) => {
  791. *
  792. * externalCodeSetup.courseSingleApi.setCourseMaterialsComponent(({
  793. * tagsStyles,
  794. * materialsStyles,
  795. * baseFontStyle,
  796. * materials,
  797. * onLinkPress,
  798. * global
  799. * }) => {
  800. *
  801. * const CourseMaterial = () => {
  802. * const user = useSelector((state) => state.user.userObject);
  803. *
  804. * return user.id === 1 ?
  805. * <>
  806. * <Text style={{ marginBottom: 20 }}> Welcome back {user.name}! We prepared the materials for you. </Text>
  807. * <WebView source={{ uri: 'https://buddyboss.com' }} style={{ width: "auto", height: 300 }} />
  808. * </>
  809. * :
  810. * <View style={global.courseRoundBox}>
  811. * {typeof materials === "string" && (
  812. * <View style={{ paddingHorizontal: 15 }}>
  813. * <HTML
  814. * tagsStyles={{ ...tagsStyles, ...materialsStyles }}
  815. * baseFontStyle={baseFontStyle(15)}
  816. * html={ent.decode(materials)}
  817. * imagesMaxWidth={DEVICE_WIDTH - 32}
  818. * onLinkPress={onLinkPress}
  819. * />
  820. * </View>
  821. * )}
  822. * </View>
  823. *
  824. * }
  825. *
  826. * return <CourseMaterial />
  827. *
  828. * })
  829. * }
  830. */
  831. setCourseMaterialsComponent = (
  832. CourseMaterialsComponent: React.ComponentType<CourseMaterialsProps> | null
  833. ) => {
  834. this.CourseMaterialsComponent = CourseMaterialsComponent;
  835. };
  836. transformCourseActionButtons: (
  837. CourseActionBtn: JSX.Element,
  838. course: TCourseViewModel,
  839. t: Function,
  840. colors: Record<any, any>,
  841. global: Record<any, any>,
  842. products: Record<any, any>[],
  843. navigation: NavigationService,
  844. startCourse: Function,
  845. continueCourse: Function,
  846. priceComponentRender: Function
  847. ) => JSX.Element = CourseActionBtn => CourseActionBtn;
  848. /**
  849. * You can transform the default course action buttons which are starting, buying or continuing a course by replacing it with your preferred action buttons.
  850. * @param {TransformCourseActionButtonsCallback} transformCourseActionButtons
  851. * @method
  852. * @example <caption> Add more components for course action </caption>
  853. *
  854. * import CourseActionButton from "@src/components/Course/CourseActionButton";
  855. * export const applyCustomCode = (externalCodeSetup) => {
  856. * externalCodeSetup.courseSingleApi.setTransformCourseActionButtons((
  857. * CourseActionBtn,
  858. * course,
  859. * t,
  860. * colors,
  861. * global,
  862. * products,
  863. * navigation,
  864. * startCourse,
  865. * continueCourse,
  866. * priceComponentRender) => {
  867. *
  868. * return <>
  869. * <View style={{ paddingHorizontal: 20, paddingVertical: 10 }}>
  870. * <Text>To continue the course, tap the button below++</Text>
  871. * </View>
  872. *
  873. * <View style={{
  874. * paddingHorizontal: 20,
  875. * paddingVertical: 16,
  876. * flexDirection: "row",
  877. * alignItems: "center",
  878. * justifyContent: "space-between"
  879. * }}>
  880. * {CourseActionBtn}
  881. * <CourseActionButton
  882. * title={"Go to Courses screen"}
  883. * onPress={() => navigation.navigate("CoursesScreen")}
  884. * style={{ backgroundColor: "cyan" }}
  885. * />
  886. * </View>
  887. * </>
  888. * })
  889. *
  890. * }
  891. */
  892. setTransformCourseActionButtons = (
  893. transformCourseActionButtons: (
  894. CourseActionBtn: JSX.Element,
  895. course: TCourseViewModel,
  896. t: Function,
  897. colors: Record<any, any>,
  898. global: Record<any, any>,
  899. products: Record<any, any>[],
  900. navigation: NavigationService,
  901. startCourse: Function,
  902. continueCourse: Function,
  903. priceComponentRender: Function
  904. ) => JSX.Element
  905. ) => {
  906. this.transformCourseActionButtons = transformCourseActionButtons;
  907. };
  908. isCourseStatusHidden = (
  909. course: TCourseViewModel,
  910. hasStarted: boolean,
  911. courseTree: Record<any, any>
  912. ) => false;
  913. /**
  914. * You can use this hook to show or hide the course status component.
  915. * @param {CourseStatusHiddenCallback} CourseStatusHiddenCallback
  916. * @method
  917. * @example <caption> Hide the course status component depending on the course title </caption>
  918. * externalCodeSetup.courseSingleApi.setIsCourseStatusHidden((course, hasStarted, courseTree) => {
  919. * if (course.title == "React Native"){
  920. * return true;
  921. * }
  922. * return false;
  923. * })
  924. */
  925. setIsCourseStatusHidden = (
  926. isCourseStatusHidden: (
  927. course: TCourseViewModel,
  928. hasStarted: boolean,
  929. courseTree: Record<any, any>
  930. ) => boolean
  931. ) => {
  932. this.isCourseStatusHidden = isCourseStatusHidden;
  933. };
  934. isCourseDescriptionHidden = (course: TCourseViewModel) => false;
  935. /**
  936. * You can use this hook to show or hide the course description component.
  937. * @param {CourseDescriptionHiddenCallback} CourseDescriptionHiddenCallback
  938. * @method
  939. * @example <caption> Hide the course description depending on the course title </caption>
  940. * externalCodeSetup.courseSingleApi.setIsCourseDescriptionHidden((course) => {
  941. * if (course.title == "React Native"){
  942. * return true;
  943. * }
  944. * return false;
  945. * })
  946. */
  947. setIsCourseDescriptionHidden = (
  948. isCourseDescriptionHidden: (course: TCourseViewModel) => boolean
  949. ) => {
  950. this.isCourseDescriptionHidden = isCourseDescriptionHidden;
  951. };
  952. CourseTitleComponent: React.ComponentType<CourseTitleProps> | null = null;
  953. /**
  954. * You can use this to replace the course title component.
  955. * For example, you can use this to change the title's font size.
  956. * @method
  957. * @param {React.ComponentType<CourseTitleProps>} CourseTitleComponent
  958. * @example
  959. *
  960. * import React from "react";
  961. * import Animated from "react-native-reanimated";
  962. * import {View} from "react-native";
  963. *
  964. * export const applyCustomCode = externalCodeSetup => {
  965. * externalCodeSetup.courseSingleApi.setCourseTitleComponent(props => {
  966. * const {
  967. * course,
  968. * _globalStyles,
  969. * colors,
  970. * styles,
  971. * lightStyle,
  972. * animatedOpacity
  973. * } = props;
  974. *
  975. * return (
  976. * <View>
  977. * <Animated.Text
  978. * style={[
  979. * _globalStyles.iosStyleScreenTitle,
  980. * styles.title,
  981. * lightStyle ? {color: "white"} : {color: colors.textColor},
  982. * animatedOpacity
  983. * ]}
  984. * >
  985. * {course.title}
  986. * </Animated.Text>
  987. * </View>
  988. * );
  989. * });
  990. * };
  991. */
  992. setCourseTitleComponent = (
  993. CourseTitleComponent: React.ComponentType<CourseTitleProps> | null
  994. ) => {
  995. this.CourseTitleComponent = CourseTitleComponent;
  996. };
  997. CourseHeaderDetails: React.ComponentType<
  998. CourseHeaderDetailsProps
  999. > | null = null;
  1000. /**
  1001. * You can use this hook to customize the Course Status component of the LearnDash course details if a course is not yet in progress.
  1002. * For example, you can use this to change the course's preview video, featured image or the "Course Includes" details.
  1003. * @method
  1004. * @param {React.ComponentType<CourseHeaderDetailsProps>} CourseHeaderDetails
  1005. * @example
  1006. *
  1007. * import Icon from "@src/components/Icon";
  1008. * import AppImageBackground from "@src/components/AppImageBackground";
  1009. * import {CourseVideo} from "@src/components/Course/CourseStatus";
  1010. * import AppAvatar from "@src/components/AppAvatar";
  1011. *
  1012. * export const applyCustomCode = (externalCodeSetup) => {
  1013. * externalCodeSetup.courseSingleApi.setCourseHeaderDetails(props => {
  1014. * const {
  1015. * courseVM,
  1016. * global,
  1017. * labels,
  1018. * colors,
  1019. * t,
  1020. * navigation,
  1021. * shouldShowParticipants,
  1022. * shouldShowImageBackground,
  1023. * shouldShowVideo,
  1024. * shouldShowCourseIncludesTitle,
  1025. * shouldShowLessonsText,
  1026. * shouldShowTopicsText,
  1027. * shouldShowQuizzesText,
  1028. * shouldShowCertificatesText,
  1029. * lessonsText,
  1030. * topicsText,
  1031. * quizzesText,
  1032. * enrolledText,
  1033. * styles
  1034. * } = props;
  1035. *
  1036. * const size = 26;
  1037. *
  1038. * return (
  1039. * <>
  1040. * <View style={styles.mediaContainer}>
  1041. * {shouldShowImageBackground && (
  1042. * <AppImageBackground
  1043. * source={{uri: courseVM.featuredUrl}}
  1044. * style={{width: "100%", height: 200}}
  1045. * />
  1046. * )}
  1047. * {shouldShowVideo && (
  1048. * <CourseVideo
  1049. * url={courseVM.videoUrl}
  1050. * feature={courseVM.featuredUrl}
  1051. * global={global}
  1052. * colors={colors}
  1053. * navigation={navigation}
  1054. * />
  1055. * )}
  1056. * </View>
  1057. *
  1058. * <View style={styles.container}>
  1059. * {shouldShowParticipants && (
  1060. * <View style={{...global.row}}>
  1061. * {courseVM.members
  1062. * .map(x => x.member_rest?.avatar?.thumb || "")
  1063. * .map((url, index) => (
  1064. * <AppAvatar
  1065. * key={url}
  1066. * style={[
  1067. * styles.avatar,
  1068. * {
  1069. * left: index === 0 ? 0 : -10 * index,
  1070. * zIndex: index
  1071. * }
  1072. * ]}
  1073. * borderRadius={size / 2}
  1074. * size={size}
  1075. * source={{
  1076. * uri: url
  1077. * }}
  1078. * />
  1079. * ))}
  1080. * <Text style={styles.enrolledText}>{enrolledText}</Text>
  1081. * </View>
  1082. * )}
  1083. * {shouldShowCourseIncludesTitle && (
  1084. * <Text style={styles.courseIncludesTitle}>
  1085. * {t("course:includesTitle", {label: labels.course})}
  1086. * </Text>
  1087. * )}
  1088. * {shouldShowLessonsText && (
  1089. * <View style={{flexDirection: "row", marginBottom: 5}}>
  1090. * <Icon
  1091. * size={30}
  1092. * webIcon={"IconAndroidGroup"}
  1093. * tintColor={colors.descLightTextColor}
  1094. * icon={{fontIconName: "book", weight: 400}}
  1095. * />
  1096. * <Text style={styles.courseIncludesText}>{lessonsText}</Text>
  1097. * </View>
  1098. * )}
  1099. * {shouldShowTopicsText && (
  1100. * <View style={{flexDirection: "row", marginBottom: 5, marginLeft: 5}}>
  1101. * <Icon
  1102. * size={24}
  1103. * webIcon={"IconAndroidGroup"}
  1104. * tintColor={colors.descLightTextColor}
  1105. * icon={{fontIconName: "text", weight: 500}}
  1106. * />
  1107. * <Text style={styles.courseIncludesText}>{topicsText}</Text>
  1108. * </View>
  1109. * )}
  1110. * {shouldShowQuizzesText && (
  1111. * <View style={{flexDirection: "row", marginLeft: 5, marginBottom: 5}}>
  1112. * <Icon
  1113. * size={24}
  1114. * webIcon={"IconAndroidGroup"}
  1115. * tintColor={colors.descLightTextColor}
  1116. * icon={{fontIconName: "question", weight: 100}}
  1117. * />
  1118. * <Text style={styles.courseIncludesText}>{quizzesText}</Text>
  1119. * </View>
  1120. * )}
  1121. * {shouldShowCertificatesText && (
  1122. * <View style={{flexDirection: "row", marginLeft: 5}}>
  1123. * <Icon
  1124. * size={26}
  1125. * webIcon={"IconAndroidGroup"}
  1126. * tintColor={colors.descLightTextColor}
  1127. * icon={{fontIconName: "certificate", weight: 400}}
  1128. * />
  1129. * <Text style={styles.courseIncludesText}>
  1130. * {t("course:certificate", {label: labels.course})} asdfasdf
  1131. * </Text>
  1132. * </View>
  1133. * )}
  1134. * </View>
  1135. * </>
  1136. * );
  1137. * });
  1138. * }
  1139. *
  1140. */
  1141. setCourseHeaderDetails = (
  1142. CourseHeaderDetails: React.ComponentType<CourseHeaderDetailsProps> | null
  1143. ) => {
  1144. this.CourseHeaderDetails = CourseHeaderDetails;
  1145. };
  1146. HeaderRightComponent: React.ComponentType<
  1147. CourseHeaderRightProps
  1148. > | null = null;
  1149. /**
  1150. * You can use this hook to customize the component on the top right corner of the CourseSingleScreen which the download icon usually occupies.
  1151. * @method
  1152. * @param {React.ComponentType<CourseHeaderRightProps>} HeaderRightComponent
  1153. * @example
  1154. * ...
  1155. *
  1156. * import AuthWrapper from "@src/components/AuthWrapper";
  1157. * import DownloadStatusIndicator from "@src/components/Downloader/DownloadStatusIndicator";
  1158. * import StatusBarWrapper from "@src/utils/StatusBarWrapper";
  1159. * import { DownloadTypes } from "@src/services/enums/downloads";
  1160. *
  1161. * export const applyCustomCode = (externalCodeSetup: any) => {
  1162. *
  1163. * externalCodeSetup.courseSingleApi.setHeaderRightComponent(({
  1164. * global,
  1165. * colors,
  1166. * t,
  1167. * courseVM,
  1168. * isShowingOpenCourseForGuestUser,
  1169. * contentStyle
  1170. * }) => (
  1171. * courseVM.hasAccess &&
  1172. * !courseVM.offlineDisabled && (
  1173. * <AuthWrapper
  1174. * actionOnGuestLogin={
  1175. * isShowingOpenCourseForGuestUser
  1176. * ? null
  1177. * : "showAuthScreen"
  1178. * }
  1179. * >
  1180. * <DownloadStatusIndicator
  1181. * lightStyle={
  1182. * contentStyle === StatusBarWrapper.lightStyle
  1183. * }
  1184. * postId={courseVM.id}
  1185. * postType={DownloadTypes.course}
  1186. * colors={colors}
  1187. * global={global}
  1188. * size={30}
  1189. * t={t}
  1190. * iconColor={colors.linkColor}
  1191. * containerStyle={{
  1192. * marginLeft: 15
  1193. * }}
  1194. * />
  1195. * </AuthWrapper>
  1196. * )
  1197. *
  1198. * ))
  1199. * }
  1200. */
  1201. setHeaderRightComponent = (
  1202. HeaderRightComponent: React.ComponentType<CourseHeaderRightProps> | null
  1203. ) => {
  1204. this.HeaderRightComponent = HeaderRightComponent;
  1205. };
  1206. DownloadProgressComponent: React.ComponentType<
  1207. DownloadProgressComponentProps
  1208. > | null = null;
  1209. /**
  1210. * You can use this hook to customize the DownloadProgressComponent which shows up when the course is currently downloading.
  1211. * @param {React.ComponentType<DownloadProgressComponentProps>} DownloadProgressComponent
  1212. * @method
  1213. * @example
  1214. *
  1215. * ...
  1216. *
  1217. * import ProgressCircle from "react-native-progress/Circle";
  1218. * import {TouchableWithoutFeedback} from "react-native";
  1219. * import Icon from "@src/components/Icon";
  1220. * export const applyCustomCode = (externalCodeSetup) => {
  1221. *
  1222. * externalCodeSetup.courseSingleApi.setDownloadProgressComponent(({
  1223. * size,
  1224. * lightStyle,
  1225. * colors,
  1226. * onPress,
  1227. * hitSlop,
  1228. * containerStyle,
  1229. * progress
  1230. * }) => {
  1231. * const iconSize = size - 10;
  1232. * const color = lightStyle ? "#fff" : colors.highlightColor;
  1233. * const unfilledColor = lightStyle
  1234. * ? "rgba(255, 255, 255, 0.24)"
  1235. * : "rgba(0, 0, 0, 0.24)";
  1236. * return (
  1237. * <TouchableWithoutFeedback onPress={onPress} hitSlop={hitSlop}>
  1238. * <View style={containerStyle}>
  1239. * <ProgressCircle
  1240. * size={size}
  1241. * progress={progress}
  1242. * thickness={2}
  1243. * unfilledColor={unfilledColor}
  1244. * animated={true}
  1245. * borderWidth={0}
  1246. * color={color}
  1247. * />
  1248. * <Icon
  1249. * icon={{fontIconName: "stop", weight: 300}}
  1250. * tintColor={color}
  1251. * styles={{height: iconSize, width: iconSize, position: "absolute"}}
  1252. * />
  1253. * </View>
  1254. * </TouchableWithoutFeedback>
  1255. * );
  1256. * })
  1257. * }
  1258. */
  1259. setDownloadProgressComponent = (
  1260. DownloadProgressComponent: React.ComponentType<
  1261. DownloadProgressComponentProps
  1262. > | null
  1263. ) => {
  1264. this.DownloadProgressComponent = DownloadProgressComponent;
  1265. };
  1266. DownloadIcon: React.ComponentType<DownloadIconProps> | null = null;
  1267. /**
  1268. * You can use this hook to customize the DownloadIcon which can be used to download a course, delete a course, or indicate if a course has already been downloaded.
  1269. *
  1270. * @param {React.ComponentType<DownloadIconProps>} DownloadIcon
  1271. * @method
  1272. * @example
  1273. *
  1274. * import React from "react";
  1275. * import Animated from "react-native-reanimated";
  1276. * import AppTouchableWithoutFeedback from "@src/components/AppTouchableWithoutFeedback";
  1277. * import Icon from "@src/components/Icon";
  1278. * export const applyCustomCode = externalCodeSetup => {
  1279. * externalCodeSetup.courseSingleApi.setDownloadIcon(
  1280. * ({colors, idleIconColor, lightStyle, start, size, hitSlop}) => {
  1281. * const iconColor =
  1282. * idleIconColor ?? (lightStyle ? "#fff" : colors.linkColor);
  1283. * const action = start;
  1284. * return (
  1285. * <AppTouchableWithoutFeedback
  1286. * onPress={action}
  1287. * debounce={500}
  1288. * hitSlop={hitSlop}
  1289. * >
  1290. *
  1291. * <Animated.View>
  1292. * <Icon
  1293. * icon={{fontIconName: "cloud-download", weight: 400}}
  1294. * styles={{height: size, width: size}}
  1295. * tintColor={iconColor}
  1296. * />
  1297. * </Animated.View>
  1298. * </AppTouchableWithoutFeedback>
  1299. * );
  1300. * }
  1301. * );
  1302. * };
  1303. *
  1304. */
  1305. setDownloadIcon = (
  1306. DownloadIcon: React.ComponentType<DownloadIconProps> | null
  1307. ) => {
  1308. this.DownloadIcon = DownloadIcon;
  1309. };
  1310. CourseStartedHeaderDetails: React.ComponentType<
  1311. CourseStartedHeaderDetailsProps
  1312. > | null = null;
  1313. /**
  1314. * You can use this hook to customize the Course Status component of the LearnDash course details if a course is in progress or is already complete.
  1315. * @param {React.ComponentType<CourseStartedHeaderDetailsProps>} CourseStartedHeaderDetails
  1316. * @method
  1317. * @example
  1318. *
  1319. * ...
  1320. *
  1321. * import Icon from "@src/components/Icon";
  1322. * import Progress from "@src/components/Progress";
  1323. * export const applyCustomCode = (externalCodeSetup) => {
  1324. *
  1325. * externalCodeSetup.courseSingleApi.setCourseStartedHeaderDetails(({
  1326. * global,
  1327. * colors,
  1328. * t,
  1329. * courseVM,
  1330. * styles,
  1331. * formatDateFunc
  1332. * }) => <View
  1333. * style={[
  1334. * global.row,
  1335. * {
  1336. * paddingHorizontal: 20,
  1337. * paddingVertical: 16,
  1338. * flex: 1
  1339. * }
  1340. * ]}
  1341. * >
  1342. * <View style={{flex: 1, marginRight: 8}}>
  1343. * {courseVM.hasAccess ? (
  1344. * <Text style={[global.title, styles.title]}>
  1345. * {!!!courseVM.completed
  1346. * ? t("course:inProgress")
  1347. * : t("course:completed")}
  1348. * </Text>
  1349. * ) : (
  1350. * <Text style={[global.title, styles.title]}>
  1351. * {t("course:enrolled")}
  1352. * </Text>
  1353. * )}
  1354. * {courseVM.modifiedDate && (
  1355. * <Text style={[global.courseDate, {marginTop: 5}]}>
  1356. * {`${t("course:lastActivity")} ${formatDateFunc(
  1357. * courseVM.modifiedDate
  1358. * )}`}
  1359. * </Text>
  1360. * )}
  1361. * </View>
  1362. * <Text style={[global.progressLargeText, styles.progressText]}>
  1363. * {courseVM.progression + "%"}
  1364. * </Text>
  1365. * <Progress
  1366. * size={32}
  1367. * checkIcon={
  1368. * <Icon
  1369. * styles={{width: 36, height: 36}}
  1370. * icon={{fontIconName: "check", weight: 200}}
  1371. * tintColor={colors.successColor}
  1372. * />
  1373. * }
  1374. * isCompleted={!!courseVM.completed}
  1375. * progress={courseVM.progression}
  1376. * colors={{
  1377. * highlightColor: colors.highlightColor,
  1378. * bodyBg: "#e7e9ec"
  1379. * }}
  1380. * />
  1381. * </View>
  1382. * );
  1383. * }
  1384. *
  1385. *
  1386. */
  1387. setCourseStartedHeaderDetails = (
  1388. CourseStartedHeaderDetails: React.ComponentType<
  1389. CourseStartedHeaderDetailsProps
  1390. > | null
  1391. ) => {
  1392. this.CourseStartedHeaderDetails = CourseStartedHeaderDetails;
  1393. };
  1394. CourseContentTitle: React.ComponentType<
  1395. CourseContentTitleProps
  1396. > | null = null;
  1397. /**
  1398. * You can use this hook to customize the "Course Content" title and the "Expand All" function.
  1399. * @param {React.ComponentType<CourseContentTitleProps>} CourseContentTitle
  1400. * @method
  1401. * @example
  1402. *
  1403. * ...
  1404. *
  1405. * export const applyCustomCode = (externalCodeSetup) => {
  1406. * externalCodeSetup.courseSingleApi.setCourseContentTitle(({
  1407. * global,
  1408. * labels,
  1409. * isCollapsable,
  1410. * expanded,
  1411. * hideAll,
  1412. * expand,
  1413. * t
  1414. * }) => (
  1415. * <View
  1416. * style={[
  1417. * global.row,
  1418. * {flex: 1, justifyContent: "space-between", marginBottom: 9}
  1419. * ]}
  1420. * >
  1421. * <Text style={[global.courseRoundBoxTitleAbove, {marginBottom: 0}]}>
  1422. * {t("course:contentTitle", {label: labels.course})}
  1423. * </Text>
  1424. * {isCollapsable && (
  1425. * <TouchableWithoutFeedback
  1426. * hitSlop={{top: 20, right: 20, bottom: 20, left: 20}}
  1427. * onPress={() => {
  1428. * if (expanded) {
  1429. * hideAll();
  1430. * } else {
  1431. * expand();
  1432. * }
  1433. * }}
  1434. * >
  1435. * <Text style={[global.link, {fontSize: 17}]}>
  1436. * {expanded ? t("courses:collapseAll") : t("courses:expandAll")}
  1437. * </Text>
  1438. * </TouchableWithoutFeedback>
  1439. * )}
  1440. * </View>
  1441. * ));
  1442. * }
  1443. *
  1444. */
  1445. setCourseContentTitle = (
  1446. CourseContentTitle: React.ComponentType<CourseContentTitleProps> | null
  1447. ) => {
  1448. this.CourseContentTitle = CourseContentTitle;
  1449. };
  1450. CourseSectionItem: React.ComponentType<CourseSectionItemProps> | null = null;
  1451. /**
  1452. * You can use this hook to customize the Item components that are displayed as course content.
  1453. * @param {React.ComponentType<CourseSectionItemProps>} CourseSectionItem
  1454. * @method
  1455. * @example <caption> Use your own icon component instead of the default IconComponent of lessons with children</caption>
  1456. *
  1457. * ...
  1458. *
  1459. * import Icon from "@src/components/Icon";
  1460. * import LearnItem from "@src/components/Course/LearnItem";
  1461. * export const applyCustomCode = (externalCodeSetup) => {
  1462. *
  1463. * externalCodeSetup.courseSingleApi.setCourseSectionItem(props => {
  1464. * const {
  1465. * item,
  1466. * progression,
  1467. * global,
  1468. * colors,
  1469. * allowNavigation,
  1470. * onItemPress,
  1471. * StepsComponent,
  1472. * IconComponent,
  1473. * blockType,
  1474. * toggleExpand,
  1475. * count,
  1476. * showAlertMessage
  1477. * } = props;
  1478. *
  1479. * let CustomIcon = IconComponent;
  1480. * if (blockType === "lesson" && count > 0) {
  1481. * CustomIcon = (
  1482. * <TouchableOpacity onPress={toggleExpand}>
  1483. * <Icon
  1484. * tintColor={colors.descLightTextColor}
  1485. * icon={{fontIconName: "heart", weight: 400}}
  1486. * styles={{
  1487. * opacity: !allowNavigation ? 0.45 : 1,
  1488. * marginRight: 10,
  1489. * width: 25,
  1490. * height: 25
  1491. * }}
  1492. * />
  1493. * </TouchableOpacity>
  1494. * );
  1495. * }
  1496. *
  1497. * return (
  1498. * <LearnItem
  1499. * key={item.id}
  1500. * style={{
  1501. * paddingLeft: 0
  1502. * }}
  1503. * item={{
  1504. * id: item.id,
  1505. * completed: item.completed,
  1506. * title: item.title,
  1507. * progression
  1508. * }}
  1509. * global={global}
  1510. * colors={colors}
  1511. * onPress={() => {
  1512. * if (allowNavigation) {
  1513. * showAlertMessage();
  1514. * return false;
  1515. * }
  1516. * onItemPress();
  1517. * }}
  1518. * allowNavigation={allowNavigation}
  1519. * beforeProgress={StepsComponent}
  1520. * beforeLabel={CustomIcon}
  1521. * />
  1522. * );
  1523. * });
  1524. * }
  1525. */
  1526. setCourseSectionItem = (
  1527. CourseSectionItem: React.ComponentType<CourseSectionItemProps> | null
  1528. ) => {
  1529. this.CourseSectionItem = CourseSectionItem;
  1530. };
  1531. QuizSectionTitle: React.ComponentType<QuizSectionTitleProps> | null = null;
  1532. /**
  1533. * You can use this hook to customize the component that displays the "Final Quizzes" text.
  1534. * @method
  1535. * @param {React.ComponentType<QuizSectionTitleProps>} QuizSectionTitle
  1536. * @example
  1537. *
  1538. * ...
  1539. *
  1540. * externalCodeSetup.courseSingleApi.setQuizSectionTitle(({global, t}) => (
  1541. * <Text style={global.courseRoundBoxSectionTitle}>
  1542. * {t("course:finalQuizzes")}
  1543. * </Text>
  1544. * ));
  1545. */
  1546. setQuizSectionTitle = (
  1547. QuizSectionTitle: React.ComponentType<QuizSectionTitleProps> | null
  1548. ) => {
  1549. this.QuizSectionTitle = QuizSectionTitle;
  1550. };
  1551. CourseDescriptionSectionTitle: React.ComponentType<
  1552. CourseDescriptionSectionTitleProps
  1553. > | null = null;
  1554. /**
  1555. * You can use this hook to customize the component that displays the "Course Description" text.
  1556. * @method
  1557. * @param {React.ComponentType<CourseDescriptionSectionTitleProps>} CourseDescriptionSectionTitle
  1558. * @example
  1559. *
  1560. * ...
  1561. *
  1562. * export const applyCustomCode = (externalCodeSetup) => {
  1563. * externalCodeSetup.courseSingleApi.setCourseDescriptionSectionTitle(
  1564. * ({shouldRenderBlocks, courseVM, global, labels, t}) => {
  1565. * if (shouldRenderBlocks || courseVM.content) {
  1566. * return (
  1567. * <Text style={[global.courseRoundBoxTitleAbove]}>
  1568. * {t("course:descriptionTitle", {
  1569. * label: labels.course
  1570. * })}
  1571. * </Text>
  1572. * );
  1573. * }
  1574. *
  1575. * return null;
  1576. * }
  1577. * );
  1578. * }
  1579. */
  1580. setCourseDescriptionSectionTitle = (
  1581. CourseDescriptionSectionTitle: React.ComponentType<
  1582. CourseDescriptionSectionTitleProps
  1583. > | null
  1584. ) => {
  1585. this.CourseDescriptionSectionTitle = CourseDescriptionSectionTitle;
  1586. };
  1587. CourseMaterialsSectionTitle: React.ComponentType<
  1588. CourseMaterialsSectionTitleProps
  1589. > | null = null;
  1590. /**
  1591. * You can use this hook to customize the component that displays the "Course Materials" text.
  1592. * @method
  1593. * @param {React.ComponentType<CourseMaterialsSectionTitleProps>} CourseMaterialsSectionTitle
  1594. * @example
  1595. *
  1596. * ...
  1597. *
  1598. * export const applyCustomCode = (externalCodeSetup) => {
  1599. * externalCodeSetup.courseSingleApi.setCourseMaterialsSectionTitle(
  1600. * ({global, t, labels}) => (
  1601. * <Text style={[global.courseRoundBoxTitleAbove, {marginBottom: 20}]}>
  1602. * {t("course:courseMaterials", {label: labels.course})}
  1603. * </Text>
  1604. * )
  1605. * );
  1606. * }
  1607. */
  1608. setCourseMaterialsSectionTitle = (
  1609. CourseMaterialsSectionTitle: React.ComponentType<
  1610. CourseMaterialsSectionTitleProps
  1611. > | null
  1612. ) => {
  1613. this.CourseMaterialsSectionTitle = CourseMaterialsSectionTitle;
  1614. };
  1615. CourseDescriptionReadMoreComponent: React.ComponentType<
  1616. CourseDescriptionReadMoreComponentProps
  1617. > | null = null;
  1618. /**
  1619. * You can use this hook to customize the "Read More" text that shows the course description in a modal when pressed.
  1620. * @method
  1621. * @param {React.ComponentType<CourseDescriptionReadMoreComponentProps>} CourseDescriptionReadMoreComponent
  1622. * @example
  1623. *
  1624. * ...
  1625. * import AppTouchableWithoutFeedback from "@src/components/AppTouchableWithoutFeedback";
  1626. * export const applyCustomCode = (externalCodeSetup) => {
  1627. * externalCodeSetup.courseSingleApi.setCourseDescriptionReadMoreComponent(({
  1628. * courseVM,
  1629. * openDescriptionModal,
  1630. * global,
  1631. * t
  1632. * }) => {
  1633. * if (courseVM.contentNative.length > 1) {
  1634. * return <View style={{alignItems: "center"}}>
  1635. * <AppTouchableWithoutFeedback
  1636. * onPress={openDescriptionModal}
  1637. * >
  1638. * <Text
  1639. * style={[
  1640. * global.linkRegular,
  1641. * {
  1642. * paddingHorizontal: 20,
  1643. * paddingVertical: 10
  1644. * }
  1645. * ]}
  1646. * >
  1647. * {t("courses:readMore")}
  1648. * </Text>
  1649. * </AppTouchableWithoutFeedback>
  1650. * </View>
  1651. * }
  1652. * return null
  1653. * })
  1654. * }
  1655. */
  1656. setCourseDescriptionReadMoreComponent = (
  1657. CourseDescriptionReadMoreComponent: React.ComponentType<
  1658. CourseDescriptionReadMoreComponentProps
  1659. > | null
  1660. ) => {
  1661. this.CourseDescriptionReadMoreComponent = CourseDescriptionReadMoreComponent;
  1662. };
  1663. WebViewOfflineComponent: React.ComponentType<
  1664. WebViewOfflineComponentProps
  1665. > | null = null;
  1666. /**
  1667. * You can use this hook to customize the component that displays a "This content is not available offline" message if a webview is used for rendering the course description while the device is offline.
  1668. * @method
  1669. * @param {React.ComponentType<WebViewOfflineComponentProps>} WebViewOfflineComponent
  1670. * @example
  1671. *
  1672. * ...
  1673. *
  1674. * import EmptyList from "@src/components/EmptyList";
  1675. *
  1676. * export const applyCustomCode = (externalCodeSetup) => {
  1677. * externalCodeSetup.courseSingleApi.setWebViewOfflineComponent(
  1678. * ({containerStyle, t, global, emptyListStyle}) => (
  1679. * <View style={containerStyle}>
  1680. * <Text>Please connect to an internet network</Text>
  1681. * <EmptyList
  1682. * emptyText={{
  1683. * title: t("common:contentOfflineMessage"),
  1684. * icon: {fontIconName: "wifi-slash", weight: 400}
  1685. * }}
  1686. * global={global}
  1687. * style={emptyListStyle}
  1688. * />
  1689. * </View>
  1690. * )
  1691. * );
  1692. * }
  1693. */
  1694. setWebViewOfflineComponent = (
  1695. WebViewOfflineComponent: React.ComponentType<
  1696. WebViewOfflineComponentProps
  1697. > | null
  1698. ) => {
  1699. this.WebViewOfflineComponent = WebViewOfflineComponent;
  1700. };
  1701. WebViewDescriptionComponent: React.ComponentType<
  1702. WebViewDescriptionComponentProps
  1703. > | null = null;
  1704. /**
  1705. * You can use this hook to replace the webview being used in the course description.
  1706. * For example, you can choose to replace it with the default react-native webview.
  1707. * @method
  1708. * @param {React.ComponentType<WebViewDescriptionComponentProps>} WebViewDescriptionComponent
  1709. * @example
  1710. *
  1711. * ...
  1712. *
  1713. * import WebViewWithMore from "@src/components/WebViewWithMore";
  1714. * export const applyCustomCode = (externalCodeSetup) => {
  1715. * externalCodeSetup.courseSingleApi.setWebViewDescriptionComponent(
  1716. * ({
  1717. * online,
  1718. * t,
  1719. * onShouldStartLoadWithRequest,
  1720. * heightLimit,
  1721. * source,
  1722. * global,
  1723. * colors,
  1724. * ModalHeaderComponent
  1725. * }) => (
  1726. * <WebViewWithMore
  1727. * online={online}
  1728. * t={t}
  1729. * onShouldStartLoadWithRequest={onShouldStartLoadWithRequest}
  1730. * disableLoadingPadding={true}
  1731. * height={heightLimit}
  1732. * source={source}
  1733. * global={global}
  1734. * colors={colors}
  1735. * ModalHeaderComponent={ModalHeaderComponent}
  1736. * />
  1737. * )
  1738. * );
  1739. * }
  1740. */
  1741. setWebViewDescriptionComponent = (
  1742. WebViewDescriptionComponent: React.ComponentType<
  1743. WebViewDescriptionComponentProps
  1744. > | null
  1745. ) => {
  1746. this.WebViewDescriptionComponent = WebViewDescriptionComponent;
  1747. };
  1748. CourseQuizItem: React.ComponentType<CourseQuizItemProps> | null = null;
  1749. /**
  1750. * You can use this hook to customize the Item components that are displayed as final quizzes.
  1751. * @param {React.ComponentType<CourseQuizItemProps>} CourseQuizItem
  1752. * @method
  1753. * @example
  1754. *
  1755. * ...
  1756. *
  1757. * import LearnItem from "@src/components/Course/LearnItem";
  1758. *
  1759. * export const applyCustomCode = (externalCodeSetup) => {
  1760. * externalCodeSetup.courseSingleApi.setCourseQuizItem(
  1761. * ({
  1762. * item,
  1763. * style,
  1764. * global,
  1765. * colors,
  1766. * skipProgress,
  1767. * onPress,
  1768. * allowNavigation,
  1769. * IconComponent
  1770. * }) => (
  1771. * <LearnItem
  1772. * key={item.id}
  1773. * style={style}
  1774. * item={item}
  1775. * global={global}
  1776. * colors={colors}
  1777. * skipProgress={skipProgress}
  1778. * onPress={onPress}
  1779. * allowNavigation={allowNavigation}
  1780. * beforeLabel={IconComponent}
  1781. * />
  1782. * )
  1783. * );
  1784. * }
  1785. *
  1786. */
  1787. setCourseQuizItem = (
  1788. CourseQuizItem: React.ComponentType<CourseQuizItemProps> | null
  1789. ) => {
  1790. this.CourseQuizItem = CourseQuizItem;
  1791. };
  1792. LessonProgressStepsComponent: React.ComponentType<
  1793. LessonProgressStepsProps
  1794. > | null = null;
  1795. /**
  1796. * You can use this to replace the course lesson progress step component.
  1797. * @method
  1798. * @param {React.ComponentType<LessonProgressStepsProps>} LessonProgressStepsComponent
  1799. * @example <caption>Replace the course lesson progress steps component</caption>
  1800. *
  1801. * export const applyCustomCode = (externalCodeSetup) => {
  1802. * externalCodeSetup.courseSingleApi.setLessonProgressStepsComponent(props => {
  1803. * const {count, completedSteps, blockType, global, t} = props;
  1804. *
  1805. * return (
  1806. * !!count &&
  1807. * count > 0 &&
  1808. * blockType === "lesson" ? (
  1809. * <Text style={[global.caption]}>
  1810. * {`${t("courses:stepsOutOf", {completedSteps, count})}`}
  1811. * </Text>
  1812. * ) : <></>
  1813. * );
  1814. * });
  1815. * }
  1816. */
  1817. setLessonProgressStepsComponent = (
  1818. LessonProgressStepsComponent: React.ComponentType<
  1819. LessonProgressStepsProps
  1820. > | null
  1821. ) => {
  1822. this.LessonProgressStepsComponent = LessonProgressStepsComponent;
  1823. };
  1824. }