Source

externalCode/lessonSingleScreen.ts

  1. import * as React from "react";
  2. import {
  3. Course,
  4. TCourseViewModel,
  5. TLessonViewModel,
  6. TTranslationFunction
  7. } from "./types";
  8. import {NavigationProp} from "@react-navigation/native";
  9. export const API_NAME = "lessonSingleScreenApi";
  10. /**
  11. * PrevNextComponentProps
  12. */
  13. type PrevNextComponentProps = {
  14. /**
  15. * Function to execute if previous or next button is a quiz
  16. */
  17. onQuizClick: Function,
  18. /**
  19. * Function to execute if previous or next button is a lesson
  20. */
  21. onLessonClick: Function,
  22. /**
  23. * Function to execute if previous or next button is a topic
  24. */
  25. onTopicClick: Function,
  26. /**
  27. * App global style
  28. */
  29. global: Record<any, any>,
  30. /**
  31. * App colors
  32. */
  33. colors: Record<any, any>,
  34. t: TTranslationFunction,
  35. /**
  36. * Data about previous lesson/topic/quiz
  37. */
  38. prevObject: Record<any, any>,
  39. /**
  40. * Data about next lesson/topic/quiz
  41. */
  42. nextObject: Record<any, any>,
  43. /**
  44. * Course id of lesson/topic/quiz
  45. */
  46. courseId: number,
  47. /**
  48. * Shows an alert with information that next object is locked
  49. */
  50. nextLockedAlert: Function
  51. };
  52. /**
  53. * PrevNextPlaceholderProps
  54. */
  55. type PrevNextPlaceholderProps = {
  56. /**
  57. * App global style
  58. */
  59. global: Record<any, any>
  60. };
  61. /**
  62. * LessonHeaderProps
  63. */
  64. type LessonHeaderProps = {
  65. /**
  66. * App global style
  67. */
  68. global: Record<any, any>,
  69. /**
  70. * App colors
  71. */
  72. colors: Record<any, any>,
  73. lesson: TLessonViewModel,
  74. /**
  75. * Learndash labels
  76. */
  77. labels: Record<string, string>,
  78. t: TTranslationFunction,
  79. paddingTop: number,
  80. /**
  81. * Function to execute if previous or next button is a quiz
  82. */
  83. onQuizClick: Function,
  84. /**
  85. * Function to execute if previous or next button is a lesson
  86. */
  87. onLessonClick: Function,
  88. /**
  89. * Function to execute if previous or next button is a topic
  90. */
  91. onTopicClick: Function,
  92. /**
  93. * Data about previous lesson/topic/quiz
  94. */
  95. prevObject: Record<any, any>,
  96. /**
  97. * Data about next lesson/topic/quiz
  98. */
  99. nextObject: Record<any, any>,
  100. /**
  101. * Course id of lesson/topic/quiz
  102. */
  103. courseId: number,
  104. /**
  105. * Shows an alert with information that next object is locked
  106. */
  107. nextLockedAlert: Function,
  108. /**
  109. * Returns default back button component
  110. */
  111. backToCourse: React.ComponentType,
  112. /**
  113. * Returns `true` if screen is still loading
  114. */
  115. loading: boolean,
  116. course: TCourseViewModel,
  117. /**
  118. * Returns `false` lesson doesn't have a timer set. Will return a component if timer for the lesson is set.
  119. */
  120. renderTimer: boolean | React.ComponentType,
  121. /**
  122. * Function which updates the seconds passed in the lesson screen container
  123. */
  124. onTimePassed: Function,
  125. navigation: NavigationProp<any, any>,
  126. /**
  127. * Returns `true` if prev/next buttons should be hidden
  128. */
  129. hidePrevNext: boolean,
  130. /**
  131. * Returns `false` if hidePrevNext is `true`. Will return buttons which can navigate through previous and next screens if hidePrevNext is `false`
  132. */
  133. prevNext: boolean | React.ComponentType
  134. };
  135. /**
  136. * LessonScreenHeaderProps
  137. */
  138. type LessonScreenHeaderProps = {
  139. /**
  140. * Default styling for left section of the header
  141. */
  142. headerLeftStyle: Record<any, any>,
  143. /**
  144. * Default styling of lesson header
  145. */
  146. style: Record<any, any>,
  147. /**
  148. * Props which can be passed to an AuthWrapper
  149. */
  150. headerRightAuthWrapperProps: Record<any, any>,
  151. /**
  152. * App global style
  153. */
  154. global: Record<any, any>,
  155. /**
  156. * App colors
  157. */
  158. colors: Record<any, any>,
  159. lesson: TLessonViewModel,
  160. /**
  161. * Learndash labels
  162. */
  163. labels: Record<string, string>,
  164. t: TTranslationFunction,
  165. paddingTop: number,
  166. /**
  167. * Function to execute if previous or next button is a quiz
  168. */
  169. onQuizClick: Function,
  170. /**
  171. * Function to execute if previous or next button is a lesson
  172. */
  173. onLessonClick: Function,
  174. /**
  175. * Function to execute if previous or next button is a topic
  176. */
  177. onTopicClick: Function,
  178. /**
  179. * Data about previous lesson/topic/quiz
  180. */
  181. prevObject: Record<any, any>,
  182. /**
  183. * Data about next lesson/topic/quiz
  184. */
  185. nextObject: Record<any, any>,
  186. /**
  187. * Course id of lesson/topic/quiz
  188. */
  189. courseId: number,
  190. /**
  191. * Shows an alert with information that next object is locked
  192. */
  193. nextLockedAlert: Function,
  194. /**
  195. * Returns default back button component
  196. */
  197. backToCourse: React.ComponentType,
  198. /**
  199. * Returns `true` if screen is still loading
  200. */
  201. loading: boolean,
  202. course: TCourseViewModel,
  203. /**
  204. * Returns `false` lesson doesn't have a timer set. Will return a component if timer for the lesson is set.
  205. */
  206. renderTimer: boolean | React.ComponentType,
  207. /**
  208. * Function which updates the seconds passed in the lesson screen container
  209. */
  210. onTimePassed: Function,
  211. navigation: NavigationProp<any, any>,
  212. /**
  213. * Returns `true` if prev/next buttons should be hidden
  214. */
  215. hidePrevNext: boolean,
  216. /**
  217. * Returns `false` if hidePrevNext is `true`. Will return buttons which can navigate through previous and next screens if hidePrevNext is `false`
  218. */
  219. prevNext: boolean | React.ComponentType
  220. };
  221. /**
  222. * TransformLessonActionButtonsCallback
  223. */
  224. type TransformLessonActionButtonsCallback = {
  225. /**
  226. * Lesson action button
  227. */
  228. LessonButton: React.ComponentType,
  229. /**
  230. * Returns `true` if "Mark Complete" button should be shown
  231. */
  232. showComplete: boolean,
  233. /**
  234. * App global style
  235. */
  236. global: Record<any, any>,
  237. /**
  238. * App colors
  239. */
  240. colors: Record<any, any>,
  241. lesson: TLessonViewModel,
  242. /**
  243. * Returns `true` if lesson is being completed
  244. */
  245. completing: boolean,
  246. /**
  247. * Learndash labels
  248. */
  249. labels: Record<string, string>
  250. };
  251. /**
  252. * VideoProgressionComponentProps
  253. */
  254. type VideoProgressionComponentProps = {
  255. /**
  256. * Default styling applied to lesson video
  257. */
  258. lessonVideoStyle: Record<any, any>,
  259. /**
  260. * Returns `true` if controls should be displayed
  261. */
  262. controls: boolean,
  263. /**
  264. * Returns `true` if auto play should be enabled
  265. */
  266. autoplay: boolean,
  267. /**
  268. * Helper function which can be called when the video has finished playing
  269. */
  270. videoCallback: Function,
  271. /**
  272. * Video url
  273. */
  274. url: string,
  275. /**
  276. * Player width
  277. */
  278. width: number,
  279. /**
  280. * Player height
  281. */
  282. height: number,
  283. /**
  284. * App global style
  285. */
  286. global: Record<any, any>,
  287. /**
  288. * Returns `true` if screen is active
  289. */
  290. isNavActive: boolean,
  291. lesson: TLessonViewModel,
  292. /**
  293. * Returns `true` if video should be shown
  294. */
  295. showVideo: boolean,
  296. /**
  297. * Returns `true` if video has already been watched
  298. */
  299. videoWatched: boolean,
  300. /**
  301. * Helper function which dispatches an action and marks the video as watched
  302. */
  303. setVideoWatched: boolean,
  304. /**
  305. * Helper function which dispatches an action to mark the lesson as complete
  306. */
  307. completeLesson: Function,
  308. /**
  309. * Helper function which considers different conditions (such as if topics inside a lesson are already completed) before calling the `completeLesson` function
  310. */
  311. onCompleteButtonClick: Function
  312. };
  313. /**
  314. * AfterMaterialsComponentProps
  315. */
  316. type AfterMaterialsComponentProps = {
  317. /**
  318. * App global style
  319. */
  320. global: Record<any, any>,
  321. /**
  322. * App colors
  323. */
  324. colors: Record<any, any>,
  325. lesson: TLessonViewModel,
  326. /**
  327. * Learndash labels
  328. */
  329. labels: Record<string, string>,
  330. t: TTranslationFunction,
  331. /**
  332. * Function to execute if previous or next button is a quiz
  333. */
  334. onQuizClick: Function,
  335. /**
  336. * Function to execute if previous or next button is a lesson
  337. */
  338. onLessonClick: Function,
  339. /**
  340. * Function to execute if previous or next button is a topic
  341. */
  342. onTopicClick: Function,
  343. /**
  344. * Data about previous lesson/topic/quiz
  345. */
  346. prevObject: Record<any, any>,
  347. /**
  348. * Data about next lesson/topic/quiz
  349. */
  350. nextObject: Record<any, any>,
  351. /**
  352. * Shows an alert with information that next object is locked
  353. */
  354. nextLockedAlert: Function,
  355. /**
  356. * Returns `true` if screen is still loading
  357. */
  358. loading: boolean,
  359. course: Course,
  360. navigation: NavigationProp<any, any>,
  361. /**
  362. * Lesson materials
  363. */
  364. materials: undefined | string
  365. };
  366. /**
  367. * LessonActionComponentProps
  368. */
  369. type LessonActionComponentProps = {
  370. /**
  371. * App global style
  372. */
  373. global: Record<any, any>,
  374. /**
  375. * App colors
  376. */
  377. colors: Record<any, any>,
  378. lesson: TLessonViewModel,
  379. /**
  380. * Learndash labels
  381. */
  382. labels: Record<string, string>,
  383. t: TTranslationFunction,
  384. /**
  385. * Helper function which considers different conditions (such as if topics inside a lesson are already completed) before calling the `completeLesson` function
  386. */
  387. onCompleteButtonClick: Function,
  388. /**
  389. * Returns `true` if "Mark Complete" button should be shown
  390. */
  391. showComplete: boolean,
  392. /**
  393. * Returns `true` if lesson is being completed
  394. */
  395. completing: boolean,
  396. /**
  397. * Returns `true` if "Mark Complete" button should be disabled
  398. */
  399. completeDisabled: boolean,
  400. /**
  401. * Data about previous lesson/topic/quiz
  402. */
  403. prevObject: Record<any, any>,
  404. /**
  405. * Data about next lesson/topic/quiz
  406. */
  407. nextObject: Record<any, any>,
  408. /**
  409. * Function to execute if previous or next button is a quiz
  410. */
  411. onQuizClick: Function,
  412. /**
  413. * Function to execute if previous or next button is a lesson
  414. */
  415. onLessonClick: Function,
  416. /**
  417. * Function to execute if previous or next button is a topic
  418. */
  419. onTopicClick: Function
  420. };
  421. /**
  422. * OfflineComponentProps
  423. */
  424. type OfflineComponentProps = {
  425. /**
  426. * App global style
  427. */
  428. global: Record<any, any>,
  429. t: TTranslationFunction,
  430. /**
  431. * Default container style
  432. */
  433. containerStyle: Record<any, any>,
  434. /**
  435. * Default style of EmptyList component
  436. */
  437. emptyListStyle: Record<any, any>
  438. };
  439. /**
  440. * WebViewContentComponentProps
  441. */
  442. type WebViewContentComponentProps = {
  443. /**
  444. * Returns `true` if device is connected to an internet network
  445. */
  446. online: boolean,
  447. t: TTranslationFunction,
  448. /**
  449. * The default function used for determining how to handle webview requests.
  450. */
  451. onShouldStartLoadWithRequest: Function,
  452. /**
  453. * Default height
  454. */
  455. height: number,
  456. /**
  457. * Contains data of the web site to be loaded in the webview
  458. */
  459. source: Record<any, any>,
  460. /**
  461. * App global style
  462. */
  463. global: Record<any, any>,
  464. /**
  465. * App colors
  466. */
  467. colors: Record<any, any>,
  468. /**
  469. * Modal which shows up when the "Read More" component is pressed
  470. */
  471. ModalHeaderComponent: React.FC
  472. };
  473. /**
  474. * LessonMaterialsSectionTitleProps
  475. */
  476. type LessonMaterialsSectionTitleProps = {
  477. /**
  478. * App global style
  479. */
  480. global: Record<any, any>,
  481. t: TTranslationFunction
  482. };
  483. /**
  484. * AssignmentsHeaderProps
  485. */
  486. type AssignmentsHeaderProps = {
  487. /**
  488. * App global style
  489. */
  490. global: Record<any, any>,
  491. t: TTranslationFunction,
  492. /**
  493. * Total number of assignments uploaded
  494. */
  495. totalAssignments: number,
  496. /**
  497. * Number of assignments approved
  498. */
  499. approved: number
  500. };
  501. /**
  502. * AssignmentItemIconProps
  503. */
  504. type AssignmentItemIconProps = {
  505. /**
  506. * Default icon for the item
  507. */
  508. icon: number,
  509. /**
  510. * Default style for the item
  511. */
  512. styles: Record<any, any>
  513. };
  514. /**
  515. * AssignmentItemTitleProps
  516. */
  517. type AssignmentItemTitleProps = {
  518. /**
  519. * Assignment item title
  520. */
  521. title: string,
  522. /**
  523. * Default style for the item
  524. */
  525. style: Record<any, any>
  526. };
  527. /**
  528. * AssignmentItemStatusProps
  529. */
  530. type AssignmentItemStatusProps = {
  531. /**
  532. * App global style
  533. */
  534. global: Record<any, any>,
  535. /**
  536. * Assignment item status
  537. */
  538. text: string,
  539. /**
  540. * Default container style
  541. */
  542. containerStyle: Record<any, any>,
  543. /**
  544. * Default text style
  545. */
  546. textStyle: Record<any, any>
  547. };
  548. /**
  549. * AssignmentItemCommentProps
  550. */
  551. type AssignmentItemCommentProps = {
  552. /**
  553. * Opens the comments modal
  554. */
  555. pressHandler: Function,
  556. /**
  557. * Default button color
  558. */
  559. tintColor: string,
  560. /**
  561. * Default container style
  562. */
  563. containerStyle: Record<any, any> | undefined,
  564. /**
  565. * Number of comments
  566. */
  567. count: number,
  568. /**
  569. * Button size
  570. */
  571. size: string,
  572. /**
  573. * App global style
  574. */
  575. global: Record<any, any>,
  576. /**
  577. * Can be used to determine if button should use light or dark style
  578. */
  579. lightMode: boolean | undefined
  580. };
  581. /**
  582. * AssignmentItemDownloadProps
  583. */
  584. type AssignmentItemDownloadProps = {
  585. /**
  586. * Downloads the assignment item
  587. */
  588. pressHandler: Function,
  589. /**
  590. * Default button color
  591. */
  592. tintColor: string,
  593. /**
  594. * Default container style
  595. */
  596. containerStyle: Record<any, any> | undefined,
  597. /**
  598. * Number of comments
  599. */
  600. count: number,
  601. /**
  602. * Button size
  603. */
  604. size: string,
  605. /**
  606. * App global style
  607. */
  608. global: Record<any, any>,
  609. /**
  610. * Can be used to determine if button should use light or dark style
  611. */
  612. lightMode: boolean | undefined
  613. };
  614. /**
  615. * AssignmentItemDownloadProgressProps
  616. */
  617. type AssignmentItemDownloadProgressProps = {
  618. /**
  619. * Download progress
  620. */
  621. progress: number,
  622. /**
  623. * Unfilled color of the ProgressCircle component
  624. */
  625. unfilledColor: string,
  626. /**
  627. * Color of the ProgressCircle component
  628. */
  629. tintColor: string
  630. };
  631. /**
  632. * MaterialsComponentProps
  633. */
  634. type MaterialsComponentProps = {
  635. /**
  636. * App global style
  637. */
  638. global: Record<any, any>,
  639. /**
  640. * App colors
  641. */
  642. colors: Record<any, any>,
  643. t: TTranslationFunction,
  644. /**
  645. * Contents of course materials field
  646. */
  647. materials: string,
  648. /**
  649. * Default styles for different HTML tags
  650. */
  651. tagsStyles: Record<any, any>,
  652. /**
  653. * Default styles for course materials
  654. */
  655. materialsStyles: Record<any, any>,
  656. /**
  657. * Function to set a font style
  658. */
  659. baseFontStyle: Function,
  660. navigation: NavigationProp<any, any>,
  661. /**
  662. * Handles link press for the HTML component
  663. */
  664. onLinkPress: Function
  665. };
  666. /**
  667. * TransformLessonViewModelCallback
  668. */
  669. type TransformLessonViewModelCallback = {
  670. viewModel: TLessonViewModel,
  671. lesson: Record<any, any>
  672. };
  673. /**
  674. * @class
  675. * Lesson Screen Hooks.
  676. * Instance name: lessonSingleScreenApi
  677. You can use this hook to customize the lesson screen such as by adding a custom call to action button and other actions.
  678. * @example
  679. * externalCodeSetup.lessonSingleScreenApi.METHOD_NAME
  680. */
  681. export class LessonSingleScreenHooksApi {
  682. /**
  683. * @deprecated
  684. * Filters navigation onPress function that is used to navigate to Topic
  685. */
  686. //@ts-ignore
  687. topicNavigationFilter = (onPress, navigation, topic, lessonId, courseId) =>
  688. onPress;
  689. /**
  690. * @deprecated
  691. */
  692. //@ts-ignore
  693. setTopicNavigationFilter = topicNavigationFilter => {
  694. this.topicNavigationFilter = topicNavigationFilter;
  695. };
  696. PrevNextComponent: React.ComponentType<PrevNextComponentProps> | null = null;
  697. /**
  698. * You can use this to replace the previous and next buttons on the lessons single screen if you want to change the default option which always displays the prev/next buttons.
  699. * @method
  700. * @param {React.ComponentType<PrevNextComponentProps>} PrevNextComponent
  701. * @example <caption> Hide the previous and next buttons if you're only using one lesson in a course </caption>
  702. *
  703. * //In custom_code/components/PrevNext.js
  704. *
  705. * import React from "react";
  706. * import PrevNext from "@src/components/Course/PrevNext";
  707. *
  708. * const PrevNextComponent = props => {
  709. *
  710. * const { onQuizClick,
  711. * onLessonClick,
  712. * onTopicClick,
  713. * global,
  714. * colors,
  715. * t,
  716. * prevObject,
  717. * nextObject,
  718. * courseId,
  719. * nextLockedAlert } = props;
  720. *
  721. * if (! nextObject && ! prevObject){
  722. * return null;
  723. * }
  724. *
  725. * return (
  726. * <PrevNext
  727. * {...{ onQuizClick, onLessonClick, onTopicClick }}
  728. * global={global}
  729. * colors={colors}
  730. * t={t}
  731. * prevObject={prevObject}
  732. * nextObject={nextObject}
  733. * courseId={courseId}
  734. * nextLockedAlert={nextLockedAlert}
  735. * />
  736. * );
  737. * };
  738. * export default PrevNextComponent;
  739. *
  740. * //In custom_code/index.js...
  741. *
  742. * ...
  743. *
  744. * import PrevNextComponent from './components/PrevNext';
  745. * export const applyCustomCode = externalCodeSetup => {
  746. * externalCodeSetup.lessonSingleScreenApi.setPrevNextComponent(props => <PrevNextComponent {...props} />)
  747. * }
  748. *
  749. * @example <caption> Hide previous and next buttons </caption>
  750. *
  751. * ...
  752. *
  753. * export const applyCustomCode = externalCodeSetup => {
  754. * externalCodeSetup.lessonSingleScreenApi.setPrevNextComponent(() => null)
  755. * }
  756. *
  757. */
  758. setPrevNextComponent = (
  759. PrevNextComponent: React.ComponentType<PrevNextComponentProps> | null
  760. ) => {
  761. this.PrevNextComponent = PrevNextComponent;
  762. };
  763. PrevNextPlaceholder: React.ComponentType<
  764. PrevNextPlaceholderProps
  765. > | null = null;
  766. /**
  767. * You can use this to replace the previous and next buttons' placeholders on the lessons single screen.
  768. * The previous and next buttons' placeholders appear while the app is still loading the lesson.
  769. * @method
  770. * @param {React.ComponentType<PrevNextPlaceholderProps>} PrevNextPlaceholder
  771. * @example <caption> Change placeholder color </caption>
  772. *
  773. * externalCodeSetup.lessonSingleScreenApi.setPrevNextPlaceholder((props) => {
  774. *
  775. * const {global} = props;
  776. *
  777. * return <>
  778. * <View
  779. * style={[
  780. * global.wrappedButton,
  781. * global.wrappedTextButton,
  782. * { marginRight: 4, width: 64, backgroundColor: "blue" }
  783. * ]}
  784. * />
  785. * <View
  786. * style={[
  787. * global.wrappedButton,
  788. * global.wrappedTextButton,
  789. * { width: 65 , backgroundColor: "red"}
  790. * ]}
  791. * />
  792. * </>
  793. * })
  794. *
  795. * @example <caption> Hide the placeholder even while the screen is still loading </caption>
  796. *
  797. * externalCodeSetup.lessonSingleScreenApi.setPrevNextPlaceholder(() => null)
  798. *
  799. */
  800. setPrevNextPlaceholder = (
  801. PrevNextPlaceholder: React.ComponentType<PrevNextPlaceholderProps> | null
  802. ) => {
  803. this.PrevNextPlaceholder = PrevNextPlaceholder;
  804. };
  805. transformLessonActionButtons: (
  806. LessonButton: JSX.Element,
  807. showComplete: boolean,
  808. global: Record<any, any>,
  809. colors: Record<any, any>,
  810. lesson: TLessonViewModel,
  811. completing: boolean,
  812. labels: Record<string, string>
  813. ) => JSX.Element = LessonButton => LessonButton;
  814. /**
  815. * You can transform the default lesson action button by replacing it with your preferred action buttons.
  816. * @param {TransformLessonActionButtonsCallback} transformLessonActionButtons
  817. * @method
  818. * @example <caption> Add more components for lesson action </caption>
  819. *
  820. * externalCodeSetup.lessonSingleScreenApi.setTransformLessonActionButtons((
  821. * LessonButton,
  822. * showComplete,
  823. * global,
  824. * colors,
  825. * lesson,
  826. * completing,
  827. * labels) => {
  828. *
  829. * const Buttons =
  830. * <View style={[global.row, {backgroundColor: "#fff"}]}>
  831. *
  832. * <View style={{ width:"45%", backgroundColor: "#fff", paddingHorizontal: 20, paddingVertical: 15 }}>
  833. * <AppTouchableOpacity
  834. * style={[
  835. * global.completeLessonButtonW,
  836. * { flex: 1, backgroundColor: colors.primaryButtonBg }
  837. * ]}
  838. * onPress={() => {
  839. * //Do function...
  840. * }}
  841. * >
  842. * <View style={global.row}>
  843. * <View style={global.linkWithArrow}>
  844. * <Text
  845. * style={{color: "#fff", fontWeight: "bold"}}
  846. * >
  847. * Questions?
  848. * </Text>
  849. * </View>
  850. * </View>
  851. * </AppTouchableOpacity>
  852. * </View>
  853. * <View style={{ width: "45%", marginLeft: "auto" }}>
  854. * {LessonButton}
  855. * </View>
  856. * </View>
  857. *
  858. * return Buttons;
  859. * })
  860. */
  861. setTransformLessonActionButtons = (
  862. transformLessonActionButtons: (
  863. LessonButton: JSX.Element,
  864. showComplete: boolean,
  865. global: Record<any, any>,
  866. colors: Record<any, any>,
  867. lesson: TLessonViewModel,
  868. completing: boolean,
  869. labels: Record<string, string>
  870. ) => JSX.Element
  871. ) => {
  872. this.transformLessonActionButtons = transformLessonActionButtons;
  873. };
  874. LessonTitleComponent: React.ComponentType<LessonHeaderProps> | null = null;
  875. /**
  876. * You can use this to replace the lesson's title component.
  877. * For example, you can use this to change the title's background or add an ellipsis feature to it.
  878. * @method
  879. * @param {React.ComponentType<LessonHeaderProps>} LessonTitleComponent
  880. * @example <caption> Change component's background and add ellipsis to text </caption>
  881. *
  882. * //In custom_code/components/LessonTitle.js...
  883. *
  884. * import React from "react";
  885. * import {View, Text} from "react-native";
  886. * import Animated from "react-native-reanimated";
  887. *
  888. * const LessonTitle = (props) => {
  889. *
  890. * const {
  891. * global,
  892. * colors,
  893. * paddingTop,
  894. * lesson,
  895. * t,
  896. * labels
  897. * } = props;
  898. *
  899. * const paddingBottom = 14;
  900. *
  901. * let backgroundColor = colors.bodyFrontBg;
  902. * if (lesson.title.includes("Android")){
  903. * backgroundColor = "red"
  904. * }
  905. *
  906. * return (
  907. * <Animated.View
  908. * style={[
  909. * {
  910. * backgroundColor: backgroundColor,
  911. * width: "100%",
  912. * shadowOffset: {width: 0, height: 1},
  913. * shadowRadius: 1,
  914. * shadowColor: "#000",
  915. * shadowOpacity: 0.05
  916. * }
  917. * ]}
  918. * >
  919. * <View
  920. * style={[
  921. * global.row,
  922. * {
  923. * justifyContent: "space-between",
  924. * alignItems: "flex-start",
  925. * paddingTop,
  926. * paddingBottom
  927. * }
  928. * ]}
  929. * >
  930. * <Animated.View
  931. * style={{
  932. * flex: 1,
  933. * paddingHorizontal: 20
  934. * }}
  935. * >
  936. * <Animated.Text
  937. * style={[
  938. * global.courseHeaderTitle,
  939. * {marginBottom: 5}
  940. * ]}
  941. * numberOfLines={1} //Adds ellipsis to lesson title
  942. * >
  943. * {lesson.title}
  944. * </Animated.Text>
  945. * <Text style={global.courseHeaderSubTitle}>
  946. * {t("lesson:count", {
  947. * label: labels.lesson,
  948. * current: lesson.order,
  949. * total: lesson.total
  950. * })}
  951. * </Text>
  952. * </Animated.View>
  953. * </View>
  954. * </Animated.View>
  955. * );
  956. * };
  957. *
  958. * export default LessonTitle;
  959. *
  960. * //In custom_code/index.js...
  961. *
  962. * ...
  963. * import LessonTitle from "./components/LessonTitle"
  964. * export const applyCustomCode = externalCodeSetup => {
  965. * externalCodeSetup.lessonSingleScreenApi.setLessonTitleComponent(props => <LessonTitle {...props} />)
  966. * }
  967. *
  968. * @example <caption> Add screen header components to lesson title </caption>
  969. *
  970. * // In custom_code/components/LessonTitle.js...
  971. * import React from "react";
  972. * import { useSelector } from "react-redux";
  973. * import { View, Text, StyleSheet } from "react-native";
  974. * import Animated from "react-native-reanimated";
  975. * import PrevNext from "@src/components/Course/PrevNext";
  976. * import Icon from "@src/components/Icon";
  977. * import TimeCounter from "@src/components/TimeCounterAutoPause";
  978. *
  979. * const LessonTitle = (props) => {
  980. *
  981. * const {
  982. * title,
  983. * global,
  984. * colors,
  985. * paddingTop,
  986. * lesson,
  987. * t,
  988. * labels,
  989. * onQuizClick,
  990. * onLessonClick,
  991. * onTopicClick,
  992. * prevObject,
  993. * nextObject,
  994. * courseId,
  995. * nextLockedAlert,
  996. * backToCourse,
  997. * course,
  998. * loading,
  999. * renderTimer,
  1000. * onTimePassed,
  1001. * navigation
  1002. * } = props;
  1003. *
  1004. * const paddingBottom = 14;
  1005. *
  1006. * const renderDefaultTimer = false
  1007. *
  1008. * const isLessonCompleting = useSelector(state => state.singleLesson.completing == lesson.id);
  1009. *
  1010. * const CustomTimer = () => {
  1011. *
  1012. * if (loading || !lesson.requireTimer || isLessonCompleting)
  1013. * return null;
  1014. *
  1015. * return <View
  1016. * style={{
  1017. * flexDirection: "row",
  1018. * alignItems: "center"
  1019. * }}
  1020. * >
  1021. * <Text>My Custom Timer</Text>
  1022. * <Icon
  1023. * icon={{fontIconName: "stopwatch", weight: 400}}
  1024. * webIcon={"IconArrowBack"}
  1025. * tintColor={colors.textIconColor}
  1026. * styles={{
  1027. * marginRight: 6,
  1028. * height: 20,
  1029. * width: 20
  1030. * }}
  1031. * />
  1032. * <TimeCounter
  1033. * paused={!navigation.isActive}
  1034. * onTimePassed={onTimePassed}
  1035. * initialSeconds={0}
  1036. * textProps={{ style: global.timer }}
  1037. * />
  1038. * </View>
  1039. * }
  1040. *
  1041. * return (
  1042. * <Animated.View
  1043. * style={[
  1044. * {
  1045. * backgroundColor: colors.bodyFrontBg,
  1046. * width: "100%",
  1047. * shadowOffset: { width: 0, height: 1 },
  1048. * shadowRadius: 1,
  1049. * shadowColor: "#000",
  1050. * shadowOpacity: 0.05,
  1051. * marginTop: 50
  1052. * }
  1053. * ]}
  1054. * >
  1055. * <View
  1056. * style={[
  1057. * global.row,
  1058. * {
  1059. * justifyContent: "space-between",
  1060. * alignItems: "flex-start",
  1061. * paddingTop,
  1062. * paddingBottom
  1063. * }
  1064. * ]}
  1065. * >
  1066. * <Animated.View
  1067. * style={{
  1068. * flex: 1,
  1069. * paddingHorizontal: 20
  1070. * }}
  1071. * >
  1072. * //ScreenHeader components
  1073. * <View style={{ flexDirection: "row", marginLeft: -20 }}>
  1074. * {backToCourse}
  1075. *
  1076. * <PrevNext
  1077. * {...{ onQuizClick, onLessonClick, onTopicClick }}
  1078. * global={global}
  1079. * colors={colors}
  1080. * t={t}
  1081. * prevObject={prevObject}
  1082. * nextObject={nextObject}
  1083. * courseId={courseId}
  1084. * nextLockedAlert={nextLockedAlert}
  1085. * />
  1086. *
  1087. * {renderDefaultTimer ? renderTimer : <CustomTimer />}
  1088. *
  1089. * </View>
  1090. * //End ScreenHeader components
  1091. *
  1092. * <Animated.Text
  1093. * style={[
  1094. * global.courseHeaderTitle,
  1095. * { marginBottom: 5 }
  1096. * ]}
  1097. * numberOfLines={1} //Adds ellipsis to lesson title
  1098. * >
  1099. * {lesson.title}
  1100. * </Animated.Text>
  1101. *
  1102. * <Text style={global.courseHeaderSubTitle}>
  1103. * {t("lesson:count", {
  1104. * label: labels.lesson,
  1105. * current: lesson.order,
  1106. * total: lesson.total
  1107. * })}
  1108. * </Text>
  1109. * </Animated.View>
  1110. * </View>
  1111. * </Animated.View>
  1112. * );
  1113. * };
  1114. *
  1115. * export default LessonTitle;
  1116. *
  1117. * //In custom_code/index.js...
  1118. *
  1119. * ...
  1120. * import LessonTitle from "./components/LessonTitle"
  1121. * export const applyCustomCode = externalCodeSetup => {
  1122. * externalCodeSetup.lessonSingleScreenApi.setLessonTitleComponent(props => <LessonTitle {...props} />)
  1123. * }
  1124. */
  1125. setLessonTitleComponent = (
  1126. LessonTitleComponent: React.ComponentType<LessonHeaderProps> | null
  1127. ) => {
  1128. this.LessonTitleComponent = LessonTitleComponent;
  1129. };
  1130. LessonScreenHeader: React.ComponentType<
  1131. LessonScreenHeaderProps
  1132. > | null = null;
  1133. /**
  1134. * You can use this hook to customize the header of the Lesson Single Screen which by default, contains the back-to-course button and the previous/next buttons.
  1135. * @method
  1136. * @param {React.ComponentType<LessonScreenHeaderProps>} LessonScreenHeader
  1137. * @example
  1138. *
  1139. * //In custom_code/components/LessonScreenHeader.js...
  1140. *
  1141. * import React from "react";
  1142. * import {View} from "react-native";
  1143. * import Animated from "react-native-reanimated";
  1144. * import {DEVICE_WIDTH} from "@src/styles/global";
  1145. * import AuthWrapper from "@src/components/AuthWrapper";
  1146. *
  1147. * const Header = ({
  1148. * headerLeftStyle,
  1149. * style,
  1150. * global,
  1151. * backToCourse,
  1152. * renderTimer,
  1153. * headerRightAuthWrapperProps,
  1154. * prevNext
  1155. * }) => {
  1156. * return (
  1157. * <Animated.View
  1158. * style={[
  1159. * global.row,
  1160. * global.fakeHeader,
  1161. * {
  1162. * backgroundColor: "transparent",
  1163. * paddingHorizontal: 10,
  1164. * overflow: "hidden"
  1165. * },
  1166. * {
  1167. * width: DEVICE_WIDTH
  1168. * },
  1169. * style
  1170. * ]}
  1171. * >
  1172. * <View
  1173. * style={[
  1174. * {
  1175. * alignItems: "center",
  1176. * justifyContent: "center",
  1177. * flexDirection: "row",
  1178. * flex: 1,
  1179. * height: "100%"
  1180. * }
  1181. * ]}
  1182. * >
  1183. * <View style={[global.headerButtonLeft, headerLeftStyle]}>
  1184. * {backToCourse}
  1185. * </View>
  1186. * <View style={[global.headerCustomTitle]}>
  1187. * {renderTimer}
  1188. * </View>
  1189. * <View style={[global.headerButtonRight]}>
  1190. * <AuthWrapper
  1191. * actionOnGuestLogin={"hide"}
  1192. * {...headerRightAuthWrapperProps}
  1193. * >
  1194. * {prevNext}
  1195. * </AuthWrapper>
  1196. * </View>
  1197. * </View>
  1198. * </Animated.View>
  1199. * );
  1200. * };
  1201. *
  1202. * export default Header;
  1203. *
  1204. * //In custom_code/index.js...
  1205. *
  1206. * import LessonScreenHeader from "./components/LessonScreenHeader";
  1207. * export const applyCustomCode = externalCodeSetup => {
  1208. * externalCodeSetup.lessonSingleScreenApi.setLessonScreenHeader(props => <LessonScreenHeader {...props}/>)
  1209. * }
  1210. */
  1211. setLessonScreenHeader = (
  1212. LessonScreenHeader: React.ComponentType<LessonScreenHeaderProps> | null
  1213. ) => {
  1214. this.LessonScreenHeader = LessonScreenHeader;
  1215. };
  1216. VideoProgressionComponent: React.ComponentType<
  1217. VideoProgressionComponentProps
  1218. > | null = null;
  1219. /**
  1220. * You can use this hook to customize the Video Progression component in the single lesson screen.
  1221. * For example, you can use this to change the size of the video player or add a custom function when the video finishes playing on a lesson.
  1222. * @method
  1223. * @param {React.ComponentType<VideoProgressionComponentProps>} VideoProgressionComponent
  1224. * @example <caption> Execute a custom callback when video has finished playing </caption>
  1225. *
  1226. * ...
  1227. * import AppVideo from "@src/components/Video/AppVideo";
  1228. *
  1229. * export const applyCustomCode = externalCodeSetup => {
  1230. *
  1231. * externalCodeSetup.lessonSingleScreenApi.setVideoProgressionComponent(props => {
  1232. * const {
  1233. * lessonVideoStyle,
  1234. * controls,
  1235. * autoPlay,
  1236. * videoCallback,
  1237. * url,
  1238. * width,
  1239. * height,
  1240. * global,
  1241. * isNavActive,
  1242. * lesson,
  1243. * videoWatched,
  1244. * setVideoWatched,
  1245. * onCompleteButtonClick
  1246. * } = props;
  1247. *
  1248. * const customCallback = settings => event => {
  1249. *
  1250. * if (event.nativeEvent.data === "ENDED") {
  1251. *
  1252. * if (!videoWatched)
  1253. * setVideoWatched();
  1254. *
  1255. * Alert.alert(
  1256. * "Video Complete",
  1257. * "You have finished watching the video. Do you want to mark the lesson as complete?",
  1258. * [
  1259. * {
  1260. * text: "Cancel",
  1261. * onPress: () => console.log("Cancel Pressed"),
  1262. * style: "cancel"
  1263. * },
  1264. * { text: "OK", onPress: () => onCompleteButtonClick() }
  1265. * ]
  1266. * );
  1267. * }
  1268. *
  1269. * }
  1270. *
  1271. * return <View style={lessonVideoStyle}>
  1272. * <Text>Please finish watching the video below to proceed...</Text>
  1273. * <AppVideo
  1274. * controls={controls}
  1275. * autoPlay={autoPlay}
  1276. * // videoCallback={videoCallback()}
  1277. * videoCallback={customCallback(lesson.settings)}
  1278. * url={url}
  1279. * width={width}
  1280. * height={height}
  1281. * global={global}
  1282. * isNavActive={isNavActive}
  1283. * />
  1284. * </View>
  1285. * })
  1286. * }
  1287. *
  1288. */
  1289. setVideoProgressionComponent = (
  1290. VideoProgressionComponent: React.ComponentType<
  1291. VideoProgressionComponentProps
  1292. > | null
  1293. ) => {
  1294. this.VideoProgressionComponent = VideoProgressionComponent;
  1295. };
  1296. AfterMaterialsComponent: React.ComponentType<
  1297. AfterMaterialsComponentProps
  1298. > | null = null;
  1299. /**
  1300. * You can use this hook to add a component at the bottom of the lesson single screen just after the component that displays the materials.
  1301. * @method
  1302. * @param {React.ComponentType<AfterMaterialsComponentProps>} AfterMaterialsComponent
  1303. * @example
  1304. *
  1305. * //In custom_code/components/LessonBottomComponent.js...
  1306. *
  1307. * import React from "react";
  1308. * import { View, Text, TouchableOpacity } from "react-native";
  1309. * const LessonBottomComponent = props => {
  1310. *
  1311. * const {
  1312. * colors,
  1313. * course,
  1314. * navigation
  1315. * } = props;
  1316. *
  1317. * const back = () => {
  1318. * navigation.navigate({
  1319. * routeName: "CoursesSingleScreen",
  1320. * params: {
  1321. * id: course.id,
  1322. * course
  1323. * },
  1324. * key: course.id.toString()
  1325. * })
  1326. * }
  1327. * return <View style={{
  1328. * backgroundColor: colors.bodyFrontBg,
  1329. * paddingHorizontal: 20,
  1330. * paddingBottom: 20,
  1331. * minHeight: 100,
  1332. * }}>
  1333. * <Text>This lesson is part of the {course.title.rendered} course </Text>
  1334. * <TouchableOpacity onPress={back}>
  1335. * <Text>
  1336. * Back To Course
  1337. * </Text>
  1338. * </TouchableOpacity>
  1339. * </View>
  1340. * }
  1341. *
  1342. * export default LessonBottomComponent;
  1343. *
  1344. * //In custom_code/index.js...
  1345. *
  1346. * ...
  1347. *
  1348. * import LessonBottomComponent from "./components/LessonBottomComponent";
  1349. * export const applyCustomCode = externalCodeSetup => {
  1350. * externalCodeSetup.lessonSingleScreenApi.setAfterMaterialsComponent(props => <LessonBottomComponent {...props}/>)
  1351. * }
  1352. */
  1353. setAfterMaterialsComponent = (
  1354. AfterMaterialsComponent: React.ComponentType<
  1355. AfterMaterialsComponentProps
  1356. > | null
  1357. ) => {
  1358. this.AfterMaterialsComponent = AfterMaterialsComponent;
  1359. };
  1360. LessonActionComponent: React.ComponentType<
  1361. LessonActionComponentProps
  1362. > | null = null;
  1363. /**
  1364. * You can use this hook to customize the "Mark Complete" / "Completed" button.
  1365. * For example, you can add your own loading animation when the "Mark Complete" button is pressed.
  1366. * @method
  1367. * @param {React.ComponentType<LessonActionComponentProps>} LessonActionComponent
  1368. * @example <caption> Add a "Completing..." text when marking the lesson complete </caption>
  1369. *
  1370. * //In custom_code/components/LessonActionComponent.js...
  1371. * import React from "react";
  1372. * import { View, Text, ActivityIndicator } from "react-native";
  1373. *
  1374. * import AuthWrapper from "@src/components/AuthWrapper";
  1375. * import AppTouchableOpacity from "@src/components/AppTouchableOpacity";
  1376. * import Icon from "@src/components/Icon";
  1377. * import { isColorDark } from "@src/utils";
  1378. *
  1379. * const LessonActionComponent = ({
  1380. * showComplete,
  1381. * global,
  1382. * colors,
  1383. * t,
  1384. * lesson,
  1385. * onCompleteButtonClick,
  1386. * completing,
  1387. * completeDisabled,
  1388. * labels
  1389. * }) => (<AuthWrapper actionOnGuestLogin={"hide"}>
  1390. * {showComplete && (
  1391. * <View
  1392. * style={[
  1393. * global.row,
  1394. * {
  1395. * backgroundColor: colors.bodyFrontBg,
  1396. * borderTopColor: colors.borderColor
  1397. * },
  1398. * global.lessonActionButtonContainer
  1399. * ]}
  1400. * >
  1401. * <AppTouchableOpacity
  1402. * style={[
  1403. * { flex: 1 },
  1404. * {
  1405. * opacity: !lesson.completed && completeDisabled ? 0.5 : 1,
  1406. * backgroundColor: !lesson.completed
  1407. * ? colors.primaryButtonBg
  1408. * : colors.bodyFrontBg
  1409. * },
  1410. * global.completeLessonButtonW
  1411. * ]}
  1412. * disabled={lesson.completed || completeDisabled}
  1413. * onPress={onCompleteButtonClick}
  1414. * >
  1415. * <View style={global.row}>
  1416. * <View style={global.linkWithArrow}>
  1417. * {!lesson.completed ? (
  1418. * completing && (
  1419. * <>
  1420. * <Text style={{color: "#fff"}}>Completing...</Text>
  1421. * <ActivityIndicator
  1422. * animating={true}
  1423. * color={colors.primaryButtonColor}
  1424. * size="small"
  1425. * style={global.lessonButtonLoadingIcon}
  1426. * />
  1427. * </>
  1428. * )
  1429. * ) : (
  1430. * <Icon
  1431. * webIcon={""}
  1432. * icon={{fontIconName: "check", weight: 200}}
  1433. * styles={global.lessonActionCompleteIcon}
  1434. * />
  1435. * )}
  1436. * <Text
  1437. * style={[
  1438. * {
  1439. * marginLeft: 10,
  1440. * color: !lesson.completed
  1441. * ? colors.primaryButtonColor
  1442. * : isColorDark(colors.bodyFrontBg)
  1443. * ? "white"
  1444. * : "black"
  1445. * },
  1446. * !lesson.completed
  1447. * ? global.completeLessonButton
  1448. * : global.completeButton
  1449. * ]}
  1450. * >
  1451. * {t(
  1452. * lesson.completed
  1453. * ? "lesson:completed"
  1454. * : "lesson:completeLesson",
  1455. * { label: labels.lesson.toLowerCase() }
  1456. * )}
  1457. * </Text>
  1458. * </View>
  1459. * </View>
  1460. * </AppTouchableOpacity>
  1461. * </View>
  1462. * )}
  1463. * </AuthWrapper>)
  1464. *
  1465. * export default LessonActionComponent;
  1466. *
  1467. * //In custom_code/index.js...
  1468. *
  1469. * ...
  1470. *
  1471. * import LessonActionComponent from "./components/LessonActionComponent";
  1472. * export const applyCustomCode = (externalCodeSetup: any) => {
  1473. * externalCodeSetup.lessonSingleScreenApi.setLessonActionComponent(props => <LessonActionComponent {...props} />)
  1474. * }
  1475. *
  1476. */
  1477. setLessonActionComponent = (
  1478. LessonActionComponent: React.ComponentType<
  1479. LessonActionComponentProps
  1480. > | null
  1481. ) => {
  1482. this.LessonActionComponent = LessonActionComponent;
  1483. };
  1484. lessonViewModelFilter = (
  1485. viewModel: TLessonViewModel | Record<any, any>,
  1486. lesson: Record<any, any>
  1487. ) => viewModel;
  1488. /**
  1489. * Sets the callback function that can change an existing lesson view model object.
  1490. * @method
  1491. * @param {TransformLessonViewModelCallback} lessonViewModelFilter
  1492. * @example <caption>Remove the native blocks in a lesson</caption>
  1493. * externalCodeSetup.lessonSingleScreenApi.setLessonViewModelFilter((viewModel, lesson) => {
  1494. * return {
  1495. * ...viewModel,
  1496. * contentNative: []
  1497. * }
  1498. * })
  1499. */
  1500. setLessonViewModelFilter = (
  1501. lessonViewModelFilter: (
  1502. viewModel: TLessonViewModel | Record<any, any>,
  1503. lesson: Record<any, any>
  1504. ) => TLessonViewModel | Record<any, any>
  1505. ) => {
  1506. this.lessonViewModelFilter = lessonViewModelFilter;
  1507. };
  1508. OfflineComponent: React.ComponentType<OfflineComponentProps> | null = null;
  1509. /**
  1510. * 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 lesson content while the device is offline.
  1511. * @method
  1512. * @param {React.ComponentType<OfflineComponentProps>} OfflineComponent
  1513. * @example
  1514. *
  1515. * ...
  1516. *
  1517. * import EmptyList from "@src/components/EmptyList";
  1518. *
  1519. * export const applyCustomCode = (externalCodeSetup) => {
  1520. * externalCodeSetup.lessonSingleScreenApi.setOfflineComponent(
  1521. * ({containerStyle, t, global, emptyListStyle}) => (
  1522. * <View style={containerStyle}>
  1523. * <EmptyList
  1524. * emptyText={{
  1525. * title: t("common:contentOfflineMessage"),
  1526. * icon: {fontIconName: "wifi-slash", weight: 400}
  1527. * }}
  1528. * global={global}
  1529. * style={emptyListStyle}
  1530. * />
  1531. * </View>
  1532. * )
  1533. * );
  1534. * }
  1535. */
  1536. setOfflineComponent = (
  1537. OfflineComponent: React.ComponentType<OfflineComponentProps> | null
  1538. ) => {
  1539. this.OfflineComponent = OfflineComponent;
  1540. };
  1541. WebViewContentComponent: React.ComponentType<
  1542. WebViewContentComponentProps
  1543. > | null = null;
  1544. /**
  1545. * You can use this hook to replace the webview being used in the lesson content.
  1546. * For example, you can choose to replace it with the default react-native webview.
  1547. * @method
  1548. * @param {React.ComponentType<WebViewContentComponentProps>} WebViewContentComponent
  1549. * @example
  1550. *
  1551. * ...
  1552. *
  1553. * import WebViewWithMore from "@src/components/WebViewWithMore";
  1554. * export const applyCustomCode = (externalCodeSetup) => {
  1555. * externalCodeSetup.lessonSingleScreenApi.setWebViewContentComponent(
  1556. * ({
  1557. * online,
  1558. * t,
  1559. * onShouldStartLoadWithRequest,
  1560. * height,
  1561. * source,
  1562. * global,
  1563. * colors,
  1564. * ModalHeaderComponent
  1565. * }) => (
  1566. * <WebViewWithMore
  1567. * online={online}
  1568. * t={t}
  1569. * onShouldStartLoadWithRequest={onShouldStartLoadWithRequest}
  1570. * height={height}
  1571. * source={source}
  1572. * global={global}
  1573. * colors={colors}
  1574. * ModalHeaderComponent={ModalHeaderComponent}
  1575. * />
  1576. * )
  1577. * );
  1578. * }
  1579. */
  1580. setWebViewContentComponent = (
  1581. WebViewContentComponent: React.ComponentType<
  1582. WebViewContentComponentProps
  1583. > | null
  1584. ) => {
  1585. this.WebViewContentComponent = WebViewContentComponent;
  1586. };
  1587. LessonMaterialsSectionTitle: React.ComponentType<
  1588. LessonMaterialsSectionTitleProps
  1589. > | null = null;
  1590. /**
  1591. * You can use this hook to customize the component that displays the "Materials" text.
  1592. * @method
  1593. * @param {React.ComponentType<LessonMaterialsSectionTitleProps>} LessonMaterialsSectionTitle
  1594. * @example
  1595. *
  1596. * ...
  1597. *
  1598. * export const applyCustomCode = (externalCodeSetup) => {
  1599. * externalCodeSetup.lessonSingleScreenApi.setLessonMaterialsSectionTitle(
  1600. * ({global, t}) => (
  1601. * <Text
  1602. * style={{
  1603. * ...global.courseRoundBoxTitleAbove,
  1604. * marginBottom: 20
  1605. * }}
  1606. * >
  1607. * {t("lesson:materials")}
  1608. * </Text>
  1609. * )
  1610. * );
  1611. * }
  1612. */
  1613. setLessonMaterialsSectionTitle = (
  1614. LessonMaterialsSectionTitle: React.ComponentType<
  1615. LessonMaterialsSectionTitleProps
  1616. > | null
  1617. ) => {
  1618. this.LessonMaterialsSectionTitle = LessonMaterialsSectionTitle;
  1619. };
  1620. AssignmentsHeader: React.ComponentType<AssignmentsHeaderProps> | null = null;
  1621. /**
  1622. *
  1623. * You can use this hook to customize the assignments title and the component that displays the total approved assignments.
  1624. * @method
  1625. * @param {React.ComponentType<AssignmentsHeaderProps>} AssignmentsHeader
  1626. * @example
  1627. *
  1628. * ...
  1629. *
  1630. * export const applyCustomCode = (externalCodeSetup) => {
  1631. * externalCodeSetup.lessonSingleScreenApi.setAssignmentsHeader(
  1632. * ({global, t, totalAssignments, approved}) => (
  1633. * <View style={{...global.row, flex: 1, marginBottom: 15}}>
  1634. * <Text style={[global.assignmentHeading, {flex: 1}]}>
  1635. * {t("assignment:heading")}
  1636. * </Text>
  1637. * {totalAssignments !== 0 && (
  1638. * <Text style={[global.assignmentDesc]}>
  1639. * {t("assignment:approvedOutOfTotal", {
  1640. * total: totalAssignments,
  1641. * approved
  1642. * })}
  1643. * </Text>
  1644. * )}
  1645. * </View>
  1646. * )
  1647. * );
  1648. * }
  1649. */
  1650. setAssignmentsHeader = (
  1651. AssignmentsHeader: React.ComponentType<AssignmentsHeaderProps> | null
  1652. ) => {
  1653. this.AssignmentsHeader = AssignmentsHeader;
  1654. };
  1655. AssignmentItemIcon: React.ComponentType<
  1656. AssignmentItemIconProps
  1657. > | null = null;
  1658. /**
  1659. * You can use this to change the icon beside the assignment title.
  1660. * For example, you can use this to remove or modify the default icon.
  1661. * @method
  1662. * @param {React.ComponentType<AssignmentItemIconProps>} AssignmentItemIcon
  1663. * @example
  1664. *
  1665. * ...
  1666. *
  1667. * import Icon from "@src/components/Icon";
  1668. * export const applyCustomCode = (externalCodeSetup) => {
  1669. * externalCodeSetup.lessonSingleScreenApi.setAssignmentItemIcon(
  1670. * ({icon, styles}) => <Icon icon={icon} styles={styles} />
  1671. * );
  1672. * }
  1673. */
  1674. setAssignmentItemIcon = (
  1675. AssignmentItemIcon: React.ComponentType<AssignmentItemIconProps> | null
  1676. ) => {
  1677. this.AssignmentItemIcon = AssignmentItemIcon;
  1678. };
  1679. AssignmentItemTitle: React.ComponentType<
  1680. AssignmentItemTitleProps
  1681. > | null = null;
  1682. /**
  1683. * You can use this to customize the AssignmentItemTitle component.
  1684. * For example, you can use this to change the font color or size of the title.
  1685. * @method
  1686. * @param {React.ComponentType<AssignmentItemTitleProps>} AssignmentItemTitle
  1687. * @example
  1688. *
  1689. * ...
  1690. *
  1691. * export const applyCustomCode = (externalCodeSetup) => {
  1692. * externalCodeSetup.lessonSingleScreenApi.setAssignmentItemTitle(({
  1693. * title,
  1694. * style
  1695. * }) => <Text style={style}>{title}</Text>)
  1696. * }
  1697. */
  1698. setAssignmentItemTitle = (
  1699. AssignmentItemTitle: React.ComponentType<AssignmentItemTitleProps> | null
  1700. ) => {
  1701. this.AssignmentItemTitle = AssignmentItemTitle;
  1702. };
  1703. AssignmentItemStatus: React.ComponentType<
  1704. AssignmentItemStatusProps
  1705. > | null = null;
  1706. /**
  1707. * You can use this to customize the AssignmentItemStatus component.
  1708. * @method
  1709. * @param {React.ComponentType<AssignmentItemStatusProps>} AssignmentItemStatus
  1710. * @example
  1711. *
  1712. * ...
  1713. *
  1714. * import {BubbleIcon} from "@src/components/BubbleIcon";
  1715. * export const applyCustomCode = (externalCodeSetup) => {
  1716. * externalCodeSetup.lessonSingleScreenApi.setAssignmentItemStatus(
  1717. * ({global, text, containerStyle, textStyle}) => (
  1718. * <BubbleIcon
  1719. * {...{
  1720. * global,
  1721. * text,
  1722. * containerStyle,
  1723. * textStyle
  1724. * }}
  1725. * />
  1726. * )
  1727. * );
  1728. * }
  1729. */
  1730. setAssignmentItemStatus = (
  1731. AssignmentItemStatus: React.ComponentType<AssignmentItemStatusProps> | null
  1732. ) => {
  1733. this.AssignmentItemStatus = AssignmentItemStatus;
  1734. };
  1735. AssignmentItemComment: React.ComponentType<
  1736. AssignmentItemCommentProps
  1737. > | null = null;
  1738. /**
  1739. * You can use this to customize the AssignmentItemComment button.
  1740. * @method
  1741. * @param {React.ComponentType<AssignmentItemCommentProps>} AssignmentItemComment
  1742. * @example
  1743. *
  1744. * ...
  1745. *
  1746. * import AssignmentCommentButton from "@src/components/Course/Assignment/AssignmentCommentButton";
  1747. * export const applyCustomCode = (externalCodeSetup) => {
  1748. * externalCodeSetup.lessonSingleScreenApi.setAssignmentItemComment(
  1749. * ({
  1750. * pressHandler,
  1751. * tintColor,
  1752. * containerStyle,
  1753. * count,
  1754. * size,
  1755. * global,
  1756. * lightMode
  1757. * }) => (
  1758. * <AssignmentCommentButton
  1759. * {...{
  1760. * pressHandler,
  1761. * tintColor,
  1762. * containerStyle,
  1763. * count,
  1764. * size,
  1765. * global,
  1766. * lightMode
  1767. * }}
  1768. * />
  1769. * )
  1770. * );
  1771. * }
  1772. */
  1773. setAssignmentItemComment = (
  1774. AssignmentItemComment: React.ComponentType<
  1775. AssignmentItemCommentProps
  1776. > | null
  1777. ) => {
  1778. this.AssignmentItemComment = AssignmentItemComment;
  1779. };
  1780. AssignmentItemDownload: React.ComponentType<
  1781. AssignmentItemDownloadProps
  1782. > | null = null;
  1783. /**
  1784. * You can use this to customize the AssignmentItemDownload button.
  1785. * For example, you can change the icon and add a confirmation modal before allowing the user to download the assignment file.
  1786. * @method
  1787. * @param {React.ComponentType<AssignmentItemDownloadProps>} AssignmentItemDownload
  1788. * @example
  1789. *
  1790. * ...
  1791. *
  1792. * import AssignmentDownloadButton from "@src/components/Course/Assignment/AssignmentDownloadButton";
  1793. * export const applyCustomCode = (externalCodeSetup) => {
  1794. * externalCodeSetup.lessonSingleScreenApi.setAssignmentItemDownload(
  1795. * ({pressHandler, tintColor, containerStyle, size, global, lightMode}) => (
  1796. * <AssignmentDownloadButton
  1797. * {...{
  1798. * pressHandler,
  1799. * tintColor,
  1800. * containerStyle,
  1801. * size,
  1802. * global,
  1803. * lightMode
  1804. * }}
  1805. * />
  1806. * )
  1807. * );
  1808. * }
  1809. *
  1810. */
  1811. setAssignmentItemDownload = (
  1812. AssignmentItemDownload: React.ComponentType<
  1813. AssignmentItemDownloadProps
  1814. > | null
  1815. ) => {
  1816. this.AssignmentItemDownload = AssignmentItemDownload;
  1817. };
  1818. AssignmentItemDownloadProgress: React.ComponentType<
  1819. AssignmentItemDownloadProgressProps
  1820. > | null = null;
  1821. /**
  1822. * You can use this to customize the progress indicator which is showed when the assignment item is being downloaded.
  1823. * @method
  1824. * @param {React.ComponentType<AssignmentItemDownloadProgressProps>} AssignmentItemDownloadProgress
  1825. * @example
  1826. *
  1827. * ...
  1828. *
  1829. * import AssignmentDownloadProgress from "@src/components/Course/Assignment/AssignmentDownloadProgress";
  1830. * export const applyCustomCode = (externalCodeSetup) => {
  1831. * externalCodeSetup.lessonSingleScreenApi.setAssignmentItemDownloadProgress(
  1832. * ({progress, unfilledColor, tintColor}) => (
  1833. * <AssignmentDownloadProgress
  1834. * {...{
  1835. * progress,
  1836. * unfilledColor,
  1837. * tintColor
  1838. * }}
  1839. * />
  1840. * )
  1841. * );
  1842. * }
  1843. *
  1844. */
  1845. setAssignmentItemDownloadProgress = (
  1846. AssignmentItemDownloadProgress: React.ComponentType<
  1847. AssignmentItemDownloadProgressProps
  1848. > | null
  1849. ) => {
  1850. this.AssignmentItemDownloadProgress = AssignmentItemDownloadProgress;
  1851. };
  1852. MaterialsComponent: React.ComponentType<
  1853. MaterialsComponentProps
  1854. > | null = null;
  1855. /**
  1856. * You can use this to customize the component that displays the materials of the lesson.
  1857. * @method
  1858. * @param {React.ComponentType<MaterialsComponentProps>} MaterialsComponent
  1859. * @example
  1860. *
  1861. * ...
  1862. *
  1863. * import {
  1864. * Dimensions,
  1865. * } from "react-native";
  1866. * import HTML from "react-native-render-html";
  1867. * import {RenderListPrefix} from "@src/components/Course/CourseMaterials"
  1868. * const ent = require("ent");
  1869. * const DEVICE_WIDTH = Dimensions.get("window").width;
  1870. *
  1871. * export const applyCustomCode = (externalCodeSetup) => {
  1872. *
  1873. * externalCodeSetup.lessonSingleScreenApi.setMaterialsComponent((props) => {
  1874. *
  1875. * const {
  1876. * tagsStyles,
  1877. * materialsStyles,
  1878. * baseFontStyle,
  1879. * materials,
  1880. * onLinkPress,
  1881. * global,
  1882. * colors
  1883. * } = props;
  1884. *
  1885. * return (
  1886. * <HTML
  1887. * tagsStyles={{...tagsStyles, ...materialsStyles}}
  1888. * baseFontStyle={baseFontStyle(15)}
  1889. * html={ent.decode(materials)}
  1890. * imagesMaxWidth={DEVICE_WIDTH - 32}
  1891. * onLinkPress={onLinkPress}
  1892. * listsPrefixesRenderers={{
  1893. * ul: (attrib, children, styles, passProps) => (
  1894. * <RenderListPrefix
  1895. * parent={"ul"}
  1896. * colors={colors}
  1897. * global={global}
  1898. * passProps={passProps}
  1899. * />
  1900. * ),
  1901. * ol: (attrib, children, styles, passProps) => (
  1902. * <RenderListPrefix
  1903. * parent={"ol"}
  1904. * colors={colors}
  1905. * global={global}
  1906. * passProps={passProps}
  1907. * />
  1908. * )
  1909. * }}
  1910. * />
  1911. * )
  1912. * });
  1913. * }
  1914. */
  1915. setMaterialsComponent = (
  1916. MaterialsComponent: React.ComponentType<MaterialsComponentProps> | null
  1917. ) => {
  1918. this.MaterialsComponent = MaterialsComponent;
  1919. };
  1920. }